# IV file viewer import xml.etree.ElementTree as ET import optparse import sys import math parser = optparse.OptionParser() parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') parser.add_option('-G', '--group', dest='group', action='append', help='Only compute for this group (may be specified multiple times)') parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') parser.add_option('-M', '--notes-stream', dest='notes_stream', action='store_true', help='Show notes per stream') parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') parser.add_option('--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') parser.add_option('--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') parser.add_option('--vel-hist', dest='vel_hist', action='store_true', help='Show a histogram distribution of velocities') parser.add_option('--vel-hist-tracks', dest='vel_hist_tracks', action='store_true', help='Show a histogram distributions of velocities per track') parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') parser.add_option('-H', '--height', dest='height', type='int', help='Height of histograms') parser.add_option('-C', '--no-color', dest='no_color', action='store_true', help='Don\'t use ANSI color escapes') parser.add_option('-x', '--aux', dest='aux', action='store_true', help='Show information about the auxiliary streams') parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') parser.add_option('-t', '--total', dest='total', action='store_true', help='Make cross-file totals') parser.set_defaults(height=20, group=[]) options, args = parser.parse_args() if not any(( options.number, options.groups, options.notes, options.notes_stream, options.histogram, options.vel_hist, options.duration, options.duty_cycle, options.aux, options.meta, options.histogram_tracks, options.vel_hist_tracks, )): print 'No computations specified! Assuming you meant --almost-all...' options.almost_all = True if options.almost_all or options.all: options.number = True options.groups = True options.notes = True options.notes_stream = True options.histogram = True options.vel_hist = True options.duration = True options.duty_cycle = True if options.all: options.aux = True options.meta = True options.histogram_tracks= True options.vel_hist_tracks = True if options.no_color: class COL: NONE='' RED='' GREEN='' BLUE='' YELLOW='' MAGENTA='' CYAN='' else: class COL: NONE='\x1b[0m' RED='\x1b[31m' GREEN='\x1b[32m' BLUE='\x1b[34m' YELLOW='\x1b[33m' MAGENTA='\x1b[35m' CYAN='\x1b[36m' def show_hist(values, height=None): if not values: print '{empty histogram}' return if height is None: height = options.height xs, ys = values.keys(), values.values() minx, maxx = min(xs), max(xs) miny, maxy = min(ys), max(ys) xv = range(int(math.floor(minx)), int(math.ceil(maxx + 1))) incs = max((maxy - miny) / height, 1) print COL.CYAN + '\t --' + '-' * len(xv) + COL.NONE for ub in range(maxy + incs, miny, -incs): print '{}{}\t | {}{}{}'.format(COL.CYAN, ub, COL.YELLOW, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv]), COL.NONE) print COL.CYAN + '\t |-' + '-' * len(xv) + COL.NONE xvs = map(str, xv) for i in range(max(map(len, xvs))): print COL.CYAN + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) + COL.NONE print xcs = map(str, [values.get(x, 0) for x in xv]) for i in range(max(map(len, xcs))): print COL.YELLOW + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xcs]) + COL.NONE print if options.total: tot_note_cnt = 0 max_note_cnt = 0 tot_pitches = {} tot_velocities = {} tot_dur = 0 max_dur = 0 tot_streams = 0 max_streams = 0 tot_notestreams = 0 max_notestreams = 0 tot_groups = {} for fname in args: print print 'File :', fname try: iv = ET.parse(fname).getroot() except Exception: import traceback traceback.print_exc() print 'Bad file :', fname, ', skipping...' continue print '\t' if options.meta: print 'Metatrack:', meta = iv.find('./meta') if len(meta): print 'exists' print '\tBPM track:', bpms = meta.find('./bpms') if len(bpms): print 'exists' for elem in bpms.iterfind('./bpm'): print '\t\tAt ticks {}, time {}: {} bpm'.format(elem.get('ticks'), elem.get('time'), elem.get('bpm')) if not (options.number or options.groups or options.notes or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle or options.aux): continue streams = iv.findall('./streams/stream') notestreams = [s for s in streams if s.get('type') == 'ns'] auxstreams = [s for s in streams if s.get('type') == 'aux'] if options.group: print 'NOTE: Restricting results to groups', options.group, 'as requested' notestreams = [ns for ns in notestreams if ns.get('group', '') in options.group] if options.number: print 'Stream count:' print '\tNotestreams:', len(notestreams) print '\tTotal:', len(streams) if options.total: tot_streams += len(streams) max_streams = max(max_streams, len(streams)) tot_notestreams += len(notestreams) max_notestreams = max(max_notestreams, len(notestreams)) if not (options.groups or options.notes or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle or options.aux): continue if options.groups: groups = {} for s in notestreams: group = s.get('group', '') groups[group] = groups.get(group, 0) + 1 if options.total: tot_groups[group] = tot_groups.get(group, 0) + 1 print 'Groups:' for name, cnt in groups.iteritems(): print '\t{} ({} streams)'.format(name, cnt) if options.aux: import midi fr = midi.FileReader() fr.RunningStatus = None # XXX Hack print 'Aux stream data:' for aidx, astream in enumerate(auxstreams): evs = astream.findall('ev') failed = 0 print '\tFrom stream {}, {} events:'.format(aidx, len(evs)) for ev in evs: try: data = eval(ev.get('data')) mev = fr.parse_midi_event(iter(data)) except AssertionError: failed += 1 else: print '\t\tAt time {}: {}'.format(ev.get('time'), mev) print '\t\t(...and {} others which failed to parse)'.format(failed) if not (options.notes or options.notes_stream or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle): continue if options.notes: note_cnt = 0 if options.notes_stream: notes_stream = [0] * len(notestreams) if options.histogram: pitches = {} if options.histogram_tracks: pitch_tracks = [{} for i in notestreams] if options.vel_hist: velocities = {} if options.vel_hist_tracks: velocities_tracks = [{} for i in notestreams] if options.duration or options.duty_cycle: max_dur = 0 if options.duty_cycle: cum_dur = [0.0] * len(notestreams) for sidx, stream in enumerate(notestreams): notes = stream.findall('note') for note in notes: pitch = float(note.get('pitch')) ampl = float(note.get('ampl', float(note.get('vel', 127.0)) / 127.0)) time = float(note.get('time')) dur = float(note.get('dur')) if options.notes: note_cnt += 1 if options.total: tot_note_cnt += 1 if options.notes_stream: notes_stream[sidx] += 1 if options.histogram: pitches[pitch] = pitches.get(pitch, 0) + 1 if options.total: tot_pitches[pitch] = tot_pitches.get(pitch, 0) + 1 if options.histogram_tracks: pitch_tracks[sidx][pitch] = pitch_tracks[sidx].get(pitch, 0) + 1 if options.vel_hist: velocities[ampl] = velocities.get(ampl, 0) + 1 if options.total: tot_velocities[ampl] = tot_velocities.get(ampl, 0) + 1 if options.vel_hist_tracks: velocities_tracks[sidx][ampl] = velocities_tracks[sidx].get(ampl, 0) + 1 if (options.duration or options.duty_cycle) and time + dur > max_dur: max_dur = time + dur if options.duty_cycle: cum_dur[sidx] += dur if options.notes and options.total: max_note_cnt = max(max_note_cnt, note_cnt) if options.histogram_tracks: for sidx, hist in enumerate(pitch_tracks): print 'Stream {} (group {}) pitch histogram:'.format(sidx, notestreams[sidx].get('group', '')) show_hist(hist) if options.vel_hist_tracks: for sidx, hist in enumerate(velocities_tracks): print 'Stream {} (group {}) velocity histogram:'.format(sidx, notestreams[sidx].get('group', '')) show_hist(hist) if options.notes_stream: for sidx, value in enumerate(notes_stream): print 'Stream {} (group {}) note count: {}'.format(sidx, notestreams[sidx].get('group', ''), value) if options.duty_cycle: for sidx, value in enumerate(cum_dur): print 'Stream {} (group {}) duty cycle: {}'.format(sidx, notestreams[sidx].get('group', ''), value / max_dur) if options.notes: print 'Total notes: {}'.format(note_cnt) if options.histogram: print 'Pitch histogram:' show_hist(pitches) if options.vel_hist: print 'Velocity histogram:' show_hist(velocities) if options.duration: print 'Playing duration: {}'.format(max_dur) if options.total: print 'Totals:' if options.number: print '\tTotal streams:', tot_streams print '\tMax streams:', max_streams print '\tTotal notestreams:', tot_notestreams print '\tMax notestreams:', max_notestreams print if options.notes: print '\tTotal notes:', tot_note_cnt print '\tMax notes:', max_note_cnt print if options.groups: print '\tGroups:' for grp, cnt in tot_groups.iteritems(): print '\t\t', grp, ':', cnt print if options.histogram: print 'Overall pitch histogram:' show_hist(tot_pitches) if options.vel_hist: print 'Overall velocity histogram:' show_hist(tot_velocities)