XML Marshalling генерирует XML без рута - PullRequest
0 голосов
/ 01 мая 2019

Я работаю над отдыхом API в Голанге и смог маршалировать данные в XML и JSON. У структуры, которая у меня есть, есть несколько записей, но xml marshalling не добавляет рут для этих записей Поэтому XML недействителен.

Я видел некоторые вопросы, которые были решены путем установки XMLName, но я думаю, что это только изменит мой тип клиента на что-то другое.

То, что я все еще мог бы попробовать, - это добавить вокруг xml маршаллированный байт и конкатинат. Но я не уверен, должен ли я так поступить.

РЕДАКТИРОВАТЬ: Очевидно, XML правильно маршалинг, так как я маршалинг структуры с несколькими объектами внутри. Затем Marshalling создает отдельные XML-документы для каждого объекта. Тогда возникает вопрос: с чем могут работать остальные клиенты, чтобы использовать данные, или это плохая практика - отправлять несколько XML-документов в ответе остальных? Браузеры не могут отобразить полученный xml.

Чтобы вернуть один XML-документ, мне нужно было бы поместить список в структуру, которую затем можно упорядочить. Я не уверен, как я могу инициализировать структуру с sqlx таким образом.

Добавление следующего кода не работает:

type custs struct {
    List []Customer `xml:"Customer"`
}

custs customers.[]Customer =  []Customer{}

Код:

package main

import (
    "database/sql"
    json "encoding/json"
    "encoding/xml"
    "fmt"
    "net/http"
)

type Customer struct {
    CustomerID int            `db:"CustomerId"`
    FirstName  string         `db:"FirstName"`
    LastName   string         `db:"LastName"`
    Company    sql.NullString `db:"Company"`
}

func tstSQLite(w http.ResponseWriter, r *http.Request) {

    contentType := r.Header.Get("Content-type")

    Customers := []Customer{}

    var err error

    fmt.Print("executing query")
    err = chinookdb.Select(&Customers, "SELECT CustomerId, FirstName,LastName, Company FROM customers")

    if err != nil {
        panic(err)
    }

    for _, g := range Customers {
        fmt.Println("FirstName:", g.FirstName, "LastName:", g.LastName)
    }

    var data []byte

    if contentType == "application/xml" {
        data, err = xml.Marshal(Customers)
    }

    if contentType == "application/json" || contentType != "application/xml" {
        data, err = json.Marshal(Customers)
    }


    w.Header().Set("Content-Type", "application/json")
    w.Write(data)
}

Что я получаю, так это xml:

   <Customer>
      <CustomerID>1</CustomerID>
      <FirstName>Luís</FirstName>
      <LastName>Gonçalves</LastName>
      <Company>
         <String>Embraer - Empresa Brasileira de Aeronáutica S.A.</String>
         <Valid>true</Valid>
      </Company>
   </Customer>
   <Customer>
      <CustomerID>2</CustomerID>
      <FirstName>Leonie</FirstName>
      <LastName>Köhler</LastName>
      <Company>
         <String />
         <Valid>false</Valid>
      </Company>
   </Customer>
   <Customer>
      <CustomerID>3</CustomerID>
      <FirstName>François</FirstName>
      <LastName>Tremblay</LastName>
      <Company>
         <String />
         <Valid>false</Valid>
      </Company>
   </Customer>

но я думаю, что так и должно быть:

<Customers>
   <Customer>
      <CustomerID>1</CustomerID>
      <FirstName>Luís</FirstName>
      <LastName>Gonçalves</LastName>
      <Company>
         <String>Embraer - Empresa Brasileira de Aeronáutica S.A.</String>
         <Valid>true</Valid>
      </Company>
   </Customer>
   <Customer>
      <CustomerID>2</CustomerID>
      <FirstName>Leonie</FirstName>
      <LastName>Köhler</LastName>
      <Company>
         <String />
         <Valid>false</Valid>
      </Company>
   </Customer>
   <Customer>
      <CustomerID>3</CustomerID>
      <FirstName>François</FirstName>
      <LastName>Tremblay</LastName>
      <Company>
         <String />
         <Valid>false</Valid>
      </Company>
   </Customer>
</Customers>

Выделенный JSON выглядит правильно, так как клиенты в скобках:

[
   {
      "CustomerID":1,
      "FirstName":"Luís",
      "LastName":"Gonçalves",
      "Company":{
         "String":"Embraer - Empresa Brasileira de Aeronáutica S.A.",
         "Valid":true
      }
   },
   {
      "CustomerID":2,
      "FirstName":"Leonie",
      "LastName":"Köhler",
      "Company":{
         "String":"",
         "Valid":false
      }
   }
]

1 Ответ

0 голосов
/ 01 мая 2019

Вы должны обернуть эту коллекцию клиентов в другую структуру как

package main

import (
    "encoding/xml"
    "fmt"
    "os"
)

func main() {

    type Customer struct {
        XMLName   xml.Name `xml:"Customer"`
        CustomerID int            `db:"CustomerId"`
        FirstName  string         `db:"FirstName"`
        LastName   string         `db:"LastName"`
        Company    string `db:"Company"`
    }

    type Customers struct {
        List []Customer
    }

    csts := []Customer{
        Customer{CustomerID: 1, FirstName: "John", LastName: "Doe", Company: "Demo Company"},
        Customer{CustomerID: 2, FirstName: "John2", LastName: "Doe2", Company: "Demo Company2"},
    }


    res := Customers {
        List: csts,
    }

    output, err := xml.MarshalIndent(res, "  ", "    ")
    if err != nil {
        fmt.Printf("error: %v\n", err)
    }

    os.Stdout.Write(output)
}

По умолчанию она может принимать имя конструкции.Вы также можете настроить имя, используя тип xml.Name.

...