TonieToolbox 0.2.1__py3-none-any.whl → 0.2.3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -72,7 +72,8 @@ def get_header_info(in_file):
72
72
 
73
73
  Returns:
74
74
  tuple: Header size, Tonie header object, file size, audio size, SHA1 sum,
75
- Opus header found flag, Opus version, channel count, sample rate, bitstream serial number
75
+ Opus header found flag, Opus version, channel count, sample rate, bitstream serial number,
76
+ Opus comments dictionary
76
77
 
77
78
  Raises:
78
79
  RuntimeError: If OGG pages cannot be found
@@ -112,17 +113,64 @@ def get_header_info(in_file):
112
113
  logger.debug("Opus header found: %s, Version: %d, Channels: %d, Sample rate: %d Hz, Serial: %d",
113
114
  opus_head_found, opus_version, channel_count, sample_rate, bitstream_serial_no)
114
115
 
116
+ # Read and parse Opus comments from the second page
117
+ opus_comments = {}
115
118
  found = OggPage.seek_to_page_header(in_file)
116
119
  if not found:
117
120
  logger.error("Second OGG page not found")
118
121
  raise RuntimeError("Second ogg page not found")
119
122
 
120
- OggPage(in_file)
123
+ second_page = OggPage(in_file)
121
124
  logger.debug("Read second OGG page")
125
+
126
+ try:
127
+ # Combine all segments data for the second page
128
+ comment_data = bytearray()
129
+ for segment in second_page.segments:
130
+ comment_data.extend(segment.data)
131
+
132
+ if comment_data.startswith(b"OpusTags"):
133
+ pos = 8 # Skip "OpusTags"
134
+ # Extract vendor string
135
+ if pos + 4 <= len(comment_data):
136
+ vendor_length = struct.unpack("<I", comment_data[pos:pos+4])[0]
137
+ pos += 4
138
+ if pos + vendor_length <= len(comment_data):
139
+ vendor = comment_data[pos:pos+vendor_length].decode('utf-8', errors='replace')
140
+ opus_comments["vendor"] = vendor
141
+ pos += vendor_length
142
+
143
+ # Extract comments count
144
+ if pos + 4 <= len(comment_data):
145
+ comments_count = struct.unpack("<I", comment_data[pos:pos+4])[0]
146
+ pos += 4
147
+
148
+ # Extract individual comments
149
+ for i in range(comments_count):
150
+ if pos + 4 <= len(comment_data):
151
+ comment_length = struct.unpack("<I", comment_data[pos:pos+4])[0]
152
+ pos += 4
153
+ if pos + comment_length <= len(comment_data):
154
+ comment = comment_data[pos:pos+comment_length].decode('utf-8', errors='replace')
155
+ pos += comment_length
156
+
157
+ # Split comment into key/value if possible
158
+ if "=" in comment:
159
+ key, value = comment.split("=", 1)
160
+ opus_comments[key] = value
161
+ else:
162
+ opus_comments[f"comment_{i}"] = comment
163
+ else:
164
+ break
165
+ else:
166
+ break
167
+ except Exception as e:
168
+ logger.error("Failed to parse Opus comments: %s", str(e))
122
169
 
123
170
  return (
124
171
  header_size, tonie_header, file_size, audio_size, sha1sum,
125
- opus_head_found, opus_version, channel_count, sample_rate, bitstream_serial_no
172
+ opus_head_found, opus_version, channel_count, sample_rate, bitstream_serial_no,
173
+ opus_comments
126
174
  )
127
175
 
128
176
 
@@ -204,7 +252,7 @@ def check_tonie_file(filename):
204
252
 
205
253
  with open(filename, "rb") as in_file:
206
254
  header_size, tonie_header, file_size, audio_size, sha1, opus_head_found, \
207
- opus_version, channel_count, sample_rate, bitstream_serial_no = get_header_info(in_file)
255
+ opus_version, channel_count, sample_rate, bitstream_serial_no, opus_comments = get_header_info(in_file)
208
256
 
