forked from brannondorsey/midi-rnn
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sample.py
executable file
·122 lines (106 loc) · 5.38 KB
/
sample.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
#!/usr/bin/env python
import argparse, os, pdb
import pretty_midi
import train
import utils
def parse_args():
parser = argparse.ArgumentParser(
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument('--experiment_dir', type=str,
default='experiments/default',
help='directory to load saved model from. ' \
'If omitted, it will use the most recent directory from ' \
'experiments/.')
parser.add_argument('--save_dir', type=str,
help='directory to save generated files to. Directory will be ' \
'created if it doesn\'t already exist. If not specified, ' \
'files will be saved to generated/ inside --experiment_dir.')
parser.add_argument('--midi_instrument', default='Acoustic Grand Piano',
help='MIDI instrument name (or number) to use for the ' \
'generated files. See https://www.midi.org/specifications/item/'\
'gm-level-1-sound-set for a full list of instrument names.')
parser.add_argument('--num_files', type=int, default=10,
help='number of midi files to sample.')
parser.add_argument('--file_length', type=int, default=1000,
help='Length of each file, measured in 16th notes.')
parser.add_argument('--prime_file', type=str,
help='prime generated files from midi file. If not specified ' \
'random windows from the validation dataset will be used for ' \
'for seeding.')
parser.add_argument('--data_dir', type=str, default='data/midi',
help='data directory containing .mid files to use for' \
'seeding/priming. Required if --prime_file is not specified')
return parser.parse_args()
def get_experiment_dir(experiment_dir):
if experiment_dir == 'experiments/default':
dirs_ = [os.path.join('experiments', d) for d in os.listdir('experiments') \
if os.path.isdir(os.path.join('experiments', d))]
experiment_dir = max(dirs_, key=os.path.getmtime)
if not os.path.exists(os.path.join(experiment_dir, 'model.json')):
utils.log('Error: {} does not exist. ' \
'Are you sure that {} is a valid experiment?' \
'Exiting.'.format(os.path.join(args.experiment_dir), 'model.json',
experiment_dir), True)
exit(1)
return experiment_dir
def main():
args = parse_args()
args.verbose = True
# prime file validation
if args.prime_file and not os.path.exists(args.prime_file):
utils.log('Error: prime file {} does not exist. Exiting.'.format(args.prime_file),
True)
exit(1)
else:
if not os.path.isdir(args.data_dir):
utils.log('Error: data dir {} does not exist. Exiting.'.format(args.prime_file),
True)
exit(1)
midi_files = [ args.prime_file ] if args.prime_file else \
[ os.path.join(args.data_dir, f) for f in os.listdir(args.data_dir) \
if '.mid' in f or '.midi' in f ]
experiment_dir = get_experiment_dir(args.experiment_dir)
utils.log('Using {} as --experiment_dir'.format(experiment_dir), args.verbose)
if not args.save_dir:
args.save_dir = os.path.join(experiment_dir, 'generated')
if not os.path.isdir(args.save_dir):
os.makedirs(args.save_dir)
utils.log('Created directory {}'.format(args.save_dir), args.verbose)
model, epoch = train.get_model(args, experiment_dir=experiment_dir)
utils.log('Model loaded from {}'.format(os.path.join(experiment_dir, 'model.json')),
args.verbose)
window_size = model.layers[0].get_input_shape_at(0)[1]
seed_generator = utils.get_data_generator(midi_files,
window_size=window_size,
batch_size=32,
num_threads=1,
max_files_in_ram=10)
# validate midi instrument name
try:
# try and parse the instrument name as an int
instrument_num = int(args.midi_instrument)
if not (instrument_num >= 0 and instrument_num <=127):
utils.log('Error: {} is not a supported instrument. Number values must be ' \
'be 0-127. Exiting'.format(args.midi_instrument), True)
exit(1)
args.midi_instrument = pretty_midi.program_to_instrument_name(instrument_num)
except ValueError as err:
# if the instrument name is a string
try:
# validate that it can be converted to a program number
_ = pretty_midi.instrument_name_to_program(args.midi_instrument)
except ValueError as er:
utils.log('Error: {} is not a valid General MIDI instrument. Exiting.'\
.format(args.midi_instrument), True)
exit(1)
# generate 10 tracks using random seeds
utils.log('Loading seed files...', args.verbose)
X, y = next(seed_generator)
generated = utils.generate(model, X, window_size,
args.file_length, args.num_files, args.midi_instrument)
for i, midi in enumerate(generated):
file = os.path.join(args.save_dir, '{}.mid'.format(i + 1))
midi.write(file.format(i + 1))
utils.log('wrote midi file to {}'.format(file), True)
if __name__ == '__main__':
main()