Unmarshaling XML в Go

У меня есть следующий XML-файл:

<pinnacle_line_feed>
    <PinnacleFeedTime>1439954818555</PinnacleFeedTime>
    <lastContest>34317132</lastContest>
    <lastGame>218491030</lastGame>
    <events>
        <event>
            <event_datetimeGMT>2015-08-21 09:50</event_datetimeGMT>
            <gamenumber>483406220</gamenumber>
            <sporttype>Aussie Rules</sporttype>
            <league>AFL</league>
            <IsLive>No</IsLive>
            <participants>
                <participant>
                    <participant_name>Hawthorn Hawks</participant_name>
                    <contestantnum>1251</contestantnum>
                    <rotnum>1251</rotnum>
                    <visiting_home_draw>Visiting</visiting_home_draw>
                    </participant>
                <participant>
                    <participant_name>Port Adelaide Power</participant_name>
                    <contestantnum>1252</contestantnum>
                    <rotnum>1252</rotnum>
                    <visiting_home_draw>Home</visiting_home_draw>
                </participant>
            </participants>
            <periods></periods>
        </event>
    </events>
</pinnacle_line_feed>

Я пытаюсь проанализировать это с Голангом, и пока написал следующее:

package main
import (
 "fmt"
 "encoding/xml"
)

type Participant struct {
 XMLName            xml.Name `xml:"participant"`
 participant_name   string `xml:"participant_name"`
 contestantnum      int `xml:"contestantnum"`
 rotnum             int `xml:"rotnum"`
 visiting_home_draw string `xml:"visiting_home_draw"`
}

type Event struct {
 XMLName           xml.Name `xml:"event"`
 event_datetimeGMT string `xml:"event_datetimeGMT"`
 gamenumber        string `xml:"gamenumber"`
 sporttype         string `xml:"sporttype"`
 league            string `xml:"league"`
 IsLive            string `xml:"IsLive"`
 Participant       []Participant `xml:"participant"`
}

type Events struct {
 XMLName xml.Name `xml:"events"`
 Event   []Event `xml:"event"`
}

type Pinnacle_Line_Feed struct {
 XMLName          xml.Name `xml:"pinnacle_line_feed"`
 PinnacleFeedTime string `xml:"PinnacleFeedTime"`
 lastContest      string `xml:"lastContest"`
 lastGame         string `xml:"lastGame"`
 Events           []Events `xml:"events"`
}

Честно говоря, я довольно много исследовал глубокий вложенный xml-анализ в Golang, и я не нашел много, чтобы помочь с Unmarshal-частью этого кода. В идеале код должен возвращать что-то похожее на словарь Python, например,

{event_datetimeGMT: 2015-08-21, gamenumber: 483406220,..., visit_home_draw: "Домой"}

РЕДАКТИРОВАТЬ:

Основываясь на комментариях Николаса ниже, я добавил следующее. Это работает без ошибок, но дает пустой результат.

 func main() {
         pinny_xml := `
         <pinnacle_line_feed>
            <PinnacleFeedTime>1439954818555</PinnacleFeedTime>
            <lastContest>34317132</lastContest>
            <lastGame>218491030</lastGame>
            <events>
                <event>
                    <event_datetimeGMT>2015-08-21 09:50</event_datetimeGMT>
                    <gamenumber>483406220</gamenumber>
                    <sporttype>Aussie Rules</sporttype>
                    <league>AFL</league>
                    <IsLive>No</IsLive>
                    <participants>
                        <participant>
                            <participant_name>Hawthorn Hawks</participant_name>
                            <contestantnum>1251</contestantnum>
                            <rotnum>1251</rotnum>
                            <visiting_home_draw>Visiting</visiting_home_draw>
                            </participant>
                        <participant>
                            <participant_name>Port Adelaide Power</participant_name>
                            <contestantnum>1252</contestantnum>
                            <rotnum>1252</rotnum>
                            <visiting_home_draw>Home</visiting_home_draw>
                        </participant>
                    </participants>
                    <periods></periods>
                </event>
            </events>
        </pinnacle_line_feed>
        `

        xmlReader := bytes.NewReader([]byte(pinny_xml))
        yourPinnacleLineFeed := new(Pinnacle_Line_Feed)
        if err := xml.NewDecoder(xmlReader).Decode(yourPinnacleLineFeed); err != nil {
            return // or log.Panic(err.Error()) if in main
        }
}