209
257
  page_count, alignment_okay, page_size_okay, total_time, \
210
258
  chapters = get_audio_info(in_file, sample_rate, tonie_header, header_size)
@@ -254,6 +302,20 @@ def check_tonie_file(filename):
254
302
  print("")
255
303
  print("[{}] File is {}valid".format("OK" if all_ok else "NOT OK", "" if all_ok else "NOT "))
256
304
  print("")
305
+
306
+ # Display Opus comments if available
307
+ if opus_comments:
308
+ print("[ii] Opus Comments:")
309
+ if "vendor" in opus_comments:
310
+ print(" Vendor: {}".format(opus_comments["vendor"]))
311
+ # Remove vendor from dict to avoid showing it twice
312
+ vendor = opus_comments.pop("vendor")
313
+
314
+ # Sort remaining comments for consistent display
315
+ for key in sorted(opus_comments.keys()):
316
+ print(" {}: {}".format(key, opus_comments[key]))
317
+ print("")
318
+
257
319
  print("[ii] Total runtime: {}".format(granule_to_time_string(total_time)))
258
320
  print("[ii] {} Tracks:".format(len(chapters)))
259
321
  for i in range(0, len(chapters)):
@@ -19,6 +19,33 @@ from .logger import get_logger
19
19
  logger = get_logger('tonie_file')
20
20
 
21
21
 
22
+ def toniefile_comment_add(buffer, length, comment_str):
23
+ """
24
+ Add a comment string to an Opus comment packet buffer.
25
+
26
+ Args:
27
+ buffer: Bytearray buffer to add comment to
28
+ length: Current position in the buffer
29
+ comment_str: Comment string to add
30
+
31
+ Returns:
32
+ int: New position in the buffer after adding comment
33
+ """
34
+ logger.debug("Adding comment: %s", comment_str)
35
+
36
+ # Add 4-byte length prefix
37
+ str_length = len(comment_str)
38
+ buffer[length:length+4] = struct.pack("<I", str_length)
39
+ length += 4
40
+
41
+ # Add the actual string
42
+ buffer[length:length+str_length] = comment_str.encode('utf-8')
43
+ length += str_length
44
+
45
+ logger.trace("Added comment of length %d, new buffer position: %d", str_length, length)
46
+ return length
47
+
48
+
22
49
  def check_identification_header(page):
23
50
  """
24
51
  Check if a page contains a valid Opus identification header.
@@ -52,37 +79,103 @@ def check_identification_header(page):
52
79
  logger.debug("Opus identification header is valid")
53
80
 
54
81
 
55
- def prepare_opus_tags(page):
82
+ def prepare_opus_tags(page, custom_tags=False, bitrate=64, vbr=True, opus_binary=None):
56
83
  """
57
84
  Prepare standard Opus tags for a Tonie file.
58
85
 
59
86
  Args:
60
87
  page: OggPage to modify
88
+ custom_tags: Whether to use custom TonieToolbox tags instead of default ones
89
+ bitrate: Actual bitrate used for encoding
90
+ vbr: Whether variable bitrate was used
91
+ opus_binary: Path to opusenc binary for version detection
61
92
 
62
93
  Returns:
63
94
  OggPage: Modified page with Tonie-compatible Opus tags
