Documentation
¶
Overview ¶
Package id3 implements a decoder and encoder for ID3 tags.
Nomenclature ¶
ID3 uses nomenclature that might be counterintuitive to people not familiar with it. Because this library uses the same choice of words the following list of clarifications is included.
tag: In ID3, tag means the entire set of metadata, not just one field of it. frame: A frame describes a single piece of information, for example a song's title.
Supported versions ¶
This library supports reading v2.3 and v2.4 tags, but only writing v2.4 tags.
The primary reason for not allowing writing older versions is that they cannot represent all data that is available with v2.4, and designing the API in a way that's both user friendly and able to reject data is not worth the trouble.
Automatic upgrading ¶
The library's internal representation of tags matches that of v2.4. When tags with an older version are being read, they will be automatically converted to v2.4.
One consequence of this is that when you read a file with v2.3 tags and immediately save it, it will now be a file with valid v2.4 tags.
The upgrade process makes the following changes to the tags:
- TYER, TDAT and TIME get replaced by TDRC
- TORY gets replaced by TDOR
- XDOR gets replaced by TDOR
- The slash as a separator for multiple values gets replaced by null bytes
One special case is the TRDA frame because there is no way to automatically convert it to v2.4. The upgrade process will not directly delete the frame, so that you can manually upgrade it if desired, but it won't be written back to the file. The frame is rarely used and insignificant, so it's not a big loss.
Accessing and manipulating frames ¶
There are two ways to access frames: Using provided getter and setter methods (there is one for every standard frame), and working directly with the underlying frames.
For frames that usually support multiple values, e.g. languages, there will be two different setters and getters: One that operates on slices and one that operates on single values. When getting a single value, it will return the first value from the underlying list. When setting a single value, it will overwrite the list with a single value.
Text frames and user text frames can be manipulated with the GetTextFrame* and SetTextFrame* class of functions. There are special methods for working with integers, slices and times. This class of functions expects the raw frame names (e.g. "TLEN"), with the special case of user text frames ("TXXX") where it expects a format of the kind "TXXX:The frame description" to address a specific user text frame.
Encodings ¶
While ID3v2 allows a variety of encodings (ISO-8859-1, UTF-16 and in v2.4 also UTF-8), this library only supports writing UTF-8. When reading frames with different encodings, they will be converted to UTF-8.
The rationale behind this is that UTF-8 is the encoding assumed by most of the Go standard library, and that the other encodings have no realistic benefits over UTF-8.
Behaviour when encountering invalid data ¶
There are two kinds of invalid data that can be encountered: Data that doesn't look like a valid frame, and data that is invalid in the context of its frame.
An example for the first case is a frame identifier that doesn't consist of only A-Z0-9. That usually happens when other programs wrote invalid data. One common case are wrongly encoded sizes, which will cause us to read random binary data.
An example for the second case is text that isn't valid according to the specified encoding.
In the first case, parsing of the entire tag will be aborted because it cannot be ensured that bad things won't happen.
In the second case only that specific frame will be dropped.
Unsupported frames ¶
Unsupported frames, like extensions by Apple, will be left untouched, meaning that they will be read and written as raw byte slices. If you know how to handle them, you can write your own function to parse and modify the content. All unsupported frames will be of type UnsupportedFrame.
Index ¶
- Constants
- Variables
- func Check(r Peeker) (bool, error)
- type Comment
- type CommentFrame
- type Decoder
- type Encoder
- type Encoding
- type Frame
- type FrameFlags
- type FrameHeader
- type FrameType
- type FramesMap
- type Header
- type HeaderFlags
- type InvalidFrameHeaderError
- type InvalidTagHeaderError
- type MusicCDIdentifierFrame
- type Peeker
- type PictureFrame
- type PictureType
- type PrivateFrame
- type Tag
- func (t *Tag) Album() string
- func (t *Tag) AlbumSortOrder() string
- func (t *Tag) Artist() string
- func (t *Tag) Artists() []string
- func (t *Tag) BPM() int
- func (t *Tag) Band() string
- func (t *Tag) Clear()
- func (t *Tag) Comments() []Comment
- func (t *Tag) Composer() string
- func (t *Tag) Composers() []string
- func (t *Tag) Conductor() string
- func (t *Tag) EncodingTime() time.Time
- func (t *Tag) GetTextFrame(name FrameType) string
- func (t *Tag) GetTextFrameNumber(name FrameType) int
- func (t *Tag) GetTextFrameSlice(name FrameType) []string
- func (t *Tag) GetTextFrameTime(name FrameType) time.Time
- func (t *Tag) HasFrame(name FrameType) bool
- func (t *Tag) ISRC() string
- func (t *Tag) Language() string
- func (t *Tag) Languages() []string
- func (t *Tag) Length() time.Duration
- func (t *Tag) Mood() string
- func (t *Tag) OriginalArtist() string
- func (t *Tag) OriginalArtists() []string
- func (t *Tag) OriginalFilename() string
- func (t *Tag) OriginalReleaseTime() time.Time
- func (t *Tag) Owner() string
- func (t *Tag) PerformerSortOrder() string
- func (t *Tag) PlaylistDelay() time.Duration
- func (t *Tag) Publisher() string
- func (t *Tag) RecordingTime() time.Time
- func (t *Tag) RemoveFrames(name FrameType)
- func (t *Tag) Sanitize()
- func (t *Tag) SetAlbum(album string)
- func (t *Tag) SetAlbumSortOrder(s string)
- func (t *Tag) SetArtist(artist string)
- func (t *Tag) SetArtists(artists []string)
- func (t *Tag) SetBPM(bpm int)
- func (t *Tag) SetBand(band string)
- func (t *Tag) SetComments(comments []Comment)
- func (t *Tag) SetComposer(composer string)
- func (t *Tag) SetComposers(composers []string)
- func (t *Tag) SetConductor(name string)
- func (t *Tag) SetEncodingTime(et time.Time)
- func (t *Tag) SetISRC(isrc string)
- func (t *Tag) SetLanguage(lang string)
- func (t *Tag) SetLanguages(langs []string)
- func (t *Tag) SetLength(d time.Duration)
- func (t *Tag) SetMood(mood string)
- func (t *Tag) SetOriginalArtist(name string)
- func (t *Tag) SetOriginalArtists(names []string)
- func (t *Tag) SetOriginalFilename(name string)
- func (t *Tag) SetOriginalReleaseTime(rt time.Time)
- func (t *Tag) SetOwner(owner string)
- func (t *Tag) SetPerformerSortOrder(s string)
- func (t *Tag) SetPlaylistDelay(d time.Duration)
- func (t *Tag) SetPublisher(publisher string)
- func (t *Tag) SetRecordingTime(rt time.Time)
- func (t *Tag) SetStationName(name string)
- func (t *Tag) SetStationOwner(owner string)
- func (t *Tag) SetTextFrame(name FrameType, value string)
- func (t *Tag) SetTextFrameNumber(name FrameType, value int)
- func (t *Tag) SetTextFrameSlice(name FrameType, value []string)
- func (t *Tag) SetTextFrameTime(name FrameType, value time.Time)
- func (t *Tag) SetTitle(title string)
- func (t *Tag) SetTitleSortOrder(s string)
- func (t *Tag) StationName() string
- func (t *Tag) StationOwner() string
- func (t *Tag) Title() string
- func (t *Tag) TitleSortOrder() string
- func (t *Tag) UserTextFrames() []UserTextInformationFrame
- func (t *Tag) Validate() error
- type TextInformationFrame
- type URLLinkFrame
- type UnimplementedFeatureError
- type UniqueFileIdentifierFrame
- type UnsupportedFrame
- type UnsupportedVersionError
- type UnsynchronisedLyricsFrame
- type UserDefinedURLLinkFrame
- type UserTextInformationFrame
- type Version
Constants ¶
const TimeFormat = "2006-01-02T15:04:05"
Variables ¶
var FrameNames = map[FrameType]string{
"AENC": "Audio encryption",
"APIC": "Attached picture",
"ASPI": "Audio seek point index",
"COMM": "Comments",
"COMR": "Commercial frame",
"ENCR": "Encryption method registration",
"EQU2": "Equalisation (2)",
"ETCO": "Event timing codes",
"GEOB": "General encapsulated object",
"GRID": "Group identification registration",
"LINK": "Linked information",
"MCDI": "Music CD identifier",
"MLLT": "MPEG location lookup table",
"OWNE": "Ownership frame",
"PRIV": "Private frame",
"PCNT": "Play counter",
"POPM": "Popularimeter",
"POSS": "Position synchronisation frame",
"RBUF": "Recommended buffer size",
"RVA2": "Relative volume adjustment (2)",
"RVRB": "Reverb",
"SEEK": "Seek frame",
"SIGN": "Signature frame",
"SYLT": "Synchronised lyric/text",
"SYTC": "Synchronised tempo codes",
"TALB": "Album/Movie/Show title",
"TBPM": "BPM (beats per minute)",
"TCOM": "Composer",
"TCON": "Content type",
"TCOP": "Copyright message",
"TDEN": "Encoding time",
"TDLY": "Playlist delay",
"TDOR": "Original release time",
"TDRC": "Recording time",
"TDRL": "Release time",
"TDTG": "Tagging time",
"TENC": "Encoded by",
"TEXT": "Lyricist/Text writer",
"TFLT": "File type",
"TIPL": "Involved people list",
"TIT1": "Content group description",
"TIT2": "Title/songname/content description",
"TIT3": "Subtitle/Description refinement",
"TKEY": "Initial key",
"TLAN": "Language(s)",
"TLEN": "Length",
"TMCL": "Musician credits list",
"TMED": "Media type",
"TMOO": "Mood",
"TOAL": "Original album/movie/show title",
"TOFN": "Original filename",
"TOLY": "Original lyricist(s)/text writer(s)",
"TORY": "Original release year",
"TOPE": "Original artist(s)/performer(s)",
"TOWN": "File owner/licensee",
"TPE1": "Lead performer(s)/Soloist(s)",
"TPE2": "Band/orchestra/accompaniment",
"TPE3": "Conductor/performer refinement",
"TPE4": "Interpreted, remixed, or otherwise modified by",
"TPOS": "Part of a set",
"TPRO": "Produced notice",
"TPUB": "Publisher",
"TRCK": "Track number/Position in set",
"TRSN": "Internet radio station name",
"TRSO": "Internet radio station owner",
"TSOA": "Album sort order",
"TSOP": "Performer sort order",
"TSOT": "Title sort order",
"TSO2": "Album Artist sort order",
"TSOC": "Composer sort oder",
"TSRC": "ISRC (international standard recording code)",
"TSSE": "Software/Hardware and settings used for encoding",
"TSST": "Set subtitle",
"TYER": "Year",
"TXXX": "User defined text information frame",
"UFID": "Unique file identifier",
"USER": "Terms of use",
"USLT": "Unsynchronised lyric/text transcription",
"WCOM": "Commercial information",
"WCOP": "Copyright/Legal information",
"WOAF": "Official audio file webpage",
"WOAR": "Official artist/performer webpage",
"WOAS": "Official audio source webpage",
"WORS": "Official Internet radio station homepage",
"WPAY": "Payment",
"WPUB": "Publishers official webpage",
"WXXX": "User defined URL link frame",
}
var Genres = []string{}/* 126 elements not displayed */
var Magic = []byte("ID3")
var PictureTypes = []string{
"Other",
"32x32 pixels 'file icon' (PNG only)",
"Other file icon",
"Cover (front)",
"Cover (back)",
"Leaflet page",
"Media (e.g. label side of CD)",
"Lead artist/lead performer/soloist",
"Artist/performer",
"Conductor",
"Band/Orchestra",
"Composer",
"Lyricist/text writer",
"Recording Location",
"During recording",
"During performance",
"Movie/video screen capture",
"A bright coloured fish",
"Illustration",
"Band/artist logotype",
"Publisher/Studio logotype",
}
Functions ¶
Types ¶
type CommentFrame ¶
type CommentFrame struct { FrameHeader Language string Description string Text string }
func (CommentFrame) Encode ¶
func (f CommentFrame) Encode() []byte
func (CommentFrame) Size ¶
func (f CommentFrame) Size() int
func (CommentFrame) Value ¶
func (f CommentFrame) Value() string
type Decoder ¶
type Decoder struct {
// contains filtered or unexported fields
}
func NewDecoder ¶
func (*Decoder) Parse ¶
Parse parses a tag.
Parse will always return a valid tag. In the case of an error, the tag will be empty.
If Parse successfully parsed a tag, the reader will be positioned immediately after the tag, which usually is directly before audio data. If there wasn't a valid tag, however, the position of the reader is undefined. If you're not sure if your reader starts with a tag at all, consider using Check first.
Parse cannot be called if either ParseHeader or ParseFrame have been called for the current tag.
func (*Decoder) ParseFrame ¶
ParseFrame reads the next ID3 frame. When it reaches padding, it will read and discard all of it and return io.EOF. This should set the reader immediately before the audio data.
ParseHeader must be called before calling ParseFrame.
func (*Decoder) ParseHeader ¶
ParseHeader parses only the ID3 header.
type Encoder ¶
type Encoder struct { // The amount of padding that will be added after the last frame. Padding int // contains filtered or unexported fields }
func NewEncoder ¶
func (*Encoder) WriteFrame ¶
func (*Encoder) WriteHeader ¶
func (*Encoder) WritePadding ¶
type Frame ¶
type Frame interface { ID() FrameType // TODO consider renaming to Type() Header() FrameHeader Value() string Encode() []byte Size() int }
type FrameFlags ¶
type FrameFlags uint16
func (FrameFlags) Compressed ¶
func (f FrameFlags) Compressed() bool
func (FrameFlags) Encrypted ¶
func (f FrameFlags) Encrypted() bool
func (FrameFlags) Grouped ¶
func (f FrameFlags) Grouped() bool
func (FrameFlags) PreserveFileAlteration ¶
func (f FrameFlags) PreserveFileAlteration() bool
func (FrameFlags) PreserveTagAlteration ¶
func (f FrameFlags) PreserveTagAlteration() bool
func (FrameFlags) ReadOnly ¶
func (f FrameFlags) ReadOnly() bool
type FrameHeader ¶
type FrameHeader struct {
// contains filtered or unexported fields
}
func (FrameHeader) Header ¶
func (h FrameHeader) Header() FrameHeader
func (FrameHeader) ID ¶
func (f FrameHeader) ID() FrameType
type Header ¶
type Header struct { Version Version Flags HeaderFlags Size int // The size of the tag (exluding the size of the header) }
type HeaderFlags ¶
type HeaderFlags byte
func (HeaderFlags) Experimental ¶
func (f HeaderFlags) Experimental() bool
func (HeaderFlags) ExtendedHeader ¶
func (f HeaderFlags) ExtendedHeader() bool
func (HeaderFlags) UndefinedSet ¶
func (f HeaderFlags) UndefinedSet() bool
func (HeaderFlags) Unsynchronisation ¶
func (f HeaderFlags) Unsynchronisation() bool
type InvalidFrameHeaderError ¶
func (InvalidFrameHeaderError) Error ¶
func (err InvalidFrameHeaderError) Error() string
type InvalidTagHeaderError ¶
type InvalidTagHeaderError struct {
Magic []byte
}
func (InvalidTagHeaderError) Error ¶
func (err InvalidTagHeaderError) Error() string
type MusicCDIdentifierFrame ¶
type MusicCDIdentifierFrame struct { FrameHeader TOC []byte }
func (MusicCDIdentifierFrame) Encode ¶
func (f MusicCDIdentifierFrame) Encode() []byte
func (MusicCDIdentifierFrame) Size ¶
func (f MusicCDIdentifierFrame) Size() int
func (MusicCDIdentifierFrame) Value ¶
func (f MusicCDIdentifierFrame) Value() string
type PictureFrame ¶
type PictureFrame struct { FrameHeader MIMEType string PictureType PictureType Description string Data []byte }
func (PictureFrame) Encode ¶
func (f PictureFrame) Encode() []byte
func (PictureFrame) Size ¶
func (f PictureFrame) Size() int
func (PictureFrame) Value ¶
func (f PictureFrame) Value() string
type PictureType ¶
type PictureType byte
func (PictureType) String ¶
func (p PictureType) String() string
type PrivateFrame ¶
type PrivateFrame struct { FrameHeader Owner []byte Data []byte }
func (PrivateFrame) Encode ¶
func (f PrivateFrame) Encode() []byte
func (PrivateFrame) Size ¶
func (f PrivateFrame) Size() int
func (PrivateFrame) Value ¶
func (f PrivateFrame) Value() string
type Tag ¶
type Tag struct { Flags HeaderFlags Frames FramesMap }
func (*Tag) AlbumSortOrder ¶
func (*Tag) EncodingTime ¶
func (*Tag) GetTextFrame ¶
GetTextFrame returns the text frame specified by name.
To access user text frames, specify the name like "TXXX:The description".
func (*Tag) GetTextFrameNumber ¶
func (*Tag) GetTextFrameSlice ¶
func (*Tag) OriginalArtist ¶
func (*Tag) OriginalArtists ¶
func (*Tag) OriginalFilename ¶
func (*Tag) OriginalReleaseTime ¶
func (*Tag) PerformerSortOrder ¶
func (*Tag) PlaylistDelay ¶
func (*Tag) RecordingTime ¶
func (*Tag) RemoveFrames ¶
func (*Tag) Sanitize ¶
func (t *Tag) Sanitize()
Sanitize will remove all frames that aren't valid. Check the documentation of (*Tag).Validate() to see what "valid" means.
func (*Tag) SetAlbumSortOrder ¶
func (*Tag) SetArtists ¶
func (*Tag) SetComments ¶
func (*Tag) SetComposer ¶
func (*Tag) SetComposers ¶
func (*Tag) SetConductor ¶
func (*Tag) SetEncodingTime ¶
func (*Tag) SetLanguage ¶
func (*Tag) SetLanguages ¶
func (*Tag) SetOriginalArtist ¶
func (*Tag) SetOriginalArtists ¶
func (*Tag) SetOriginalFilename ¶
func (*Tag) SetOriginalReleaseTime ¶
func (*Tag) SetPerformerSortOrder ¶
func (*Tag) SetPlaylistDelay ¶
func (*Tag) SetPublisher ¶
func (*Tag) SetRecordingTime ¶
func (*Tag) SetStationName ¶
func (*Tag) SetStationOwner ¶
func (*Tag) SetTextFrame ¶
func (*Tag) SetTextFrameNumber ¶
func (*Tag) SetTextFrameSlice ¶
func (*Tag) SetTitleSortOrder ¶
func (*Tag) StationName ¶
func (*Tag) StationOwner ¶
func (*Tag) TitleSortOrder ¶
func (*Tag) UserTextFrames ¶
func (t *Tag) UserTextFrames() []UserTextInformationFrame
UserTextFrames returns all TXXX frames.
func (*Tag) Validate ¶
Validate checks whether the tags are conforming to the specification.
This entails two checks: Whether only frames that are covered by the specification are present and whether all values are within valid ranges.
It is well possible that reading existing files will result in invalid tags.
Calling Save() will not automatically validate the tags and will happily write invalid tags.
Assuming that the original file was valid and that only the getter/setter methods were used the generated tags should always be valid.
type TextInformationFrame ¶
type TextInformationFrame struct { FrameHeader Text string }
func (TextInformationFrame) Encode ¶
func (f TextInformationFrame) Encode() []byte
func (TextInformationFrame) Size ¶
func (f TextInformationFrame) Size() int
func (TextInformationFrame) Value ¶
func (f TextInformationFrame) Value() string
type URLLinkFrame ¶
type URLLinkFrame struct { FrameHeader URL string }
func (URLLinkFrame) Encode ¶
func (f URLLinkFrame) Encode() []byte
func (URLLinkFrame) Size ¶
func (f URLLinkFrame) Size() int
func (URLLinkFrame) Value ¶
func (f URLLinkFrame) Value() string
type UnimplementedFeatureError ¶
type UnimplementedFeatureError struct {
Feature string
}
func (UnimplementedFeatureError) Error ¶
func (err UnimplementedFeatureError) Error() string
type UniqueFileIdentifierFrame ¶
type UniqueFileIdentifierFrame struct { FrameHeader Owner string Identifier []byte }
func (UniqueFileIdentifierFrame) Encode ¶
func (f UniqueFileIdentifierFrame) Encode() []byte
func (UniqueFileIdentifierFrame) Size ¶
func (f UniqueFileIdentifierFrame) Size() int
func (UniqueFileIdentifierFrame) Value ¶
func (f UniqueFileIdentifierFrame) Value() string
type UnsupportedFrame ¶
type UnsupportedFrame struct { FrameHeader Data []byte }
func (UnsupportedFrame) Encode ¶
func (f UnsupportedFrame) Encode() []byte
func (UnsupportedFrame) Size ¶
func (f UnsupportedFrame) Size() int
func (UnsupportedFrame) Value ¶
func (f UnsupportedFrame) Value() string
type UnsupportedVersionError ¶
type UnsupportedVersionError struct {
Version Version
}
func (UnsupportedVersionError) Error ¶
func (err UnsupportedVersionError) Error() string
type UnsynchronisedLyricsFrame ¶
type UnsynchronisedLyricsFrame struct { FrameHeader Language string Description string Lyrics string }
func (UnsynchronisedLyricsFrame) Encode ¶
func (f UnsynchronisedLyricsFrame) Encode() []byte
func (UnsynchronisedLyricsFrame) Size ¶
func (f UnsynchronisedLyricsFrame) Size() int
func (UnsynchronisedLyricsFrame) Value ¶
func (f UnsynchronisedLyricsFrame) Value() string
type UserDefinedURLLinkFrame ¶
type UserDefinedURLLinkFrame struct { FrameHeader Description string URL string }
func (UserDefinedURLLinkFrame) Encode ¶
func (f UserDefinedURLLinkFrame) Encode() []byte
func (UserDefinedURLLinkFrame) Size ¶
func (f UserDefinedURLLinkFrame) Size() int
func (UserDefinedURLLinkFrame) Value ¶
func (f UserDefinedURLLinkFrame) Value() string
type UserTextInformationFrame ¶
type UserTextInformationFrame struct { FrameHeader Description string Text string }
func (UserTextInformationFrame) Encode ¶
func (f UserTextInformationFrame) Encode() []byte
func (UserTextInformationFrame) Size ¶
func (f UserTextInformationFrame) Size() int
func (UserTextInformationFrame) Value ¶
func (f UserTextInformationFrame) Value() string