2 ответа

Решение

Вы можете сделать это так:

xmlReader := bytes.NewReader([]byte(your_xml_as_a_string_here))
yourPinnacleLineFeed := new(Pinnacle_Line_Feed)
if err := xml.NewDecoder(xmlReader).Decode(yourPinnacleLineFeed); err != nil {
    return // or log.Panic(err.Error()) if in main
}

Это если ваш xml-файл является строкой. Если вы получаете его из Интернета (как, например, текст ответа http), resp.Body уже удовлетворит Reader, так что вы можете пропустить первую строку. Если вы открываете настоящий файл в ОС, вы также можете открыть его как Reader, тоже самое.

РЕДАКТИРОВАТЬ: еще две вещи:

  • Вы можете вкладывать структуры и удалять поля xml.Name для простоты и ясности
  • Я также заметил, что вы забыли уровень участников в структурах, что привело к тому, что ни один из участников не будет разоблачен

Вот более простая версия, которая работает, с дополнительной функцией, чтобы проверить, что у вас внутри результатов:

package main

import (
    "bytes"
    "encoding/json"
    "encoding/xml"
    "fmt"
    "log"
)

type Pinnacle_Line_Feed struct {
    PinnacleFeedTime string `xml:"PinnacleFeedTime"`
    LastContest      string `xml:"lastContest"`
    LastGame         string `xml:"lastGame"`
    Events           struct {
        Event []struct {
            Event_datetimeGMT string `xml:"event_datetimeGMT"`
            Gamenumber        string `xml:"gamenumber"`
            Sporttype         string `xml:"sporttype"`
            League            string `xml:"league"`
            IsLive            string `xml:"IsLive"`
            Participants      struct {
                Participant []struct {
                    Participant_name   string `xml:"participant_name"`
                    Contestantnum      int    `xml:"contestantnum"`
                    Rotnum             int    `xml:"rotnum"`
                    Visiting_home_draw string `xml:"visiting_home_draw"`
                } `xml:"participant"`
            } `xml:"participants"`
        } `xml:"event"`
    } `xml:"events"`
}

func main() {
    pinny_xml := `
         <pinnacle_line_feed>
            <PinnacleFeedTime>1439954818555</PinnacleFeedTime>
            <lastContest>34317132</lastContest>
            <lastGame>218491030</lastGame>
            <events>
                <event>
                    <event_datetimeGMT>2015-08-21 09:50</event_datetimeGMT>
                    <gamenumber>483406220</gamenumber>
                    <sporttype>Aussie Rules</sporttype>
                    <league>AFL</league>
                    <IsLive>No</IsLive>
                    <participants>
                        <participant>
                            <participant_name>Hawthorn Hawks</participant_name>
                            <contestantnum>1251</contestantnum>
                            <rotnum>1251</rotnum>
                            <visiting_home_draw>Visiting</visiting_home_draw>
                        </participant>
                        <participant>
                            <participant_name>Port Adelaide Power</participant_name>
                            <contestantnum>1252</contestantnum>
                            <rotnum>1252</rotnum>
                            <visiting_home_draw>Home</visiting_home_draw>
                        </participant>
                    </participants>
                    <periods></periods>
                </event>
            </events>
        </pinnacle_line_feed>
    `

    xmlReader := bytes.NewReader([]byte(pinny_xml))
    yourPinnacleLineFeed := new(Pinnacle_Line_Feed)
    if err := xml.NewDecoder(xmlReader).Decode(yourPinnacleLineFeed); err != nil {
        log.Panic(err.Error())
    }

    printX(yourPinnacleLineFeed)
}

func printX(x interface{}) (err error) {
    var xBytes []byte
    xBytes, err = json.MarshalIndent(x, "", "  ")
    if err != nil {
        return
    }
    fmt.Println(string(xBytes))
    return
}

Вам нужно прописать все поля в ваших структурах. Для правильной работы декодеру xml необходимо экспортировать поля. Идите на игровую площадку здесь.

Другие вопросы по тегам