2019-01-05 21:35:20 +01:00
|
|
|
package mapblockparser
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2019-02-15 09:08:22 +01:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
2019-01-21 11:31:50 +01:00
|
|
|
"mapserver/coords"
|
2019-01-21 13:45:35 +01:00
|
|
|
"strconv"
|
2019-01-05 21:35:20 +01:00
|
|
|
)
|
|
|
|
|
2019-02-06 08:38:08 +01:00
|
|
|
func Parse(data []byte, mtime int64, pos *coords.MapBlockCoords) (*MapBlock, error) {
|
2019-01-05 21:35:20 +01:00
|
|
|
if len(data) == 0 {
|
|
|
|
return nil, errors.New("no data")
|
|
|
|
}
|
|
|
|
|
2019-02-15 08:36:48 +01:00
|
|
|
timer := prometheus.NewTimer(parseDuration)
|
2019-02-15 09:08:22 +01:00
|
|
|
defer timer.ObserveDuration()
|
2019-02-15 08:36:48 +01:00
|
|
|
|
2019-01-18 15:18:42 +01:00
|
|
|
mapblock := NewMapblock()
|
|
|
|
mapblock.Mtime = mtime
|
2019-01-21 13:45:35 +01:00
|
|
|
mapblock.Pos = pos
|
2019-01-18 15:18:42 +01:00
|
|
|
mapblock.Size = len(data)
|
|
|
|
|
2019-01-05 21:35:20 +01:00
|
|
|
// version
|
|
|
|
mapblock.Version = data[0]
|
|
|
|
|
2019-02-01 13:54:59 +01:00
|
|
|
if mapblock.Version < 25 || mapblock.Version > 28 {
|
2019-02-01 14:50:39 +01:00
|
|
|
return nil, errors.New("mapblock-version not supported: " + strconv.Itoa(int(mapblock.Version)))
|
2019-02-01 13:54:59 +01:00
|
|
|
}
|
|
|
|
|
2019-01-05 21:35:20 +01:00
|
|
|
//flags
|
|
|
|
flags := data[1]
|
|
|
|
mapblock.Underground = (flags & 0x01) == 0x01
|
|
|
|
|
2019-04-02 16:23:12 +02:00
|
|
|
var offset int
|
|
|
|
|
|
|
|
if mapblock.Version >= 27 {
|
|
|
|
offset = 4
|
|
|
|
} else {
|
|
|
|
//u16 lighting_complete not present
|
|
|
|
offset = 2
|
|
|
|
}
|
|
|
|
|
|
|
|
content_width := data[offset]
|
|
|
|
params_width := data[offset+1]
|
2019-01-06 21:00:24 +01:00
|
|
|
|
|
|
|
if content_width != 2 {
|
|
|
|
return nil, errors.New("content_width = " + strconv.Itoa(int(content_width)))
|
|
|
|
}
|
|
|
|
|
|
|
|
if params_width != 2 {
|
|
|
|
return nil, errors.New("params_width = " + strconv.Itoa(int(params_width)))
|
|
|
|
}
|
|
|
|
|
2019-01-05 21:35:20 +01:00
|
|
|
//mapdata (blocks)
|
2019-02-01 13:54:59 +01:00
|
|
|
if mapblock.Version >= 27 {
|
|
|
|
offset = 6
|
|
|
|
|
|
|
|
} else {
|
|
|
|
offset = 4
|
2019-02-01 14:50:39 +01:00
|
|
|
|
2019-02-01 13:54:59 +01:00
|
|
|
}
|
2019-01-05 21:35:20 +01:00
|
|
|
|
|
|
|
//metadata
|
2019-01-22 13:17:30 +01:00
|
|
|
count, err := parseMapdata(mapblock, data[offset:])
|
2019-01-05 21:35:20 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
offset += count
|
|
|
|
|
2019-01-22 13:17:30 +01:00
|
|
|
count, err = parseMetadata(mapblock, data[offset:])
|
2019-01-05 21:35:20 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-01-07 19:53:05 +01:00
|
|
|
offset += count
|
|
|
|
|
|
|
|
//static objects
|
|
|
|
|
|
|
|
offset++ //static objects version
|
|
|
|
staticObjectsCount := readU16(data, offset)
|
|
|
|
offset += 2
|
|
|
|
for i := 0; i < staticObjectsCount; i++ {
|
|
|
|
offset += 13
|
|
|
|
dataSize := readU16(data, offset)
|
|
|
|
offset += dataSize + 2
|
|
|
|
}
|
|
|
|
|
|
|
|
//timestamp
|
|
|
|
offset += 4
|
|
|
|
|
|
|
|
//mapping version
|
|
|
|
offset++
|
|
|
|
|
|
|
|
numMappings := readU16(data, offset)
|
|
|
|
offset += 2
|
|
|
|
for i := 0; i < numMappings; i++ {
|
|
|
|
nodeId := readU16(data, offset)
|
|
|
|
offset += 2
|
|
|
|
|
|
|
|
nameLen := readU16(data, offset)
|
|
|
|
offset += 2
|
|
|
|
|
|
|
|
blockName := string(data[offset : offset+nameLen])
|
|
|
|
offset += nameLen
|
|
|
|
|
|
|
|
mapblock.BlockMapping[nodeId] = blockName
|
|
|
|
}
|
|
|
|
|
2019-02-15 08:36:48 +01:00
|
|
|
parsedMapBlocks.Inc()
|
2019-01-22 13:17:30 +01:00
|
|
|
return mapblock, nil
|
2019-01-05 21:35:20 +01:00
|
|
|
}
|