Skip to content

Commit 95d5725

Browse files
authored
Add unmarshalling of Links (#4)
* Add links support * Handle single relationship only link
1 parent 166ae20 commit 95d5725

File tree

5 files changed

+244
-6
lines changed

5 files changed

+244
-6
lines changed

constants.go

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ const (
77
annotationClientID = "client-id"
88
annotationAttribute = "attr"
99
annotationRelation = "relation"
10+
annotationLinks = "links"
1011
annotationOmitEmpty = "omitempty"
1112
annotationISO8601 = "iso8601"
1213
annotationRFC3339 = "rfc3339"

models_test.go

+10-4
Original file line numberDiff line numberDiff line change
@@ -56,10 +56,16 @@ type Post struct {
5656
}
5757

5858
type Comment struct {
59-
ID int `jsonapi:"primary,comments"`
60-
ClientID string `jsonapi:"client-id"`
61-
PostID int `jsonapi:"attr,post_id"`
62-
Body string `jsonapi:"attr,body"`
59+
ID int `jsonapi:"primary,comments"`
60+
ClientID string `jsonapi:"client-id"`
61+
PostID int `jsonapi:"attr,post_id"`
62+
Body string `jsonapi:"attr,body"`
63+
Impressions *Impressions `jsonapi:"relation,impressions"`
64+
URL *Links `jsonapi:"links,omitempty"`
65+
}
66+
67+
type Impressions struct {
68+
URL *Links `jsonapi:"links,omitempty"`
6369
}
6470

6571
type Book struct {

request.go

+82-2
Original file line numberDiff line numberDiff line change
@@ -274,6 +274,7 @@ func unmarshalNode(data *Node, model reflect.Value, included *map[string]*Node)
274274
json.NewDecoder(buf).Decode(relationship)
275275

276276
data := relationship.Data
277+
links := relationship.Links
277278
models := reflect.New(fieldValue.Type()).Elem()
278279

279280
for _, n := range data {
@@ -291,6 +292,37 @@ func unmarshalNode(data *Node, model reflect.Value, included *map[string]*Node)
291292
models = reflect.Append(models, m)
292293
}
293294

295+
if links != nil {
296+
var assignedValue bool
297+
linkModel := reflect.New(fieldValue.Type().Elem().Elem())
298+
linkModelValue := linkModel.Elem()
299+
linkModelType := linkModelValue.Type()
300+
301+
for i := 0; i < linkModelValue.NumField(); i++ {
302+
fieldType := linkModelType.Field(i)
303+
tag := fieldType.Tag.Get("jsonapi")
304+
if tag == "" {
305+
continue
306+
}
307+
fieldValue := linkModelValue.Field(i)
308+
args := strings.Split(tag, ",")
309+
annotation := args[0]
310+
if annotation == annotationLinks {
311+
value, err := unmarshalAttribute(*links, args, fieldType, fieldValue)
312+
if err != nil {
313+
er = err
314+
break
315+
}
316+
317+
assign(fieldValue, value)
318+
assignedValue = true
319+
}
320+
}
321+
if assignedValue {
322+
models = reflect.Append(models, linkModel)
323+
}
324+
}
325+
294326
fieldValue.Set(models)
295327
} else {
296328
// to-one relationships
@@ -309,13 +341,23 @@ func unmarshalNode(data *Node, model reflect.Value, included *map[string]*Node)
309341
relationship can have a data node set to null (e.g. to disassociate the relationship)
310342
so unmarshal and set fieldValue only if data obj is not null
311343
*/
312-
if relationship.Data == nil {
344+
345+
if relationship.Data == nil && relationship.Links == nil {
313346
continue
314347
}
315348

349+
node := &Node{}
350+
if relationship.Data != nil {
351+
node = relationship.Data
352+
}
353+
354+
if relationship.Links != nil {
355+
node.Links = relationship.Links
356+
}
357+
316358
m := reflect.New(fieldValue.Type().Elem())
317359
if err := unmarshalNode(
318-
fullNode(relationship.Data, included),
360+
fullNode(node, included),
319361
m,
320362
included,
321363
); err != nil {
@@ -324,9 +366,23 @@ func unmarshalNode(data *Node, model reflect.Value, included *map[string]*Node)
324366
}
325367

326368
fieldValue.Set(m)
369+
}
370+
371+
} else if annotation == annotationLinks {
372+
links := data.Links
373+
if links == nil {
374+
continue
375+
}
327376

377+
structField := fieldType
378+
value, err := unmarshalAttribute(*links, args, structField, fieldValue)
379+
if err != nil {
380+
er = err
381+
break
328382
}
329383

384+
assign(fieldValue, value)
385+
330386
} else {
331387
er = fmt.Errorf(unsupportedStructTagMsg, annotation)
332388
}
@@ -409,6 +465,11 @@ func unmarshalAttribute(
409465
return
410466
}
411467

468+
if fieldValue.Type() == reflect.TypeOf(&Links{}) {
469+
value, err = handleLinks(attribute, args, fieldValue)
470+
return
471+
}
472+
412473
// Handle field of type struct
413474
if fieldValue.Type().Kind() == reflect.Struct {
414475
value, err = handleStruct(attribute, fieldValue)
@@ -466,6 +527,25 @@ func handleMapStringSlice(attribute interface{}, fieldValue reflect.Value) (refl
466527
return reflect.ValueOf(values), nil
467528
}
468529

530+
func handleLinks(attribute interface{}, args []string, fieldValue reflect.Value) (reflect.Value, error) {
531+
b, err := json.Marshal(attribute)
532+
if err != nil {
533+
return reflect.Value{}, err
534+
}
535+
var v interface{}
536+
err = json.Unmarshal(b, &v)
537+
if err != nil {
538+
return reflect.Value{}, err
539+
}
540+
541+
var values = map[string]interface{}{}
542+
for k, v := range v.(map[string]interface{}) {
543+
values[k] = v.(interface{})
544+
}
545+
546+
return reflect.ValueOf(values), nil
547+
}
548+
469549
func handleTime(attribute interface{}, args []string, fieldValue reflect.Value) (reflect.Value, error) {
470550
var isIso8601 bool
471551
var isRFC3339 bool

request_test.go

+144
Original file line numberDiff line numberDiff line change
@@ -772,6 +772,150 @@ func TestUnmarshalManyPayload(t *testing.T) {
772772
}
773773
}
774774

775+
func TestOnePayload_WithRelationLinks(t *testing.T) {
776+
selfLink := "http://example.com/articles/1/relationships/author"
777+
relatedLink := "http://example.com/articles/1/author"
778+
sample := map[string]interface{}{
779+
"data": map[string]interface{}{
780+
"type": "posts",
781+
"id": "1",
782+
"attributes": map[string]interface{}{
783+
"body": "First",
784+
"title": "Post",
785+
},
786+
"relationships": map[string]interface{}{
787+
"comments": map[string]interface{}{
788+
"links": map[string]string{
789+
"self": selfLink,
790+
"related": relatedLink,
791+
},
792+
},
793+
},
794+
},
795+
}
796+
797+
data, err := json.Marshal(sample)
798+
if err != nil {
799+
t.Fatal(err)
800+
}
801+
in := bytes.NewReader(data)
802+
803+
out := new(Post)
804+
805+
if err := UnmarshalPayload(in, out); err != nil {
806+
t.Fatal(err)
807+
}
808+
809+
if len(out.Comments) == 0 {
810+
t.Fatal("Was expecting non-empty Comments")
811+
}
812+
813+
if out.Comments[0].URL == nil {
814+
t.Fatal("Was expecting a non nil ptr Link field")
815+
}
816+
817+
links := *out.Comments[0].URL
818+
if links["self"] != selfLink {
819+
t.Fatalf("Was expecting self Link to equal %s, but got %s", selfLink, links["self"])
820+
}
821+
822+
if links["related"] != relatedLink {
823+
t.Fatalf("Was expecting related Link to equal %s, but got %s", relatedLink, links["related"])
824+
}
825+
}
826+
827+
func TestOnePayload_WithLinks(t *testing.T) {
828+
selfLink := "http://example.com/articles/1/relationships/author"
829+
relatedLink := "http://example.com/articles/1/author"
830+
sample := map[string]interface{}{
831+
"data": map[string]interface{}{
832+
"type": "comments",
833+
"id": "1",
834+
"attributes": map[string]interface{}{
835+
"body": "First",
836+
"title": "Post",
837+
},
838+
"links": map[string]string{
839+
"self": selfLink,
840+
"related": relatedLink,
841+
},
842+
},
843+
}
844+
845+
data, err := json.Marshal(sample)
846+
if err != nil {
847+
t.Fatal(err)
848+
}
849+
in := bytes.NewReader(data)
850+
851+
out := new(Comment)
852+
853+
if err := UnmarshalPayload(in, out); err != nil {
854+
t.Fatal(err)
855+
}
856+
857+
if out.URL == nil {
858+
t.Fatal("Was expecting a non nil ptr Link field")
859+
}
860+
861+
links := *out.URL
862+
if links["self"] != selfLink {
863+
t.Fatalf("Was expecting self Link to equal %s, but got %s", selfLink, links["self"])
864+
}
865+
866+
if links["related"] != relatedLink {
867+
t.Fatalf("Was expecting related Link to equal %s, but got %s", relatedLink, links["related"])
868+
}
869+
}
870+
871+
func TestOnePayload_RelationshipLinks(t *testing.T) {
872+
selfLink := "http://example.com/articles/1/relationships/author"
873+
relatedLink := "http://example.com/articles/1/author"
874+
sample := map[string]interface{}{
875+
"data": map[string]interface{}{
876+
"type": "comments",
877+
"id": "1",
878+
"attributes": map[string]interface{}{
879+
"body": "First",
880+
"title": "Post",
881+
},
882+
"relationships": map[string]interface{}{
883+
"impressions": map[string]interface{}{
884+
"links": map[string]string{
885+
"self": selfLink,
886+
"related": relatedLink,
887+
},
888+
},
889+
},
890+
},
891+
}
892+
893+
data, err := json.Marshal(sample)
894+
if err != nil {
895+
t.Fatal(err)
896+
}
897+
in := bytes.NewReader(data)
898+
899+
out := new(Comment)
900+
901+
if err := UnmarshalPayload(in, out); err != nil {
902+
t.Fatal(err)
903+
}
904+
905+
if out.Impressions == nil {
906+
t.Fatal("Was expecting a non nil ptr Link field")
907+
}
908+
909+
links := *out.Impressions.URL
910+
if links["self"] != selfLink {
911+
t.Fatalf("Was expecting self Link to equal %s, but got %s", selfLink, links["self"])
912+
}
913+
914+
if links["related"] != relatedLink {
915+
t.Fatalf("Was expecting related Link to equal %s, but got %s", relatedLink, links["related"])
916+
}
917+
}
918+
775919
func TestManyPayload_withLinks(t *testing.T) {
776920
firstPageURL := "http://somesite.com/movies?page[limit]=50&page[offset]=50"
777921
prevPageURL := "http://somesite.com/movies?page[limit]=50&page[offset]=0"

response.go

+7
Original file line numberDiff line numberDiff line change
@@ -441,6 +441,13 @@ func visitModelNode(model interface{}, included *map[string]*Node,
441441
}
442442
}
443443
}
444+
} else if annotation == annotationLinks {
445+
// This is for marshalling.
446+
// This is left blank intentionally as the handling
447+
// of `jsonapi:"links"` is done below via the
448+
// Linkable interface.
449+
// And the logic for Marshalling links should be done
450+
// via that interface.
444451

445452
} else {
446453
er = ErrBadJSONAPIStructTag

0 commit comments

Comments
 (0)