64
95
  """
65
96
  logger.debug("Preparing Opus tags for Tonie compatibility")
66
97
  page.segments.clear()
67
- segment = OpusPacket(None)
68
- segment.size = len(OPUS_TAGS[0])
69
- segment.data = bytearray(OPUS_TAGS[0])
70
- segment.spanning_packet = True
71
- segment.first_packet = True
72
- page.segments.append(segment)
73
-
74
- segment = OpusPacket(None)
75
- segment.size = len(OPUS_TAGS[1])
76
- segment.data = bytearray(OPUS_TAGS[1])
77
- segment.spanning_packet = False
78
- segment.first_packet = False
79
- page.segments.append(segment)
98
+
99
+ if not custom_tags:
100
+ # Use the default hardcoded tags for backward compatibility
101
+ segment = OpusPacket(None)
102
+ segment.size = len(OPUS_TAGS[0])
103
+ segment.data = bytearray(OPUS_TAGS[0])
104
+ segment.spanning_packet = True
105
+ segment.first_packet = True
106
+ page.segments.append(segment)
107
+
108
+ segment = OpusPacket(None)
109
+ segment.size = len(OPUS_TAGS[1])
110
+ segment.data = bytearray(OPUS_TAGS[1])
111
+ segment.spanning_packet = False
112
+ segment.first_packet = False
113
+ page.segments.append(segment)
114
+ else:
115
+ # Use custom tags for TonieToolbox
116
+ # Create buffer for opus tags (similar to teddyCloud implementation)
117
+ logger.debug("Creating custom Opus tags")
118
+ comment_data = bytearray(0x1B4) # Same size as in teddyCloud
119
+
120
+ # OpusTags signature
121
+ comment_data_pos = 0
122
+ comment_data[comment_data_pos:comment_data_pos+8] = b"OpusTags"
123
+ comment_data_pos += 8
124
+
125
+ # Vendor string
126
+ comment_data_pos = toniefile_comment_add(comment_data, comment_data_pos, "TonieToolbox")
127
+
128
+ # Number of comments (3 comments: version, encoder info, and encoder options)
129
+ comments_count = 3
130
+ comment_data[comment_data_pos:comment_data_pos+4] = struct.pack("<I", comments_count)
131
+ comment_data_pos += 4
132
+
133
+ # Add version information
134
+ from . import __version__
135
+ version_str = f"version={__version__}"
136
+ comment_data_pos = toniefile_comment_add(comment_data, comment_data_pos, version_str)
137
+
138
+ # Get actual opusenc version
139
+ from .dependency_manager import get_opus_version
140
+ encoder_info = get_opus_version(opus_binary)
141
+ comment_data_pos = toniefile_comment_add(comment_data, comment_data_pos, f"encoder={encoder_info}")
142
+
143
+ # Create encoder options string with actual settings
144
+ vbr_opt = "--vbr" if vbr else "--cbr"
145
+ encoder_options = f"encoder_options=--bitrate {bitrate} {vbr_opt}"
146
+ comment_data_pos = toniefile_comment_add(comment_data, comment_data_pos, encoder_options)
147
+
148
+ # Add padding
149
+ remain = len(comment_data) - comment_data_pos - 4
150
+ comment_data[comment_data_pos:comment_data_pos+4] = struct.pack("<I", remain)
151
+ comment_data_pos += 4
152
+ comment_data[comment_data_pos:comment_data_pos+4] = b"pad="
153
+
154
+ # Create segments - handle data in chunks of 255 bytes maximum
155
+ comment_data = comment_data[:comment_data_pos + remain] # Trim to actual used size
156
+
157
+ # Split large data into smaller segments (each <= 255 bytes)
158
+ remaining_data = comment_data
159
+ first_segment = True
160
+
161
+ while remaining_data:
162
+ chunk_size = min(255, len(remaining_data))
163
+ segment = OpusPacket(None)
164
+ segment.size = chunk_size
165
+ segment.data = remaining_data[:chunk_size]
166
+ segment.spanning_packet = len(remaining_data) > chunk_size # More data follows
167
+ segment.first_packet = first_segment
168
+ page.segments.append(segment)
169
+
170
+ remaining_data = remaining_data[chunk_size:]
171
+ first_segment = False
172
+
80
173
  page.correct_values(0)
81
174
  logger.trace("Opus tags prepared with %d segments", len(page.segments))
82
175
  return page
83
176
 
84
177
 
85
- def copy_first_and_second_page(in_file, out_file, timestamp, sha):
178
+ def copy_first_and_second_page(in_file, out_file, timestamp, sha, use_custom_tags=True, bitrate=64, vbr=True, opus_binary=None):
86
179
  """
