package tests import ( "bytes" "fmt" "io" "os" "path/filepath" "reflect" "testing" "time" "unsafe" ap "github.com/mariusor/activitypub.go/activitypub" a "github.com/mariusor/activitypub.go/activitystreams" j "github.com/mariusor/activitypub.go/jsonld" ) const dir = "./mocks" var stopOnFailure = false type testPair struct { expected bool blank interface{} result interface{} } type tests map[string]testPair type visit struct { a1 unsafe.Pointer a2 unsafe.Pointer typ reflect.Type } type canErrorFunc func(format string, args ...interface{}) func assertDeepEquals(t canErrorFunc, x, y interface{}) bool { if x == nil || y == nil { return x == y } v1 := reflect.ValueOf(x) v2 := reflect.ValueOf(y) if v1.Type() != v2.Type() { t("%T != %T", x, y) return false } return deepValueEqual(t, v1, v2, make(map[visit]bool), 0) } func deepValueEqual(t canErrorFunc, v1, v2 reflect.Value, visited map[visit]bool, depth int) bool { if !v1.IsValid() || !v2.IsValid() { return v1.IsValid() == v2.IsValid() } if v1.Type() != v2.Type() { t("types differ %s != %s", v1.Type().Name(), v2.Type().Name()) return false } // We want to avoid putting more in the visited map than we need to. // For any possible reference cycle that might be encountered, // hard(t) needs to return true for at least one of the types in the cycle. hard := func(k reflect.Kind) bool { switch k { case reflect.Map, reflect.Slice, reflect.Ptr, reflect.Interface: return true } //t("Invalid type for %s", k) return false } if v1.CanAddr() && v2.CanAddr() && hard(v1.Kind()) { addr1 := unsafe.Pointer(v1.UnsafeAddr()) addr2 := unsafe.Pointer(v2.UnsafeAddr()) if uintptr(addr1) > uintptr(addr2) { // Canonicalize order to reduce number of entries in visited. // Assumes non-moving garbage collector. addr1, addr2 = addr2, addr1 } // Short circuit if references are already seen. typ := v1.Type() v := visit{addr1, addr2, typ} if visited[v] { return true } // Remember for later. visited[v] = true } switch v1.Kind() { case reflect.Array: for i := 0; i < v1.Len(); i++ { if !deepValueEqual(t, v1.Index(i), v2.Index(i), visited, depth+1) { t("Arrays not equal at index %d %s %s", i, v1.Index(i), v2.Index(i)) return false } } return true case reflect.Slice: if v1.IsNil() != v2.IsNil() { t("One of the slices is not nil %s[%d] vs %s[%d]", v1.Type().Name(), v1.Len(), v2.Type().Name(), v2.Len()) return false } if v1.Len() != v2.Len() { t("Slices lengths are different %s[%d] vs %s[%d]", v1.Type().Name(), v1.Len(), v2.Type().Name(), v2.Len()) return false } if v1.Pointer() == v2.Pointer() { return true } for i := 0; i < v1.Len(); i++ { if !deepValueEqual(t, v1.Index(i), v2.Index(i), visited, depth+1) { t("Slices elements at pos %d are not equal %#v vs %#v", i, v1.Index(i), v2.Index(i)) return false } } return true case reflect.Interface: if v1.IsNil() || v2.IsNil() { if v1.IsNil() == v2.IsNil() { return true } var isNil1, isNil2 string if v1.IsNil() { isNil1 = "is" } else { isNil1 = "isn't" } if v2.IsNil() { isNil2 = "is" } else { isNil2 = "isn't" } t("Interface %q %s nil and %q %s nil", v1.Type().Name(), isNil1, v2.Type().Name(), isNil2) return false } return deepValueEqual(t, v1.Elem(), v2.Elem(), visited, depth+1) case reflect.Ptr: if v1.Pointer() == v2.Pointer() { return true } return deepValueEqual(t, v1.Elem(), v2.Elem(), visited, depth+1) case reflect.Struct: for i, n := 0, v1.NumField(); i < n; i++ { if !deepValueEqual(t, v1.Field(i), v2.Field(i), visited, depth+1) { t("Struct fields at pos %d %q:%q and %q:%q are not deeply equal", i, v1.Type().Field(i).Name, v1.Field(i).Type().Name(), v2.Type().Field(i).Name, v2.Field(i).Type().Name()) if v1.Field(i).CanAddr() && v2.Field(i).CanAddr() { t(" Values: %#v - %#v", v1.Field(i).Interface(), v2.Field(i).Interface()) } return false } } return true case reflect.Map: if v1.IsNil() != v2.IsNil() { t("Maps are not nil", v1.Type().Name(), v2.Type().Name()) return false } if v1.Len() != v2.Len() { t("Maps don't have the same length %d vs %d", v1.Len(), v2.Len()) return false } if v1.Pointer() == v2.Pointer() { return true } for _, k := range v1.MapKeys() { val1 := v1.MapIndex(k) val2 := v2.MapIndex(k) if !val1.IsValid() || !val2.IsValid() || !deepValueEqual(t, v1.MapIndex(k), v2.MapIndex(k), visited, depth+1) { t("Maps values at index %s are not equal", k.String()) return false } } return true case reflect.Func: if v1.IsNil() && v2.IsNil() { return true } // Can't do better than this: return false } return true // i guess? } var zLoc, _ = time.LoadLocation("UTC") var allTests = tests{ "empty": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{}, }, "link_simple": testPair{ expected: true, blank: &a.Link{}, result: &a.Link{ Type: a.LinkType, Href: a.IRI("http://example.org/abc"), HrefLang: a.LangRef("en"), MediaType: a.MimeType("text/html"), Name: a.NaturalLanguageValue{{ a.NilLangRef, "An example link", }}, }, }, "object_with_url": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{ URL: a.IRI("http://littr.git/api/accounts/system"), }, }, "object_with_url_collection": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{ URL: a.ItemCollection{ a.IRI("http://littr.git/api/accounts/system"), a.IRI("http://littr.git/~system"), }, }, }, "object_simple": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{ Type: a.ObjectType, ID: a.ObjectID("http://www.test.example/object/1"), Name: a.NaturalLanguageValue{{ a.NilLangRef, "A Simple, non-specific object", }}, }, }, "object_with_tags": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{ Type: a.ObjectType, ID: a.ObjectID("http://www.test.example/object/1"), Name: a.NaturalLanguageValue{{ a.NilLangRef, "A Simple, non-specific object", }}, Tag: a.ItemCollection{ &a.Object{ Name: a.NaturalLanguageValue{{ a.NilLangRef, "#my_tag", }}, ID: a.ObjectID("http://example.com/tag/my_tag"), }, &a.Mention{ Name: a.NaturalLanguageValue{{ a.NilLangRef, "@ana", }}, Type: a.MentionType, ID: a.ObjectID("http://example.com/users/ana"), }, }, }, }, "object_with_replies": testPair{ expected: true, blank: &a.Object{}, result: &a.Object{ Type: a.ObjectType, ID: a.ObjectID("http://www.test.example/object/1"), Replies: &a.Collection{ Parent: a.Parent{ ID: a.ObjectID("http://www.test.example/object/1/replies"), Type: a.CollectionType, }, TotalItems: 1, Items: a.ItemCollection{ &a.Object{ ID: a.ObjectID("http://www.test.example/object/1/replies/2"), Type: a.ArticleType, Name: a.NaturalLanguageValue{{a.NilLangRef, "Example title"}}, }, }, }, }, }, "activity_simple": testPair{ expected: true, blank: &a.Activity{}, result: &a.Activity{ Parent: a.Parent{ Type: a.ActivityType, Summary: a.NaturalLanguageValue{{a.NilLangRef, "Sally did something to a note"}}, }, Actor: &a.Person{ Parent: a.Parent{ Type: a.PersonType, Name: a.NaturalLanguageValue{{a.NilLangRef, "Sally"}}, }, }, Object: &a.Object{ Type: a.NoteType, Name: a.NaturalLanguageValue{{a.NilLangRef, "A Note"}}, }, }, }, "person_with_outbox": testPair{ expected: true, blank: &a.Person{}, result: &a.Person{ Parent: a.Parent{ ID: a.ObjectID("http://example.com/accounts/ana"), Type: a.PersonType, Name: a.NaturalLanguageValue{{a.NilLangRef, "ana"}}, URL: a.IRI("http://example.com/accounts/ana"), }, PreferredUsername: a.NaturalLanguageValue{{a.NilLangRef, "Ana"}}, Outbox: &a.OrderedCollection{ Parent: a.Parent{ ID: a.ObjectID("http://example.com/accounts/ana/outbox"), Type: a.OrderedCollectionType, URL: a.IRI("http://example.com/outbox"), }, }, }, }, "ordered_collection": testPair{ expected: true, blank: &a.OrderedCollection{}, result: &a.OrderedCollection{ Parent: a.Parent{ ID: a.ObjectID("http://example.com/outbox"), Type: a.OrderedCollectionType, URL: a.IRI("http://example.com/outbox"), }, TotalItems: 1, OrderedItems: a.ItemCollection{ &a.Object{ ID: a.ObjectID("http://example.com/outbox/53c6fb47"), Type: a.ArticleType, Name: a.NaturalLanguageValue{{a.NilLangRef, "Example title"}}, Content: a.NaturalLanguageValue{{a.NilLangRef, "Example content!"}}, URL: a.IRI("http://example.com/53c6fb47"), MediaType: a.MimeType("text/markdown"), Published: time.Date(2018, time.July, 5, 16, 46, 44, 0, zLoc), Generator: a.IRI("http://example.com"), AttributedTo: a.IRI("http://example.com/accounts/alice"), }, }, }, }, "ordered_collection_page": testPair{ expected: true, blank: &a.OrderedCollectionPage{}, result: &a.OrderedCollectionPage{ PartOf: a.IRI("http://example.com/outbox"), Next: a.IRI("http://example.com/outbox?page=3"), Prev: a.IRI("http://example.com/outbox?page=1"), Current: a.IRI("http://example.com/outbox?page=2"), OrderedCollection: a.OrderedCollection{ Parent: a.Parent{ ID: a.ObjectID("http://example.com/outbox?page=2"), Type: a.OrderedCollectionPageType, URL: a.IRI("http://example.com/outbox?page=2"), }, TotalItems: 1, OrderedItems: a.ItemCollection{ &a.Object{ ID: a.ObjectID("http://example.com/outbox/53c6fb47"), Type: a.ArticleType, Name: a.NaturalLanguageValue{{a.NilLangRef, "Example title"}}, Content: a.NaturalLanguageValue{{a.NilLangRef, "Example content!"}}, URL: a.IRI("http://example.com/53c6fb47"), MediaType: a.MimeType("text/markdown"), Published: time.Date(2018, time.July, 5, 16, 46, 44, 0, zLoc), Generator: a.IRI("http://example.com"), AttributedTo: a.IRI("http://example.com/accounts/alice"), }, }, }, }, }, "natural_language_values": { expected: true, blank: &a.NaturalLanguageValue{}, result: &a.NaturalLanguageValue{ { a.NilLangRef, ` `}, {a.LangRef("en"), "Ana got apples ⓐ"}, {a.LangRef("fr"), "Aná a des pommes ⒜"}, {a.LangRef("ro"), "Ana are mere"}, }, }, "activity_create_simple": { expected: true, blank: &a.Create{}, result: &a.Create{ Parent: a.Parent{ Type: a.CreateType, }, Actor: a.IRI("https://littr.git/api/accounts/anonymous"), Object: &a.Object{ Type: a.NoteType, AttributedTo: a.IRI("https://littr.git/api/accounts/anonymous"), InReplyTo: a.IRI("https://littr.git/api/accounts/system/outbox/7ca154ff"), Content: a.NaturalLanguageValue{{a.NilLangRef, "

Hello world

"}}, To: a.ItemCollection{a.IRI("https://www.w3.org/ns/activitystreams#Public")}, }, }, }, "like_activity_with_iri_actor": { expected: true, blank: &ap.LikeActivity{}, result: &ap.LikeActivity{ Activity: &a.Like{ Parent: a.Parent{ Type: a.LikeType, }, Actor: a.IRI("https://littr.git/api/accounts/24d4b96f"), Object: &a.Object{ ID: a.ObjectID("https://littr.git/api/accounts/ana/liked/7ca154ff"), Type: a.ArticleType, }, }, Published: time.Date(2018, time.September, 6, 15, 15, 9, 0, zLoc), }, }, } func getFileContents(path string) ([]byte, error) { f, err := os.Open(path) if err != nil { return nil, err } st, err := f.Stat() if err != nil { return nil, err } data := make([]byte, st.Size()) io.ReadFull(f, data) data = bytes.Trim(data, "\x00") return data, nil } func Test_ActivityPubUnmarshall(t *testing.T) { var err error var f = t.Errorf if len(allTests) == 0 { t.Skip("No tests found") } for k, pair := range allTests { var data []byte path := filepath.Join(dir, fmt.Sprintf("%s.json", k)) data, err = getFileContents(path) if err != nil { f("Error: %s for %s", err, path) continue } object := pair.blank err = j.Unmarshal(data, object) if err != nil { f("Error: %s for %s", err, data) continue } expLbl := "" if !pair.expected { expLbl = "not be " } status := assertDeepEquals(f, object, pair.result) if pair.expected != status { if stopOnFailure { f = t.Fatalf } f("\n%#v\n should %sequal to expected\n%#v", object, expLbl, pair.result) continue } if !status { oj, err := j.Marshal(object) if err != nil { f(err.Error()) } tj, err := j.Marshal(pair.result) if err != nil { f(err.Error()) } f("\n%s\n should %sequal to expected\n%s", oj, expLbl, tj) } if err == nil { fmt.Printf(" --- %s: %s\n %s\n", "PASS", k, path) } } }