1
1
mirror of https://github.com/wader/fq.git synced 2024-11-26 21:55:57 +03:00
fq/format/rtmp/amf0.go
2022-07-19 18:33:50 +02:00

151 lines
3.5 KiB
Go

package rtmp
// https://rtmp.veriskope.com/pdf/amf0-file-format-specification.pdf
import (
"github.com/wader/fq/format"
"github.com/wader/fq/pkg/decode"
"github.com/wader/fq/pkg/interp"
"github.com/wader/fq/pkg/scalar"
)
func init() {
interp.RegisterFormat(decode.Format{
Name: format.AMF0,
Description: "Action Message Format 0",
DecodeFn: amf0Decode,
})
}
const (
typeNumber = 0x0
typeBoolean = 0x1
typeString = 0x2
typeObject = 0x3
typeMovieClip = 0x4
typeNull = 0x5
typeUndefined = 0x6
typeReference = 0x7
typeECMAArray = 0x8
typeObjectEnd = 0x9
typeStrictArray = 0xa
typeDate = 0xb
typeLongString = 0xc
typeUnsupported = 0xd
typeRecordSet = 0xe
typeXML = 0xf
typeTypedObject = 0x10
)
var typeNames = scalar.UToSymStr{
typeNumber: "number",
typeBoolean: "boolean",
typeString: "string",
typeObject: "object",
typeMovieClip: "movie_clip",
typeNull: "null",
typeUndefined: "undefined",
typeReference: "reference",
typeECMAArray: "ecma_array",
typeObjectEnd: "object_end",
typeStrictArray: "strict_array",
typeDate: "date",
typeLongString: "long_string",
typeUnsupported: "unsupported",
typeRecordSet: "record_set",
typeXML: "xml",
typeTypedObject: "typed_object",
}
func amf0DecodeValue(d *decode.D) {
typ := d.FieldU8("type", typeNames)
switch typ {
case typeNumber:
d.FieldF64("value")
case typeBoolean:
d.FieldU8("value")
case typeString:
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
case typeObject:
d.FieldArray("value", func(d *decode.D) {
var typ uint64
for typ != typeObjectEnd {
d.FieldStruct("pair", func(d *decode.D) {
d.FieldStruct("key", func(d *decode.D) {
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
})
typ = d.PeekBits(8)
d.FieldStruct("value", amf0DecodeValue)
})
}
})
case typeNull:
d.FieldValueNil("value")
case typeUndefined:
d.FieldValueNil("value") // TODO: ?
case typeReference:
d.FieldU16("value") // TODO: index pointer
case typeECMAArray:
d.FieldU32("count")
d.FieldArray("value", func(d *decode.D) {
var typ uint64
for typ != typeObjectEnd {
d.FieldStruct("entry", func(d *decode.D) {
d.FieldStruct("key", func(d *decode.D) {
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
})
typ = d.PeekBits(8)
d.FieldStruct("value", amf0DecodeValue)
})
}
})
case typeObjectEnd:
// nop
case typeStrictArray:
length := d.FieldU32("count")
d.FieldArray("value", func(d *decode.D) {
for i := uint64(0); i < length; i++ {
d.FieldStruct("entry", amf0DecodeValue)
}
})
case typeDate:
d.FieldF64("date_time")
d.FieldS16("local_data_time_offset")
case typeLongString:
l := d.FieldU32("length")
d.FieldUTF8("value", int(l))
case typeXML:
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
case typeTypedObject:
d.FieldStruct("class_name", func(d *decode.D) {
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
})
d.FieldArray("value", func(d *decode.D) {
var typ uint64
for typ != typeObjectEnd {
d.FieldStruct("pair", func(d *decode.D) {
d.FieldStruct("key", func(d *decode.D) {
l := d.FieldU16("length")
d.FieldUTF8("value", int(l))
})
typ = d.PeekBits(8)
d.FieldStruct("value", amf0DecodeValue)
})
}
})
default:
d.Fatalf("unknown type %d", typ)
}
}
func amf0Decode(d *decode.D, _ any) any {
amf0DecodeValue(d)
return nil
}