87
180
  Copy and modify the first two pages of an Opus file for a Tonie file.
88
181
 
@@ -91,9 +184,10 @@ def copy_first_and_second_page(in_file, out_file, timestamp, sha):
91
184
  out_file: Output file handle
92
185
  timestamp: Timestamp to use for the Tonie file
93
186
  sha: SHA1 hash object to update with written data
94
-
95
- Raises:
96
- RuntimeError: If OGG pages cannot be found
187
+ use_custom_tags: Whether to use custom TonieToolbox tags
188
+ bitrate: Actual bitrate used for encoding
189
+ vbr: Whether VBR was used
190
+ opus_binary: Path to opusenc binary
97
191
  """
98
192
  logger.debug("Copying first and second pages with timestamp %d", timestamp)
99
193
  found = OggPage.seek_to_page_header(in_file)
@@ -116,7 +210,7 @@ def copy_first_and_second_page(in_file, out_file, timestamp, sha):
116
210
  page = OggPage(in_file)
117
211
  page.serial_no = timestamp
118
212
  page.checksum = page.calc_checksum()
119
- page = prepare_opus_tags(page)
213
+ page = prepare_opus_tags(page, use_custom_tags, bitrate, vbr, opus_binary)
120
214
  page.write_page(out_file, sha)
121
215
  logger.debug("Second page written successfully")
122
216
 
@@ -277,7 +371,8 @@ def fix_tonie_header(out_file, chapters, timestamp, sha):
277
371
 
278
372
 
279
373
  def create_tonie_file(output_file, input_files, no_tonie_header=False, user_timestamp=None,
280
- bitrate=96, vbr=True, ffmpeg_binary=None, opus_binary=None, keep_temp=False, auto_download=False):
374
+ bitrate=96, vbr=True, ffmpeg_binary=None, opus_binary=None, keep_temp=False, auto_download=False,
375
+ use_custom_tags=True):
281
376
  """
282
377
  Create a Tonie file from input files.
283
378
 
