package id3ed import ( "bytes" "encoding/binary" "fmt" "io" "strconv" ) type ID3v11Tags struct { SongName string Artist string Album string Year int Comment string Track int Genre string } // Retrieves ID3v1.1 field values of provided io.ReadSeeker (usually a file) func GetID3v11Tags(rs io.ReadSeeker) (*ID3v11Tags, error) { // set reader to the last 128 bytes _, err := rs.Seek(-int64(ID3V1SIZE), io.SeekEnd) if err != nil { return nil, fmt.Errorf("could not seek: %s", err) } tag, err := read(rs, 3) if err != nil { return nil, err } if !bytes.Equal(tag, []byte("TAG")) { // no TAG, given file does not use ID3v1 return nil, fmt.Errorf("does not use ID3v1: expected %s; got %s", "TAG", tag) } songname, err := readToString(rs, 30) if err != nil { return nil, err } artist, err := readToString(rs, 30) if err != nil { return nil, err } album, err := readToString(rs, 30) if err != nil { return nil, err } yearStr, err := readToString(rs, 4) if err != nil { return nil, err } year, err := strconv.Atoi(yearStr) if err != nil { return nil, fmt.Errorf("could not convert yearbytes into int: %s", err) } comment, err := readToString(rs, 28) if err != nil { return nil, err } // skip 1 null byte _, err = read(rs, 1) if err != nil { return nil, err } trackByte, err := read(rs, 1) if err != nil { return nil, err } track, err := binary.ReadVarint(bytes.NewBuffer(trackByte)) if err != nil { return nil, err } genreByte, err := read(rs, 1) if err != nil { return nil, err } genreInt, err := binary.ReadVarint(bytes.NewBuffer(genreByte)) if err != nil { return nil, err } genre, exists := id3v1genres[int(genreInt)] if !exists { genre = "" } return &ID3v11Tags{ SongName: songname, Artist: artist, Album: album, Year: year, Comment: comment, Track: int(track), Genre: genre, }, nil } // Writes given ID3v1.1 tags to dst func WriteID3v11Tags(dst io.WriteSeeker, tags ID3v11Tags) error { dst.Seek(0, io.SeekEnd) // TAG _, err := dst.Write([]byte("TAG")) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Song name err = writeToExtend(dst, []byte(tags.SongName), 30) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Artist err = writeToExtend(dst, []byte(tags.Artist), 30) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Album err = writeToExtend(dst, []byte(tags.Album), 30) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Year err = writeToExtend(dst, []byte(fmt.Sprint(tags.Year)), 4) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Comment err = writeToExtend(dst, []byte(tags.Comment), 28) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } _, err = dst.Write([]byte{0}) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } // Track trackBytes := make([]byte, 1) binary.PutVarint(trackBytes, int64(tags.Track)) // binary.BigEndian.PutUint16(trackBytes, uint16(tags.Track)) _, err = dst.Write(trackBytes) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } //Genre genreCode := getKey(id3v1genres, tags.Genre) if genreCode == -1 { // if no genre found - encode genre code as 255 genreCode = 255 } genrebyte := make([]byte, 1) binary.PutVarint(genrebyte, int64(genreCode)) err = writeToExtend(dst, genrebyte, 1) if err != nil { return fmt.Errorf("could not write to dst: %s", err) } return nil }