10 type ThreadedNews struct {
11 Categories map[string]NewsCategoryListData15 `yaml:"Categories"`
14 type NewsCategoryListData15 struct {
15 Type []byte `yaml:"Type"` // Size 2 ; Bundle (2) or category (3)
16 Count []byte // Article or SubCategory count Size 2
18 Name string `yaml:"Name"` //
19 Articles map[uint32]*NewsArtData `yaml:"Articles"` // Optional, if Type is Category
20 SubCats map[string]NewsCategoryListData15 `yaml:"SubCats"`
21 GUID []byte // Size 16
22 AddSN []byte // Size 4
23 DeleteSN []byte // Size 4
26 func (newscat *NewsCategoryListData15) GetNewsArtListData() NewsArtListData {
27 var newsArts []NewsArtList
28 var newsArtsPayload []byte
30 for i, art := range newscat.Articles {
32 binary.BigEndian.PutUint32(ID, i)
34 newArt := NewsArtList{
37 ParentID: art.ParentArt,
38 Flags: []byte{0, 0, 0, 0},
39 FlavorCount: []byte{0, 0},
40 Title: []byte(art.Title),
41 Poster: []byte(art.Poster),
42 ArticleSize: art.DataSize(),
44 newsArts = append(newsArts, newArt)
47 sort.Sort(byID(newsArts))
49 for _, v := range newsArts {
50 newsArtsPayload = append(newsArtsPayload, v.Payload()...)
53 nald := NewsArtListData{
54 ID: []byte{0, 0, 0, 0},
57 Description: []byte{},
58 NewsArtList: newsArtsPayload,
64 // NewsArtData represents single news article
65 type NewsArtData struct {
66 Title string `yaml:"Title"`
67 Poster string `yaml:"Poster"`
68 Date []byte `yaml:"Date"` // size 8
69 PrevArt []byte `yaml:"PrevArt"` // size 4
70 NextArt []byte `yaml:"NextArt"` // size 4
71 ParentArt []byte `yaml:"ParentArt"` // size 4
72 FirstChildArt []byte `yaml:"FirstChildArtArt"` // size 4
73 DataFlav []byte `yaml:"DataFlav"` // "text/plain"
74 Data string `yaml:"Data"`
77 func (art *NewsArtData) DataSize() []byte {
78 dataLen := make([]byte, 2)
79 binary.BigEndian.PutUint16(dataLen, uint16(len(art.Data)))
84 type NewsArtListData struct {
85 ID []byte `yaml:"ID"` // Size 4
86 Name []byte `yaml:"Name"`
87 Description []byte `yaml:"Description"` // not used?
88 NewsArtList []byte // List of articles Optional (if article count > 0)
92 func (nald *NewsArtListData) Payload() []byte {
93 count := make([]byte, 4)
94 binary.BigEndian.PutUint32(count, uint32(nald.Count))
96 out := append(nald.ID, count...)
97 out = append(out, []byte{uint8(len(nald.Name))}...)
98 out = append(out, nald.Name...)
99 out = append(out, []byte{uint8(len(nald.Description))}...)
100 out = append(out, nald.Description...)
101 out = append(out, nald.NewsArtList...)
106 // NewsArtList is a summarized version of a NewArtData record for display in list view
107 type NewsArtList struct {
109 TimeStamp []byte // Year (2 bytes), milliseconds (2 bytes) and seconds (4 bytes)
110 ParentID []byte // Size 4
111 Flags []byte // Size 4
112 FlavorCount []byte // Size 2
114 Title []byte // string
116 // Poster Poster string
118 FlavorList []NewsFlavorList
119 // Flavor list… Optional (if flavor count > 0)
120 ArticleSize []byte // Size 2
123 type byID []NewsArtList
125 func (s byID) Len() int {
128 func (s byID) Swap(i, j int) {
129 s[i], s[j] = s[j], s[i]
131 func (s byID) Less(i, j int) bool {
132 return binary.BigEndian.Uint32(s[i].ID) < binary.BigEndian.Uint32(s[j].ID)
135 func (nal *NewsArtList) Payload() []byte {
136 out := append(nal.ID, nal.TimeStamp...)
137 out = append(out, nal.ParentID...)
138 out = append(out, nal.Flags...)
140 out = append(out, []byte{0, 1}...)
142 out = append(out, []byte{uint8(len(nal.Title))}...)
143 out = append(out, nal.Title...)
144 out = append(out, []byte{uint8(len(nal.Poster))}...)
145 out = append(out, nal.Poster...)
146 out = append(out, []byte{0x0a, 0x74, 0x65, 0x78, 0x74, 0x2f, 0x70, 0x6c, 0x61, 0x69, 0x6e}...) // TODO: wat?
147 out = append(out, nal.ArticleSize...)
152 type NewsFlavorList struct {
154 // Flavor text size MIME type string
158 func (newscat *NewsCategoryListData15) MarshalBinary() (data []byte, err error) {
159 count := make([]byte, 2)
160 binary.BigEndian.PutUint16(count, uint16(len(newscat.Articles)+len(newscat.SubCats)))
162 out := append(newscat.Type, count...)
164 if bytes.Equal(newscat.Type, []byte{0, 3}) {
165 // Generate a random GUID // TODO: does this need to be random?
166 b := make([]byte, 16)
167 _, err := rand.Read(b)
172 out = append(out, b...) // GUID
173 out = append(out, []byte{0, 0, 0, 1}...) // Add SN (TODO: not sure what this is)
174 out = append(out, []byte{0, 0, 0, 2}...) // Delete SN (TODO: not sure what this is)
177 out = append(out, newscat.nameLen()...)
178 out = append(out, []byte(newscat.Name)...)
183 // ReadNewsCategoryListData parses a byte slice into a NewsCategoryListData15 struct
184 // For use on the client side
185 func ReadNewsCategoryListData(payload []byte) NewsCategoryListData15 {
186 ncld := NewsCategoryListData15{
191 if bytes.Equal(ncld.Type, []byte{0, 3}) {
192 ncld.GUID = payload[4:20]
193 ncld.AddSN = payload[20:24]
194 ncld.AddSN = payload[24:28]
195 ncld.Name = string(payload[29:])
197 ncld.Name = string(payload[5:])
203 func (newscat *NewsCategoryListData15) nameLen() []byte {
204 return []byte{uint8(len(newscat.Name))}
207 // TODO: re-implement as bufio.Scanner interface
208 func ReadNewsPath(newsPath []byte) []string {
209 if len(newsPath) == 0 {
212 pathCount := binary.BigEndian.Uint16(newsPath[0:2])
214 pathData := newsPath[2:]
217 for i := uint16(0); i < pathCount; i++ {
218 pathLen := pathData[2]
219 paths = append(paths, string(pathData[3:3+pathLen]))
221 pathData = pathData[pathLen+3:]
227 func (s *Server) GetNewsCatByPath(paths []string) map[string]NewsCategoryListData15 {
228 cats := s.ThreadedNews.Categories
229 for _, path := range paths {
230 cats = cats[path].SubCats