@@ -292,6 +387,7 @@ def create_tonie_file(output_file, input_files, no_tonie_header=False, user_time
292
387
  opus_binary: Path to opusenc binary
293
388
  keep_temp: Whether to keep temporary opus files for testing
294
389
  auto_download: Whether to automatically download dependencies if not found
390
+ use_custom_tags: Whether to use dynamic comment tags generated with toniefile_comment_add
295
391
  """
296
392
  from .audio_conversion import get_opus_tempfile
297
393
 
@@ -369,7 +465,7 @@ def create_tonie_file(output_file, input_files, no_tonie_header=False, user_time
369
465
  try:
370
466
  if next_page_no == 2:
371
467
  logger.debug("Processing first file: copying first and second page")
372
- copy_first_and_second_page(handle, out_file, timestamp, sha1)
468
+ copy_first_and_second_page(handle, out_file, timestamp, sha1, use_custom_tags, bitrate, vbr, opus_binary)
373
469
  else:
374
470
  logger.debug("Processing subsequent file: skipping first and second page")
375
471
  other_size = max_size
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: TonieToolbox
3
- Version: 0.2.1
3
+ Version: 0.2.3
4
4
  Summary: Convert audio files to Tonie box compatible format
5
5
  Home-page: https://github.com/Quentendo64/TonieToolbox
6
6
  Author: Quentendo64
@@ -46,6 +46,7 @@ A Python tool for converting audio files to Tonie box compatible format (TAF - T
46
46
  - [Basic Usage](#basic-usage)
47
47
  - [Advanced Options](#advanced-options)
48
48
  - [Common Usage Examples](#common-usage-examples)
49
+ - [Media Tags](#media-tags)
49
50
  - [Technical Details](#technical-details)
50
51
  - [TAF File Structure](#taf-tonie-audio-format-file-structure)
51
52
  - [File Analysis](#file-analysis)
@@ -67,15 +68,17 @@ The tool provides several capabilities:
67
68
  - Split Tonie files into individual opus tracks
68
69
  - Compare two TAF files for debugging differences
69
70
  - Support various input formats through FFmpeg conversion
71
+ - Extract and use audio media tags (ID3, Vorbis Comments, etc.) for better file naming
70
72
 
71
73
  ## Requirements
72
74
 
73
75
  - Python 3.6 or higher
74
- - FFmpeg (for converting non-opus audio files).
76
+ - FFmpeg (for converting non-opus audio files)
75
77
  - opus-tools (specifically `opusenc` for encoding to opus format)
78
+ - mutagen (for reading audio file metadata, auto-installed when needed)
76
79
 
77
80
  Make sure FFmpeg and opus-tools are installed on your system and accessible in your PATH.
78
- If the requirements are not found in PATH. TonieToolbox will download the missing requirements.
81
+ If the requirements are not found in PATH. TonieToolbox will download the missing requirements with --auto-download.
79
82
 
80
83
  ## Installation
81
84
 
@@ -166,7 +169,8 @@ Output:
166
169
  ```
167
170
  usage: TonieToolbox.py [-h] [-v] [-t TIMESTAMP] [-f FFMPEG] [-o OPUSENC]
168
171
  [-b BITRATE] [-c] [-a TAG] [-n] [-i] [-s] [-r] [-O]
169
- [-A] [-k] [-C FILE2] [-D] [-d] [-T] [-q] [-Q]
172
+ [-A] [-k] [-C FILE2] [-D] [-m] [--name-template TEMPLATE]
173
+ [--show-tags] [-d] [-T] [-q] [-Q]
170
174
  SOURCE [TARGET]
171
175
 
172
176
  Create Tonie compatible file from Ogg opus file(s).
@@ -198,6 +202,12 @@ optional arguments:
198
202
  -D, --detailed-compare
199
203
  Show detailed OGG page differences when comparing files
200
204
 
205
+ Media Tag Options:
206
+ -m, --use-media-tags Use media tags from audio files for naming
207
+ --name-template TEMPLATE
208
+ Template for naming files using media tags. Example: "{album} - {artist}"
209
+ --show-tags Show available media tags from input files
210
+
201
211
  Version Check Options:
202
212
  -S, --skip-update-check
203
213
  Skip checking for updates
@@ -269,6 +279,52 @@ Process a music collection with nested album folders and save TAF files alongsid
269
279
  tonietoolbox --recursive --output-to-source "\Hörspiele\"
270
280
  ```
271
281
 
282
+ ### Media Tags
283
+
284
+ TonieToolbox can read metadata tags from audio files (such as ID3 tags in MP3 files, Vorbis comments in FLAC/OGG files, etc.) and use them to create more meaningful filenames or display information about your audio collection.
285
+
286
+ #### View available tags in audio files:
287
+
288
+ To see what tags are available in your audio files:
289
+
290
+ ```
291
+ tonietoolbox --show-tags input.mp3
292
+ ```
293
+
294
+ This will display all readable tags from the file, which can be useful for creating naming templates.
295
+
296
+ #### Use media tags for file naming:
297
+
298
+ To use the metadata from audio files when generating output filenames:
299
+
300
+ ```
301
+ tonietoolbox input.mp3 --use-media-tags
302
+ ```
303
+
304
+ For single files, this will use a default template of "{title} - {artist}" for the output filename.
305
+
306
+ #### Custom naming templates:
307
+
308
+ You can specify custom templates for generating filenames based on the audio metadata:
309
+
310
+ ```
311
+ tonietoolbox input.mp3 --use-media-tags --name-template "{artist} - {album} - {title}"
312
+ ```
313
+
314
+ #### Recursive processing with media tags:
315
+
316
+ When processing folders recursively, media tags can provide more consistent naming:
317
+
318
+ ```
319
+ tonietoolbox --recursive --use-media-tags "Music/Collection/"
320
+ ```
321
+
322
+ This will attempt to use the album information from the audio files for naming the output files:
323
+
324
+ ```
325
+ tonietoolbox --recursive --use-media-tags --name-template "{date} - {album} ({artist})" "Music/Collection/"
326
+ ```
327
+
272
328
  ## Technical Details
273
329
 
274
330
  ### TAF (Tonie Audio Format) File Structure
@@ -0,0 +1,22 @@
1
+ TonieToolbox/__init__.py,sha256=mRuWBCYSLc6tAHyVDd70qM5LHgWH7R8ClFbFfILlEso,96
2
+ TonieToolbox/__main__.py,sha256=Otltlkl9Cd36BYs5_23ufCkdxguFFVzuVDwSvWDSPfI,17078
3
+ TonieToolbox/audio_conversion.py,sha256=ra72qsE8j2GEP_4kqDT9m6aKlnnREZhZAlpf7y83pA0,11202
4
+ TonieToolbox/constants.py,sha256=QQWQpnCI65GByLlXLOkt2n8nALLu4m6BWp0zuhI3M04,2021
5
+ TonieToolbox/dependency_manager.py,sha256=dRy8JxV3Dv1x8uWIXMonU0ArnCRknI1rkrj8MkJcyl4,27916
6
+ TonieToolbox/filename_generator.py,sha256=RqQHyGTKakuWR01yMSnFVMU_HfLw3rqFxKhXNIHdTlg,3441
7
+ TonieToolbox/logger.py,sha256=Up9fBVkOZwkY61_645bX4tienCpyVSkap-FeTV0v730,1441
8
+ TonieToolbox/media_tags.py,sha256=T8osaU101iGC43NaMbbHD8mwNh-dGa_uPj7IPlhMIno,16091
9
+ TonieToolbox/ogg_page.py,sha256=-ViaIRBgh5ayfwmyplL8QmmRr5P36X8W0DdHkSFUYUU,21948
10
+ TonieToolbox/opus_packet.py,sha256=OcHXEe3I_K4mWPUD55prpG42sZxJsEeAxqSbFxBmb0c,7895
11
+ TonieToolbox/recursive_processor.py,sha256=m70-oJ4MRXvI9OMy14sC9UsQIxwVEnEoA3hhLGdHhL4,13457
12
+ TonieToolbox/tonie_analysis.py,sha256=kp4Wx4cTDddtF2AlS6IX4xs1vQ-mpZ0gsAy4-UdRAAM,23287
13
+ TonieToolbox/tonie_file.py,sha256=vY0s8X4ln35ZXpdpGmBcIxgpTJAjduiVvBh34WObyrw,19647
14
+ TonieToolbox/tonie_header.proto,sha256=WaWfwO4VrwGtscK2ujfDRKtpeBpaVPoZhI8iMmR-C0U,202
15
+ TonieToolbox/tonie_header_pb2.py,sha256=s5bp4ULTEekgq6T61z9fDkRavyPM-3eREs20f_Pxxe8,3665
16
+ TonieToolbox/version_handler.py,sha256=7Zx-pgzAUhz6jMplvNal1wHyxidodVxaNcAV0EMph5k,9778
17
+ tonietoolbox-0.2.3.dist-info/licenses/LICENSE.md,sha256=rGoga9ZAgNco9fBapVFpWf6ri7HOBp1KRnt1uIruXMk,35190
18
+ tonietoolbox-0.2.3.dist-info/METADATA,sha256=2LiL3ouh2jIdA3CD7i8nSZY-PStXbhkT8GNTZVvpgrM,12444
19
+ tonietoolbox-0.2.3.dist-info/WHEEL,sha256=pxyMxgL8-pra_rKaQ4drOZAegBVuX-G_4nRHjjgWbmo,91
20
+ tonietoolbox-0.2.3.dist-info/entry_points.txt,sha256=oqpeyBxel7aScg35Xr4gZKnf486S5KW9okqeBwyJxxc,60
21
+ tonietoolbox-0.2.3.dist-info/top_level.txt,sha256=Wkkm-2p7I3ENfS7ZbYtYUB2g-xwHrXVlERHfonsOPuE,13
22
+ tonietoolbox-0.2.3.dist-info/RECORD,,
@@ -1,5 +1,5 @@
1
1
  Wheel-Version: 1.0
2
- Generator: setuptools (78.1.1)
2
+ Generator: setuptools (79.0.0)
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
5
5
 
@@ -1,21 +0,0 @@
1
- TonieToolbox/__init__.py,sha256=141g35CQjQQB9-oCGkOKQkDV9GWkssN0IhK1vh1FBoc,96
2
- TonieToolbox/__main__.py,sha256=KE5x6KMLQZGd4nwjk71AmlxMZOgnN-lXLBNDi6XkHJc,10917
3
- TonieToolbox/audio_conversion.py,sha256=ra72qsE8j2GEP_4kqDT9m6aKlnnREZhZAlpf7y83pA0,11202
4
- TonieToolbox/constants.py,sha256=QQWQpnCI65GByLlXLOkt2n8nALLu4m6BWp0zuhI3M04,2021
5
- TonieToolbox/dependency_manager.py,sha256=fWtYp_UQDKrgIKcOyy95w7Grk_wYx5Fadyg8ulpb7nE,23451
6
- TonieToolbox/filename_generator.py,sha256=RqQHyGTKakuWR01yMSnFVMU_HfLw3rqFxKhXNIHdTlg,3441
7
- TonieToolbox/logger.py,sha256=Up9fBVkOZwkY61_645bX4tienCpyVSkap-FeTV0v730,1441
8
- TonieToolbox/ogg_page.py,sha256=-ViaIRBgh5ayfwmyplL8QmmRr5P36X8W0DdHkSFUYUU,21948
9
- TonieToolbox/opus_packet.py,sha256=OcHXEe3I_K4mWPUD55prpG42sZxJsEeAxqSbFxBmb0c,7895
10
- TonieToolbox/recursive_processor.py,sha256=vhQzC05bJVRPX8laj_5lxuRD40eLsZatzwCoCavMsmY,9304
11
- TonieToolbox/tonie_analysis.py,sha256=4eOzxHL_g0TJFhuexNHcZXivxZ7eb5xfb9-efUZ02W0,20344
12
- TonieToolbox/tonie_file.py,sha256=nIS4qhpBKIyPvTU39yYljRidpY6cz78halXlz3HJy9w,15294
13
- TonieToolbox/tonie_header.proto,sha256=WaWfwO4VrwGtscK2ujfDRKtpeBpaVPoZhI8iMmR-C0U,202
14
- TonieToolbox/tonie_header_pb2.py,sha256=s5bp4ULTEekgq6T61z9fDkRavyPM-3eREs20f_Pxxe8,3665
15
- TonieToolbox/version_handler.py,sha256=7Zx-pgzAUhz6jMplvNal1wHyxidodVxaNcAV0EMph5k,9778
16
- tonietoolbox-0.2.1.dist-info/licenses/LICENSE.md,sha256=rGoga9ZAgNco9fBapVFpWf6ri7HOBp1KRnt1uIruXMk,35190
17
- tonietoolbox-0.2.1.dist-info/METADATA,sha256=mMJ6-auDHNFK12NcDU_HNQlwiC6Y98fprLJICBihaxI,10514
18
- tonietoolbox-0.2.1.dist-info/WHEEL,sha256=lTU6B6eIfYoiQJTZNc-fyaR6BpL6ehTzU3xGYxn2n8k,91
19
- tonietoolbox-0.2.1.dist-info/entry_points.txt,sha256=oqpeyBxel7aScg35Xr4gZKnf486S5KW9okqeBwyJxxc,60
20
- tonietoolbox-0.2.1.dist-info/top_level.txt,sha256=Wkkm-2p7I3ENfS7ZbYtYUB2g-xwHrXVlERHfonsOPuE,13
21
- tonietoolbox-0.2.1.dist-info/RECORD,,