def find_atoms(size, datastream): """ This function is a generator that will yield either "stco" or "co64" when either atom is found. datastream can be assumed to be 8 bytes into the stco or co64 atom when the value is yielded. It is assumed that datastream will be at the end of the atom after the value has been yielded and processed. size is the number of bytes to the end of the atom in the datastream. """ stop = datastream.tell() + size while datastream.tell() < stop: try: atom_size, atom_type = read_atom(datastream) except: log.exception("Error reading next atom!") raise FastStartException() if atom_type in ["trak", "mdia", "minf", "stbl"]: # Known ancestor atom of stco or co64, search within it! for atype in find_atoms(atom_size - 8, datastream): yield atype elif atom_type in ["stco", "co64"]: yield atom_type else: # Ignore this atom, seek to the end of it. datastream.seek(atom_size - 8, os.SEEK_CUR)
def _find_atoms_ex(parent_atom, datastream): """ Yield either "stco" or "co64" Atoms from datastream. datastream will be 8 bytes into the stco or co64 atom when the value is yielded. It is assumed that datastream will be at the end of the atom after the value has been yielded and processed. parent_atom is the parent atom, a 'moov' or other ancestor of CO atoms in the datastream. """ stop = parent_atom.position + parent_atom.size while datastream.tell() < stop: try: atom = _read_atom_ex(datastream) except: log.exception("Error reading next atom!") raise FastStartException() if atom.name in ["trak", "mdia", "minf", "stbl"]: # Known ancestor atom of stco or co64, search within it! for res in _find_atoms_ex(atom, datastream): yield res elif atom.name in ["stco", "co64"]: yield atom else: # Ignore this atom, seek to the end of it. datastream.seek(atom.position + atom.size)
def _ensure_valid_index(index): """ Ensure the minimum viable atoms are present in the index. Raise FastStartException if not. """ top_level_atoms = set([item.name for item in index]) for key in ["moov", "mdat"]: if key not in top_level_atoms: log.error("%s atom not found, is this a valid MOV/MP4 file?" % key) raise FastStartException()
def get_index(datastream): """ Return an index of top level atoms, their absolute byte-position in the file and their size in a list: index = [ ("ftyp", 0, 24), ("moov", 25, 2658), ("free", 2683, 8), ... ] The tuple elements will be in the order that they appear in the file. """ index = [] log.debug("Getting index of top level atoms...") # Read atoms until we catch an error while (datastream): try: skip = 8 atom_size, atom_type = read_atom(datastream) if atom_size == 1: atom_size = struct.unpack(">Q", datastream.read(8))[0] skip = 16 log.debug("%s: %s" % (atom_type, atom_size)) except: break index.append((atom_type, datastream.tell() - skip, atom_size)) if atom_size == 0: if atom_type == "mdat": # Some files may end in mdat with no size set, which generally # means to seek to the end of the file. We can just stop indexing # as no more entries will be found! break else: # Weird, but just continue to try to find more atoms atom_size = skip datastream.seek(atom_size - skip, os.SEEK_CUR) # Make sure the atoms we need exist top_level_atoms = set([item[0] for item in index]) for key in ["moov", "mdat"]: if key not in top_level_atoms: log.error("%s atom not found, is this a valid MOV/MP4 file?" % key) raise FastStartException() return index
def process(infilename, outfilename, limit=0): """ Convert a Quicktime/MP4 file for streaming by moving the metadata to the front of the file. This method writes a new file. If limit is set to something other than zero it will be used as the number of bytes to write of the atoms following the moov atom. This is very useful to create a small sample of a file with full headers, which can then be used in bug reports and such. """ datastream = open(infilename, "rb") # Get the top level atom index index = get_index(datastream) mdat_pos = 999999 free_size = 0 # Make sure moov occurs AFTER mdat, otherwise no need to run! for atom, pos, size in index: # The atoms are guaranteed to exist from get_index above! if atom == "moov": moov_pos = pos moov_size = size elif atom == "mdat": mdat_pos = pos elif atom == "free" and pos < mdat_pos: # This free atom is before the mdat! free_size += size log.info("Removing free atom at %d (%d bytes)" % (pos, size)) elif atom == "\x00\x00\x00\x00" and pos < mdat_pos: # This is some strange zero atom with incorrect size free_size += 8 log.info("Removing strange zero atom at %s (8 bytes)" % pos) # Offset to shift positions offset = moov_size - free_size if moov_pos < mdat_pos: # moov appears to be in the proper place, don't shift by moov size offset -= moov_size if not free_size: # No free atoms and moov is correct, we are done! log.error("This file appears to already be setup for streaming!") raise FastStartException() # Read and fix moov datastream.seek(moov_pos) moov = StringIO(datastream.read(moov_size)) # Ignore moov identifier and size, start reading children moov.seek(8) for atom_type in find_atoms(moov_size - 8, moov): # Read either 32-bit or 64-bit offsets ctype, csize = atom_type == "stco" and ("L", 4) or ("Q", 8) # Get number of entries version, entry_count = struct.unpack(">2L", moov.read(8)) log.info("Patching %s with %d entries" % (atom_type, entry_count)) # Read entries entries = struct.unpack(">" + ctype * entry_count, moov.read(csize * entry_count)) # Patch and write entries moov.seek(-csize * entry_count, os.SEEK_CUR) moov.write( struct.pack(">" + ctype * entry_count, *[entry + offset for entry in entries])) log.info("Writing output...") outfile = open(outfilename, "wb") # Write ftype for atom, pos, size in index: if atom == "ftyp": datastream.seek(pos) outfile.write(datastream.read(size)) # Write moov moov.seek(0) outfile.write(moov.read()) # Write the rest written = 0 atoms = [item for item in index if item[0] not in ["ftyp", "moov", "free"]] for atom, pos, size in atoms: datastream.seek(pos) # Write in chunks to not use too much memory for x in range(size / CHUNK_SIZE): outfile.write(datastream.read(CHUNK_SIZE)) written += CHUNK_SIZE if limit and written >= limit: # A limit was set and we've just passed it, stop writing! break if size % CHUNK_SIZE: outfile.write(datastream.read(size % CHUNK_SIZE)) written += (size % CHUNK_SIZE) if limit and written >= limit: # A limit was set and we've just passed it, stop writing! break
def process(infilename, outfilename, limit=float('inf')): """ Convert a Quicktime/MP4 file for streaming by moving the metadata to the front of the file. This method writes a new file. If limit is set to something other than zero it will be used as the number of bytes to write of the atoms following the moov atom. This is very useful to create a small sample of a file with full headers, which can then be used in bug reports and such. """ datastream = open(infilename, "rb") # Get the top level atom index index = get_index(datastream) mdat_pos = 999999 free_size = 0 # Make sure moov occurs AFTER mdat, otherwise no need to run! for atom in index: # The atoms are guaranteed to exist from get_index above! if atom.name == "moov": moov_atom = atom moov_pos = atom.position elif atom.name == "mdat": mdat_pos = atom.position elif atom.name == "free" and atom.position < mdat_pos: # This free atom is before the mdat! free_size += atom.size log.info("Removing free atom at %d (%d bytes)" % (atom.position, atom.size)) elif atom.name == "\x00\x00\x00\x00" and atom.position < mdat_pos: # This is some strange zero atom with incorrect size free_size += 8 log.info("Removing strange zero atom at %s (8 bytes)" % atom.position) # Offset to shift positions offset = moov_atom.size - free_size if moov_pos < mdat_pos: # moov appears to be in the proper place, don't shift by moov size offset -= moov_atom.size if not free_size: # No free atoms and moov is correct, we are done! log.error("This file appears to already be setup for streaming!") raise FastStartException() # Read and fix moov moov = _patch_moov(datastream, moov_atom, offset) log.info("Writing output...") outfile = open(outfilename, "wb") # Write ftype for atom in index: if atom.name == "ftyp": log.debug("Writing ftyp... (%d bytes)" % atom.size) datastream.seek(atom.position) outfile.write(datastream.read(atom.size)) # Write moov bytes = moov.getvalue() log.debug("Writing moov... (%d bytes)" % len(bytes)) outfile.write(bytes) # Write the rest atoms = [ item for item in index if item.name not in ["ftyp", "moov", "free"] ] for atom in atoms: log.debug("Writing %s... (%d bytes)" % (atom.name, atom.size)) datastream.seek(atom.position) # for compatability, allow '0' to mean no limit cur_limit = limit or float('inf') cur_limit = min(cur_limit, atom.size) for chunk in get_chunks(datastream, CHUNK_SIZE, cur_limit): outfile.write(chunk) # Close and set permissions outfile.close() try: shutil.copymode(infilename, outfilename) except: log.warn("Could not copy file permissions!")