- Serato track tag parsing and modification, including cue points, track color, beatgrid, waveform, autogain, etc.
- Serato library database parsing and modification
- Serato crate parsing and modification
- Serato smart crate parsing and rule modification
- Dynamic beatgrid analysis that can be saved to a track's beatgrid.
- Serato USB export that beats Serato's sync by putting all files in 1 folder (without duplicates) and only copying changed files, unlike Serato's sync which takes forever and creates many duplicate file locations
Currently designed for Python 3.12+. If you would like backwards compatibility with an older version, please reach out!
USB:
- Export crates (including Smart Crates) to USB. Beats Serato's sync by keeping all files in 1 folder and by comparing files for changes (instead of always copying, like Serato does)
Tracks:
- Change hot cue's text (i.e. change to all caps; change "c" to "CHORUS", etc.)
- Set a hot cue's text based on its color (i.e. if RED, set to "EXIT")
- Set a track's color (i.e. if has hot cues, change to BLUE)
- Set a piece of metadata due to a track's color (i.e. if track is green, set "grouping" to "TAGGED") (this is useful since can't create smart crates by track color in Serato)
- Analyze a track's Dynamic Beatgrid and save it to the beatgrid Serato tag.
- Snap cue points to the nearest beat with a configurable tolerance (e.g. 1/16, 1/8, 1/4 of a beat).
Database:
- Rename a file while changing its location in the database as well, so that it doesn't go missing.
- After changing a track's metadata, modify database values for a specific track so that you don't have to "Reload Id3 Tags" in Serato, the change appears in Serato instantly.
Crate:
- Read a crate's tracks
- Remove a track from a crate
- Add a track to a crate
- List available crates
Smart Crate:
- Read a smart crate's tracks and rules
- Modify a smart crate's rules
- List available smart crates
Code examples are below.
pip install serato-toolsThe following deps are optional, but must be installed based on what features you want to use:
- For getting/modifying track metadata including tags, cue points, beagrid, and waveform, must install
pip install mutagen - For viewing a track's waveform, must install
pip install pillow - For beatgrid analysis, must install
pip install numpyandpip install librosa
Rekordbox has Dynamic Beatgrid Analysis but Serato doesn't. This analyzes a non-consistent BPM across a track, such as a track with live drums, and snaps a beatgrid marker to each beat.
NOTE: This feature has only been tested on a couple of tracks. Recommend reviewing the resulting beatgrid in Serato— some grid markers may require adjustment. It seems to be working pretty great though!
>>> serato_analyze_beatgrid "Music/Dubstep/Mind Splitter - YAPPIN'.mp3"NOTE: replaces existing crates on flash drive! (but does not delete existing track files) (TODO: ability to merge with existing)
>>> serato_usb_export --drive E --crate_matcher *house* *techno* --root_crate="Dave USB"This renames the file path, and also changes the path in the database to point to the new filename, so that the renamed file is not missing in the library.
NOTE: Recommended to make a backup of the database file elsewhere, before modifying via this package, in case an unforseen bug appears.
from serato_tools.database_v2 import DatabaseV2
db = DatabaseV2()
db.rename_track_file(src="5udo - One - Original Mix.mp3", dest="5udo - One.mp3")NOTE: Recommended to make a backup of the database file elsewhere, before modifying via this package, in case an unforseen bug appears.
from serato_tools.database_v2 import DatabaseV2
now = int(time.time())
def modify_uadd(filename: str, prev_val: Any):
print(f'Serato library change - Changed "date added" to today: {filename}')
return now
def modify_tadd(filename: str, prev_val: Any):
return str(now)
def remove_group(filename: str, prev_val: Any):
return " "
db = DatabaseV2()
# a list of field keys can be found in serato_tools.database_v2
db.modify_file(
rules=[
{"field": DatabaseV2.Fields.DATE_ADDED_U, "files": files_set_date, "func": modify_uadd},
{"field": DatabaseV2.Fields.DATE_ADDED_T, "files": files_set_date, "func": modify_tadd},
{"field": DatabaseV2.Fields.GROUPING, "func": remove_group}, # all files
]
)from serato_tools.track_cues_v2 import TrackCuesV2, TRACK_COLORS
tags = TrackCuesV2(file)
tags.set_track_color('/Users/Username/Music/Dubstep/Raaket - ILL.mp3',
TRACK_COLORS["purple"],
delete_tags_v1=True
# Must delete delete_tags_v1 in order for track color change to appear in Serato (since we never change tags_v1 along with it (TODO)). Not sure what tags_v1 is even for, probably older versions of Serato. Have found no issues with deleting this, but use with caution if running an older version of Serato.
)
tags.save()After you have a good beatgrid (including dynamic beatgrids), you can snap cue positions so they land exactly on beats, within a configurable tolerance:
serato_snap_cues_v2 "/Users/Username/Music/Dubstep/Some Track.mp3" --snap_cues --snap_tolerance 1/16--snap_cuesenables snapping for all cue entries in theSerato Markers2tag.--snap_toleranceis a fraction of a beat (e.g.1/16,1/8,1/4) that a cue point can come within a beat in order to "snap" to it.
Only cue points within the tolerance window of some beat are moved; others are left unchanged.
import dataclasses
from mutagen.id3._frames import TIT1
from serato_tools.track_cues_v2 import TrackCuesV2, CUE_COLORS, TRACK_COLORS
def track_rule(track: TrackCuesV2.TrackCuesInfo) -> TrackCuesV2.TrackCuesInfo | None:
# Set ID3 grouping field from track color
if track.color is not None:
track_color = track.color.color
if track_color == TRACK_COLORS["limegreen3"]:
tagfile.tags.setall("TIT1", [TIT1(text="TAGGED")])
elif track_color in [TRACK_COLORS["white"], TRACK_COLORS["grey"], TRACK_COLORS["black"]]:
tagfile.tags.setall("TIT1", [TIT1(text="UNTAGGED")])
# Normalize cue colors and names
new_cues = []
for c in track.cues:
# Make "close to red", red.
cue_color = CUE_COLORS["red"] if c.color in [CUE_COLORS["pinkred"], CUE_COLORS["magenta"]] else c.color
# Make all cuenames all-caps
cue_name = c.name.strip().upper()
new_cues.append(dataclasses.replace(c, color=cue_color, name=cue_name))
new_track = dataclasses.replace(track, cues=new_cues)
return new_track if new_track != track else None
tags = TrackCuesV2(file)
tags.modify_entries(track_rule, delete_tags_v1=True)
tags.save()from serato_tools.crate import Crate
print('\n'.join(Crate.get_crate_files()))
from serato_tools.smart_crate import SmartCrate
print('\n'.join(SmartCrate.get_crate_files()))via command line
one crate
>>> serato_smartcrate '/Users/Username/Music/_Serato_/SmartCrates/Dubstep.scrate' --set_rules --grouping UNTAGGEDall crates
>>> serato_smartcrate --all --set_rules --grouping UNTAGGEDvia code
from serato_tools.smart_crate import SmartCrate
crate = SmartCrate('/Users/Username/Music/_Serato_/SmartCrates/Dubstep.scrate')
def modify_rule(rule: SmartCrate.Rule):
if rule.field != SmartCrate.RULE_FIELD["grouping"]:
return rule
rule.set_value(SmartCrate.Fields.RULE_VALUE_TEXT, "UNTAGGED")
return rule
crate.modify_rules(modify_rule)
crate.save()from serato_tools.crate import Crate
crate = Crate('/Users/Username/Music/_Serato_/Subcrates/Dubstep.crate')
print(crate)
# OUTPUT:
#
# Crate containing 81 tracks:
# Music/Dubstep/Saka - backitup.mp3
# Music/Dubstep/Mind Splitter - YAPPIN'.mp3
# Music/Dubstep/Flozone - DO IT.mp3
# Music/Dubstep/Evalution - Throw It Back.mp3
# ...
crate.print()
# OUTPUT:
#
# [ ('vrsn', 1.0/Serato ScratchLive Crate),
# ('osrt', [('brev', b'\x00')]),
# ('ovct', [('tvcn', 'key'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'artist'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'song'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'bpm'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'playCount'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'length'), ('tvcw', '0')]),
# ('ovct', [('tvcn', 'added'), ('tvcw', '0')]),
# ( 'otrk',
# [ ( 'ptrk',
# 'Music/Dubstep/Flozone - Candy Paint')]),
# ( 'otrk',
# [ ( 'ptrk',
# 'Music/Dubstep/Mind Splitter - LISTEN TO ME')]),
# ('otrk', [('ptrk', 'Music/Dubstep/Flozone - DO IT')]),
# ...
# Example: Add a track to the crate and save it as a new crate
crate.add_track('/Users/Username/Music/Dubstep/Chozen - I Wanna Dance.mp3')
crate.save_to_file('/Users/Username/Music/Dubstep/New Crate.crate')from serato_tools.smart_crate import SmartCrate
s_crate = Crate('/Users/Username/Music/_Serato_/SmartCrates/Dubstep.scrate')
print(s_crate)Original writeup on Serato GEOB tag discoveries: blog post
| GEOB Tag | Research Progress | Contents | Script File |
|---|---|---|---|
Serato Analysis |
Done | Serato version information | |
Serato Autotags |
Done | BPM and Gain values | track_autotags.py |
Serato BeatGrid |
Mostly done | Beatgrid Markers | track_beatgrid.py |
Serato Markers2 |
Mostly done | Hotcues, Saved Loops, etc. (The main one used in newer versions of Serato) |
track_cues_v2.py |
Serato Markers_ |
Mostly done | Hotcues, Saved Loops, etc. (Old, not used in newer versions of Serato) |
track_cues_v1.py |
Serato Offsets_ |
Not started | ??? | |
Serato Overview |
Done | Waveform data | track_waveform.py |
The different file/tag formats that Serato uses to store the information are documented in docs/fileformats.md, a script to dump the tag data can be found at track_tagdump.py.
- Serato track file tag parsing and modification from https://github.com/Holzhaus/serato-tags , which appears to be no longer maintained
- Serato crate parsing and modification from https://github.com/sharst/seratopy
- Dynamic beatgrid analysis from https://github.com/heyitsmass/audio
If you want a new feature, or have a bug fix, please put in a PR!