216 lines
4.7 KiB
Go
216 lines
4.7 KiB
Go
|
package plist
|
||
|
|
||
|
import (
|
||
|
"encoding/base64"
|
||
|
"encoding/xml"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"runtime"
|
||
|
"strings"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
type xmlPlistParser struct {
|
||
|
reader io.Reader
|
||
|
xmlDecoder *xml.Decoder
|
||
|
whitespaceReplacer *strings.Replacer
|
||
|
ntags int
|
||
|
}
|
||
|
|
||
|
func (p *xmlPlistParser) parseDocument() (pval cfValue, parseError error) {
|
||
|
defer func() {
|
||
|
if r := recover(); r != nil {
|
||
|
if _, ok := r.(runtime.Error); ok {
|
||
|
panic(r)
|
||
|
}
|
||
|
if _, ok := r.(invalidPlistError); ok {
|
||
|
parseError = r.(error)
|
||
|
} else {
|
||
|
// Wrap all non-invalid-plist errors.
|
||
|
parseError = plistParseError{"XML", r.(error)}
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
for {
|
||
|
if token, err := p.xmlDecoder.Token(); err == nil {
|
||
|
if element, ok := token.(xml.StartElement); ok {
|
||
|
pval = p.parseXMLElement(element)
|
||
|
if p.ntags == 0 {
|
||
|
panic(invalidPlistError{"XML", errors.New("no elements encountered")})
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
} else {
|
||
|
// The first XML parse turned out to be invalid:
|
||
|
// we do not have an XML property list.
|
||
|
panic(invalidPlistError{"XML", err})
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (p *xmlPlistParser) parseXMLElement(element xml.StartElement) cfValue {
|
||
|
var charData xml.CharData
|
||
|
switch element.Name.Local {
|
||
|
case "plist":
|
||
|
p.ntags++
|
||
|
for {
|
||
|
token, err := p.xmlDecoder.Token()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.EndElement); ok && el.Name.Local == "plist" {
|
||
|
break
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.StartElement); ok {
|
||
|
return p.parseXMLElement(el)
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
case "string":
|
||
|
p.ntags++
|
||
|
err := p.xmlDecoder.DecodeElement(&charData, &element)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
return cfString(charData)
|
||
|
case "integer":
|
||
|
p.ntags++
|
||
|
err := p.xmlDecoder.DecodeElement(&charData, &element)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
s := string(charData)
|
||
|
if len(s) == 0 {
|
||
|
panic(errors.New("invalid empty <integer/>"))
|
||
|
}
|
||
|
|
||
|
if s[0] == '-' {
|
||
|
sn, base := unsignedGetBase(s[1:])
|
||
|
n := mustParseInt("-"+sn, base, 64)
|
||
|
return &cfNumber{signed: true, value: uint64(n)}
|
||
|
}
|
||
|
sn, base := unsignedGetBase(s)
|
||
|
n := mustParseUint(sn, base, 64)
|
||
|
return &cfNumber{signed: false, value: n}
|
||
|
case "real":
|
||
|
p.ntags++
|
||
|
err := p.xmlDecoder.DecodeElement(&charData, &element)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
n := mustParseFloat(string(charData), 64)
|
||
|
return &cfReal{wide: true, value: n}
|
||
|
case "true", "false":
|
||
|
p.ntags++
|
||
|
p.xmlDecoder.Skip()
|
||
|
|
||
|
b := element.Name.Local == "true"
|
||
|
return cfBoolean(b)
|
||
|
case "date":
|
||
|
p.ntags++
|
||
|
err := p.xmlDecoder.DecodeElement(&charData, &element)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
t, err := time.ParseInLocation(time.RFC3339, string(charData), time.UTC)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
return cfDate(t)
|
||
|
case "data":
|
||
|
p.ntags++
|
||
|
err := p.xmlDecoder.DecodeElement(&charData, &element)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
str := p.whitespaceReplacer.Replace(string(charData))
|
||
|
|
||
|
l := base64.StdEncoding.DecodedLen(len(str))
|
||
|
bytes := make([]uint8, l)
|
||
|
l, err = base64.StdEncoding.Decode(bytes, []byte(str))
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
return cfData(bytes[:l])
|
||
|
case "dict":
|
||
|
p.ntags++
|
||
|
var key *string
|
||
|
keys := make([]string, 0, 32)
|
||
|
values := make([]cfValue, 0, 32)
|
||
|
for {
|
||
|
token, err := p.xmlDecoder.Token()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.EndElement); ok && el.Name.Local == "dict" {
|
||
|
if key != nil {
|
||
|
panic(errors.New("missing value in dictionary"))
|
||
|
}
|
||
|
break
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.StartElement); ok {
|
||
|
if el.Name.Local == "key" {
|
||
|
var k string
|
||
|
p.xmlDecoder.DecodeElement(&k, &el)
|
||
|
key = &k
|
||
|
} else {
|
||
|
if key == nil {
|
||
|
panic(errors.New("missing key in dictionary"))
|
||
|
}
|
||
|
keys = append(keys, *key)
|
||
|
values = append(values, p.parseXMLElement(el))
|
||
|
key = nil
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if len(keys) == 1 && keys[0] == "CF$UID" && len(values) == 1 {
|
||
|
if integer, ok := values[0].(*cfNumber); ok {
|
||
|
return cfUID(integer.value)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return &cfDictionary{keys: keys, values: values}
|
||
|
case "array":
|
||
|
p.ntags++
|
||
|
values := make([]cfValue, 0, 10)
|
||
|
for {
|
||
|
token, err := p.xmlDecoder.Token()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.EndElement); ok && el.Name.Local == "array" {
|
||
|
break
|
||
|
}
|
||
|
|
||
|
if el, ok := token.(xml.StartElement); ok {
|
||
|
values = append(values, p.parseXMLElement(el))
|
||
|
}
|
||
|
}
|
||
|
return &cfArray{values}
|
||
|
}
|
||
|
err := fmt.Errorf("encountered unknown element %s", element.Name.Local)
|
||
|
if p.ntags == 0 {
|
||
|
// If out first XML tag is invalid, it might be an openstep data element, ala <abab> or <0101>
|
||
|
panic(invalidPlistError{"XML", err})
|
||
|
}
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
func newXMLPlistParser(r io.Reader) *xmlPlistParser {
|
||
|
return &xmlPlistParser{r, xml.NewDecoder(r), strings.NewReplacer("\t", "", "\n", "", " ", "", "\r", ""), 0}
|
||
|
}
|