t_as2_vocab.ml 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648
  1. open Alcotest
  2. open Seppo_lib
  3. module A = Assrt
  4. let set_up = "set_up", `Quick, (fun () ->
  5. Unix.chdir "../../../test/"
  6. )
  7. let tc_err = "tc_err", `Quick, (fun () ->
  8. match {|[]|}
  9. |> Ezjsonm.from_string
  10. |> As2_vocab.Activitypub.Decode.obj with
  11. | Error (e : Decoders_ezjsonm.Decode.error) ->
  12. e |> Format.asprintf "%a" Decoders_ezjsonm.Decode.pp_error
  13. |> check string __LOC__ {|I tried the following decoders but they all failed:
  14. "core_obj" decoder: Expected a string, but got []
  15. "core_obj event" decoder:
  16. Expected an object with an attribute "type", but got [] |};
  17. e |> Decoders_ezjsonm.Decode.string_of_error
  18. |> check string __LOC__ {|I tried the following decoders but they all failed:
  19. "core_obj" decoder: Expected a string, but got []
  20. "core_obj event" decoder:
  21. Expected an object with an attribute "type", but got [] |};
  22. | _ -> failwith __LOC__
  23. (* diaspora profile json e.g.
  24. https://pod.diaspora.software/u/hq
  25. https://pod.diaspora.software/people/7bca7c80311b01332d046c626dd55703
  26. *)
  27. )
  28. let tc_person = "tc_person", `Quick, (fun () ->
  29. let empty : As2_vocab.Types.actor = {
  30. typ = "Person";
  31. id = Uri.empty;
  32. inbox = Uri.empty;
  33. outbox = Uri.empty;
  34. followers = None;
  35. following = None;
  36. attachment = [];
  37. discoverable = false;
  38. generator = None;
  39. icon = [];
  40. image = None;
  41. manually_approves_followers = false;
  42. name = None;
  43. name_map = [];
  44. preferred_username = None;
  45. preferred_username_map = [];
  46. public_key = {
  47. id = Uri.empty;
  48. owner = None;
  49. pem = "";
  50. signatureAlgorithm = None;
  51. };
  52. published = Some Ptime.epoch;
  53. summary = None;
  54. summary_map = [];
  55. url = [];
  56. } in
  57. let base = (Uri.of_string "https://example.com/su/") in
  58. let lang = As2_vocab.Constants.ActivityStreams.und in
  59. {empty with id=Uri.make ~path:"id/" ()}
  60. |> As2_vocab.Encode.actor~lang ~base
  61. |> Ezjsonm.value_to_string ~minify:false
  62. |> check string __LOC__ {|{
  63. "@context": [
  64. "https://www.w3.org/ns/activitystreams",
  65. "https://w3id.org/security/v1",
  66. {
  67. "schema": "http://schema.org#",
  68. "PropertyValue": "schema:PropertyValue",
  69. "value": "schema:value",
  70. "@language": "und"
  71. }
  72. ],
  73. "type": "Person",
  74. "id": "https://example.com/su/id/",
  75. "inbox": "https://example.com/su/",
  76. "outbox": "https://example.com/su/",
  77. "publicKey": {
  78. "@context": [
  79. {
  80. "@language": null
  81. }
  82. ],
  83. "id": "https://example.com/su/",
  84. "publicKeyPem": ""
  85. },
  86. "published": "1970-01-01T00:00:00Z",
  87. "manuallyApprovesFollowers": false,
  88. "discoverable": false
  89. }|};
  90. let p = {|{
  91. "@context": [
  92. "https://www.w3.org/ns/activitystreams",
  93. "https://w3id.org/security/v1",
  94. {
  95. "schema": "http://schema.org#",
  96. "PropertyValue": "schema:PropertyValue",
  97. "value": "schema:value",
  98. "@language": "und"
  99. }
  100. ],
  101. "type": "Person",
  102. "id": "https://example.com/su/id/",
  103. "inbox": "https://example.com/su/",
  104. "outbox": "https://example.com/su/",
  105. "publicKey": {
  106. "id": "https://example.com/su/",
  107. "owner": "https://example.com/su/",
  108. "publicKeyPem": ""
  109. },
  110. "published": "1970-01-01T00:00:00Z",
  111. "manuallyApprovesFollowers": false,
  112. "discoverable": false,
  113. "attachment": []
  114. }|}
  115. |> Ezjsonm.value_from_string
  116. |> As2_vocab.Decode.actor
  117. |> Result.get_ok in
  118. p.id
  119. |> Uri.to_string
  120. |> check string __LOC__ "https://example.com/su/id/"
  121. )
  122. let tc_actor_3rd = "tc_actor_3rd", `Quick, (fun () ->
  123. Logr.info (fun m -> m "%s.%s" "As2_vocab" "actor_3rd");
  124. let ok loc fn na =
  125. let j = fn |> File.in_channel Ezjsonm.from_channel in
  126. let p = j |> As2_vocab.Decode.actor |> Result.get_ok in
  127. p.name |> Option.get |> check string loc na
  128. and oki loc fn id =
  129. let j = fn |> File.in_channel Ezjsonm.from_channel in
  130. let p = j |> As2_vocab.Decode.actor |> Result.get_ok in
  131. p.id |> Uri.to_string |> check string loc (id)
  132. and _err loc fn e =
  133. let j = fn |> File.in_channel Ezjsonm.from_channel in
  134. j |> As2_vocab.Decode.actor |> Result.get_error
  135. |> Decoders_ezjsonm.Decode.string_of_error
  136. |> check string loc e
  137. and fal _loc fn =
  138. let j = fn |> File.in_channel Ezjsonm.from_channel in
  139. assert (j |> As2_vocab.Decode.actor |> Result.is_error)
  140. in
  141. ok __LOC__ "data/ap/actor/akkoma.0.json" "Sean Tilley";
  142. ok __LOC__ "data/ap/actor/akkoma.1.json" "Kinetix";
  143. ok __LOC__ "data/ap/actor/bonfire.0.json" "stpaultim";
  144. ok __LOC__ "data/ap/actor/bridgy.0.json" "Tantek Çelik";
  145. ok __LOC__ "data/ap/actor/friendica.0.json" "Michael Vogel";
  146. ok __LOC__ "data/ap/actor/friendica.1.json" {|Fediphoto Lineage|};
  147. ok __LOC__ "data/ap/actor/gnusocial.0.json" "Diogo Peralta Cordeiro";
  148. ok __LOC__ "data/ap/actor/gnusocial.1.json" "admin de gnusocial.net";
  149. ok __LOC__ "data/ap/actor/gnusocial.2.json" "diogo";
  150. ok __LOC__ "data/ap/actor/gotosocial.0.json" "Gerben";
  151. ok __LOC__ "data/ap/actor/gotosocial.1.json" "Gerben";
  152. oki __LOC__ "data/ap/actor/gotosocial.1b.json" "https://social.nlnet.nl/users/gerben";
  153. ok __LOC__ "data/ap/actor/honk.0.json" "boyter";
  154. (* ok __LOC__ "data/ap/actor/lemmy.0.json" ""; *)
  155. ok __LOC__ "data/ap/actor/mastodon.0.json" "Yet Another #Seppo! 🌻";
  156. ok __LOC__ "data/ap/actor/mastodon.1.json" "#Seppo";
  157. ok __LOC__ "data/ap/actor/mastodon.2.json" "Marcus Rohrmoser 🌍";
  158. ok __LOC__ "data/ap/actor/mini.0.json" "Yet Another #Seppo! 🌻";
  159. ok __LOC__ "data/ap/actor/misskey.0.json" "しゅいろ";
  160. ok __LOC__ "data/ap/actor/mobilizon.0.json" {|The ContribUtopists|};
  161. ok __LOC__ "data/ap/actor/peertube.0.json" "edps";
  162. ok __LOC__ "data/ap/actor/peertube.1.json" {|Q3 Marcus|};
  163. ok __LOC__ "data/ap/actor/peertube.2.json" {|edps|};
  164. ok __LOC__ "data/ap/actor/peertube.3.json" {|Framasoft|};
  165. ok __LOC__ "data/ap/actor/pixelfed.0.json" {|#Seppo|};
  166. ok __LOC__ "data/ap/actor/pleroma.0.json" "@fediverse@mro.name";
  167. ok __LOC__ "data/ap/actor/smithereen.0.json" {|Григорий Клюшников|};
  168. ok __LOC__ "data/ap/actor/snac.0.json" {|The Real Grunfink|};
  169. ok __LOC__ "data/ap/actor/threads.0.json" "Ben Savage";
  170. ok __LOC__ "data/ap/actor/tootik.0.json" {|EOIN GAIRLEOG|};
  171. ok __LOC__ "data/ap/actor/zap.0.json" "mike";
  172. ok __LOC__ "data/ap/profile/@actapopuli@fediverse.blog.json" "actapopuli";
  173. ok __LOC__ "data/ap/profile/@administrator@gnusocial.net.json" {|admin de gnusocial.net|};
  174. ok __LOC__ "data/ap/profile/@dansup@pixelfed.social.json" {|dansup|};
  175. ok __LOC__ "data/ap/profile/@gargron@mastodon.social.json" {|Eugen Rochko|};
  176. ok __LOC__ "data/ap/profile/@kainoa@calckey.social.json" {|Kainoa |};
  177. ok __LOC__ "data/ap/profile/@karolat@stereophonic.space.json" {|karolat|};
  178. ok __LOC__ "data/ap/profile/@lemmy_support@lemmy.ml.json" {|Lemmy Support|};
  179. ok __LOC__ "data/ap/profile/@manton@manton.org.json" {|Manton Reece|};
  180. ok __LOC__ "data/ap/profile/@matt@write.as.json" {|Matt|};
  181. ok __LOC__ "data/ap/profile/@mike@macgirvin.com.json" {|Mike Macgirvin|};
  182. ok __LOC__ "data/ap/profile/@peertube@framapiaf.org.json" {|PeerTube|};
  183. ok __LOC__ "data/ap/profile/@syuilo@misskey.io.json" {|:peroro_sama:しゅいろ:peroro_sama:|};
  184. ok __LOC__ "data/ap/profile/@tobias@friendi.ca.json" {|Tobias|};
  185. (*
  186. err __LOC__ "data/ap/actor/gotosocial.1b.json" {|Expected an object with an attribute "inbox", but got
  187. {"@context":["https://www.w3.org/ns/activitystreams","https://w3id.org/security/v1"],"id":"https://social.nlnet.nl/users/gerben","preferredUsername":"gerben","publicKey":{"id":"https://social.nlnet.nl/users/gerben/main-key","owner":"https://social.nlnet.nl/users/gerben","publicKeyPem":"-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA35UDYAz/pgrIi4O1UDf9\n9W1RnEPVtj730gfujlIQtzWIsBs/8n6zQbK0QBtRSYKhwzkUDMpNUh+Mm/CONjKI\nSwps6mEknlmi3VbyviGYs7LX7SuJAfLIl1hT2eftXQLObKgzkbUcVbS0zwNlBeIX\nPJpHyAVphuI4a02Be458g3y3nSK7Imb4dN2GUhLX2FDOZ6s/EC9HcglZiiz+9hI2\n0Rb4tXjsjbBXe6ofowdXDPmD20Al3v816m4kYfgGmLK86ItFGj8u+hoK8tcC0yWk\ni7KUL3bg9iqp7wBhkjTZ/vVVIf7RG8q2gPRe1BVsi+GBDmV7dmupgicT6UbmAi5h\nTwIDAQAB\n-----END PUBLIC KEY-----\n"},"type":"Person"}|};
  188. *)
  189. fal __LOC__ "data/ap/actor/diaspora.0.json";
  190. ()
  191. (*
  192. ok __LOC__ "data/ap/actor/natur.0.json" "しゅいろ";
  193. ok __LOC__ "data/ap/actor/natur.1.json" "しゅいろ";
  194. ok __LOC__ "data/ap/actor/sharkey.0.json" {||};
  195. ok __LOC__ "data/ap/actor/sharkey.1.json" {||};
  196. ok __LOC__ "data/ap/profile/@framasoft@mobilizon.fr.json" {||};
  197. ok __LOC__ "data/ap/profile/@Greensky@open.audio.json" {||};
  198. *)
  199. )
  200. let tc_actor_decode = "tc_actor_decode", `Quick, (fun () ->
  201. let fn = "data/ap/actor/peertube.3.json" in
  202. let j = fn |> File.in_channel Ezjsonm.from_channel in
  203. let p = j |> As2_vocab.Decode.actor |> Result.get_ok in
  204. p.name |> Option.get |> check string __LOC__ "Framasoft";
  205. p.icon |> List.length |> check int __LOC__ 2;
  206. ()
  207. )
  208. let tc_actor_encode = "tc_actor_encode", `Quick, (fun () ->
  209. let base = Uri.empty
  210. and lang = None
  211. and minify = false
  212. in
  213. let s = "data/ap/actor/peertube.3.json"
  214. |> File.in_channel Ezjsonm.from_channel
  215. |> As2_vocab.Decode.actor |> Result.get_ok
  216. |> As2_vocab.Encode.actor ~base ~lang
  217. |> Ezjsonm.value_to_string ~minify in
  218. s |>
  219. check string
  220. (*
  221. check string
  222. *)
  223. __LOC__ {|{
  224. "type": "Person",
  225. "id": "https://framatube.org/accounts/framasoft",
  226. "inbox": "https://framatube.org/accounts/framasoft/inbox",
  227. "outbox": "https://framatube.org/accounts/framasoft/outbox",
  228. "followers": "https://framatube.org/accounts/framasoft/followers",
  229. "following": "https://framatube.org/accounts/framasoft/following",
  230. "name": "Framasoft",
  231. "url": "https://framatube.org/accounts/framasoft",
  232. "preferredUsername": "framasoft",
  233. "publicKey": {
  234. "@context": [
  235. {
  236. "@language": null
  237. }
  238. ],
  239. "id": "https://framatube.org/accounts/framasoft#main-key",
  240. "owner": "https://framatube.org/accounts/framasoft",
  241. "publicKeyPem": "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAuRh3frgIg866D0y0FThp\nSUkJImMcHGkUvpYQYv2iUgarZZtEbwT8PfQf0bJazy+cP8KqQmMDf5PBhT7dfdny\nf/GKGMw9Olc+QISeKDj3sqZ3Csrm4KV4avMGCfth6eSU7LozojeSGCXdUFz/8UgE\nfhV4mJjEX/FbwRYoKlagv5rY9mkX5XomzZU+z9j6ZVXyofwOwJvmI1hq0SYDv2bc\neB/RgIh/H0nyMtF8o+0CT42FNEET9j9m1BKOBtPzwZHmitKRkEmui5cK256s1laB\nT61KHpcD9gQKkQ+I3sFEzCBUJYfVo6fUe+GehBZuAfq4qDhd15SfE4K9veDscDFI\nTwIDAQAB\n-----END PUBLIC KEY-----"
  242. },
  243. "published": "2018-03-01T15:16:17Z",
  244. "manuallyApprovesFollowers": false,
  245. "discoverable": false,
  246. "icon": [
  247. {
  248. "type": "Image",
  249. "url": "https://framatube.org/lazy-static/avatars/1dbda4f0-1f7f-40f2-b962-85fd0c144661.png"
  250. },
  251. {
  252. "type": "Image",
  253. "url": "https://framatube.org/lazy-static/avatars/f73876f5-1d45-4f8a-942a-d3d5d5ac5dc1.png"
  254. }
  255. ]
  256. }|}
  257. )
  258. let tc_actor_encode_issue35 = "tc_actor_encode", `Quick, (fun () ->
  259. let base = Uri.empty
  260. and url = [ "https://example.com" |> Uri.of_string ]
  261. in
  262. let (<:) = function
  263. | (_, None) -> fun _ -> []
  264. | (field, Some vl) -> fun ty -> [field, ty vl] in
  265. let (@?.) field vl = (field, match vl with | [] -> None | l -> Some l) in
  266. let list (a : 'a Decoders_ezjsonm.Encode.encoder) (b : 'a list) : Ezjsonm.value = match b with
  267. | [] -> Decoders_ezjsonm.Encode.null (* none => omit *)
  268. | [b] -> Decoders_ezjsonm.Encode.encode_value a b (* single => value *)
  269. | b -> Decoders_ezjsonm.Encode.list a b in
  270. let uri ~base u = u |> Uri.resolve "https" base |> Uri.to_string |> Decoders_ezjsonm.Encode.string in
  271. (match
  272. "url" @?. url <: list (uri ~base);
  273. with
  274. | [("url",j)] -> j
  275. |> Ezjsonm.value_to_string
  276. |> check string __LOC__ {|"https://example.com"|}
  277. | _ -> fail __LOC__
  278. ))
  279. let tc_note_decode = "tc_note_dcode", `Quick, (fun () ->
  280. let j = "data/ap/note/mastodon.json" |> File.in_channel Ezjsonm.from_channel in
  281. let n = j |> As2_vocab.Decode.note |> Result.get_ok in
  282. n.id |> Uri.to_string |> check string __LOC__ "https://digitalcourage.social/users/mro/statuses/111403080326863922";
  283. (match n.in_reply_to with
  284. | [u] -> u |> Uri.to_string |> check string __LOC__ "https://chaos.social/users/qyliss/statuses/111403054651938519"
  285. | _ -> failwith "none");
  286. let n = "data/ap/inbox/create/note/akkoma.json" |> File.in_channel Ezjsonm.from_channel
  287. |> As2_vocab.Decode.(create note) |> Result.get_ok in
  288. let _,co = n.obj.content_map |> List.hd in
  289. co
  290. |> Assrt.equals_string __LOC__
  291. {|<p>Oh yeah! Also: trying to flip the pencil over to use an eraser doesn’t work. You have to select an eraser first. </p><p>What the fuck? My shitty Wacom Bamboo tablet from ten years ago can do that!</p>|}
  292. )
  293. let tc_create_article_decode = "tc_create_article_decode", `Quick, (fun () ->
  294. "data/ap/inbox/create/article/friendica.json"
  295. |> File.in_channel Ezjsonm.from_channel
  296. |> As2_vocab.Decode.(create note) |> Result.get_error
  297. |> Decoders_ezjsonm.Decode.string_of_error
  298. |> check string __LOC__ "in field \"object\":\n in field \"type\": expected Note (received Article), but got \"Article\""
  299. )
  300. let tc_reject_decode = "tc_reject_decode", `Quick, (fun () ->
  301. Logr.info (fun m -> m "%s.%s" "As2_vocab" "reject_decode");
  302. let j = "data/ap/inbox/reject/follow/2024-08-17-124924-steve.json" |> File.in_channel Ezjsonm.from_channel in
  303. let n = j |> As2_vocab.Decode.(reject follow) |> Result.get_ok in
  304. n.id
  305. |> Uri.to_string
  306. |> check string __LOC__ {|https://social.technoetic.com/users/steve#rejects/follows/|};
  307. n.obj.id
  308. |> Uri.to_string
  309. |> check string __LOC__ {|https://social.technoetic.com/users/steve#subscribe|}
  310. )
  311. let tc_webfinger_sunshine = "tc_webfinger_sunshine", `Quick, (fun () ->
  312. let q = "data/webfinger/mini.json" |> File.in_channel Ezjsonm.from_channel
  313. |> As2_vocab.Activitypub.Decode.Webfinger.query_result
  314. |> Result.get_ok in
  315. q.subject |> check string __LOC__ "acct:ursi@example.com";
  316. q.links |> List.length |> check int __LOC__ 3;
  317. let q = "data/webfinger/zap.json" |> File.in_channel Ezjsonm.from_channel
  318. |> As2_vocab.Activitypub.Decode.Webfinger.query_result
  319. |> Result.get_ok in
  320. q.subject |> check string __LOC__ "acct:mike@macgirvin.com";
  321. q.links |> List.length |> check int __LOC__ 3;
  322. let q = "data/webfinger/atom.json" |> File.in_channel Ezjsonm.from_channel
  323. |> As2_vocab.Activitypub.Decode.Webfinger.query_result
  324. |> Result.get_ok in
  325. q.subject |> check string __LOC__ "acct:ursi@example.com";
  326. q.links |> List.length |> check int __LOC__ 3
  327. )
  328. let tc_profile_sunshine = "tc_profile_sunshine", `Quick, (fun () ->
  329. let q = "data/ap/actor/mini.0.json" |> File.in_channel Ezjsonm.from_channel
  330. |> As2_vocab.Activitypub.Decode.actor
  331. |> Result.get_ok in
  332. q.name |> Option.get |> check string __LOC__ "Yet Another #Seppo! 🌻";
  333. q.preferred_username |> Option.get |> check string __LOC__ "ursi";
  334. q.attachment |> List.length |> check int "" 0;
  335. let q = "data/ap/actor/mastodon.0.json" |> File.in_channel Ezjsonm.from_channel
  336. |> As2_vocab.Activitypub.Decode.actor
  337. |> Result.get_ok in
  338. (match q.attachment with
  339. | [a;b;_] ->
  340. a.name |> check string __LOC__ "Support";
  341. b.value |> check string __LOC__ {|<a href="https://seppo.social">Seppo.Social</a>|};
  342. | _ -> check int "" 0 1
  343. );
  344. q.image |> Option.get |> Uri.to_string |> check string __LOC__ "https://example.com/me-banner.jpg";
  345. "" |> check string __LOC__ "";
  346. let q = "data/ap/actor/akkoma.0.json" |> File.in_channel Ezjsonm.from_channel
  347. |> As2_vocab.Activitypub.Decode.actor
  348. |> Result.get_ok in
  349. q.name |> Option.get |> check string __LOC__ "Sean Tilley";
  350. let q = "data/ap/actor/gnusocial.0.json" |> File.in_channel Ezjsonm.from_channel
  351. |> As2_vocab.Activitypub.Decode.actor
  352. |> Result.get_ok in
  353. q.preferred_username |> Option.get |> check string __LOC__ "diogo";
  354. let q = "data/ap/actor/lemmy.0.json" |> File.in_channel Ezjsonm.from_channel
  355. |> As2_vocab.Activitypub.Decode.actor
  356. |> Result.get_ok in
  357. q.preferred_username |> Option.get |> check string __LOC__ "nutomic";
  358. let q = "data/ap/actor/mastodon.0.json" |> File.in_channel Ezjsonm.from_channel
  359. |> As2_vocab.Activitypub.Decode.actor
  360. |> Result.get_ok in
  361. q.preferred_username |> Option.get |> check string __LOC__ "ursi";
  362. let q = "data/ap/actor/peertube.0.json" |> File.in_channel Ezjsonm.from_channel
  363. |> As2_vocab.Activitypub.Decode.actor
  364. |> Result.get_ok in
  365. q.preferred_username |> Option.get |> check string __LOC__ "edps";
  366. let q = "data/ap/actor/zap.0.json" |> File.in_channel Ezjsonm.from_channel
  367. |> As2_vocab.Activitypub.Decode.actor
  368. |> Result.get_ok in
  369. q.preferred_username |> Option.get |> check string __LOC__ "mike";
  370. let q = "data/ap/actor/bridgy.0.json" |> File.in_channel Ezjsonm.from_channel
  371. |> As2_vocab.Activitypub.Decode.actor
  372. |> Result.get_ok in
  373. q.preferred_username |> Option.get |> check string __LOC__ "tantek.com";
  374. assert true
  375. )
  376. let tc_encode = "tc_encode", `Quick, (fun () ->
  377. let minify = false in
  378. let base = Uri.of_string "http://example.com/foo/" in
  379. let module E = Decoders_ezjsonm.Encode in
  380. E.encode_string E.obj [("k", `String "v")]
  381. |> check string __LOC__ {|{"k":"v"}|};
  382. Ezjsonm.value_to_string (E.obj [("k", `String "v")])
  383. |> check string __LOC__ {|{"k":"v"}|};
  384. let e = {Rfc4287.Entry.empty with
  385. id = Uri.make ~path:"a/b/" ?fragment:(Some "c") ();
  386. lang = Rfc4287.Rfc4646 "de";
  387. title = "uhu";
  388. published = (Rfc3339.T "2023-03-07T01:23:45Z") ;
  389. updated = (Rfc3339.T "2023-03-07T01:23:46Z");
  390. categories = [
  391. ((Label (Single "lbl")),(Term (Single "term")),Uri.make ~path:"t/" ());
  392. ];
  393. content = "Das war aber einfach";
  394. } in
  395. let n = e |> Ap.Note.of_rfc4287 in
  396. let j = n |> As2_vocab.Encode.note ~base in
  397. j |> Ezjsonm.value_to_string ~minify
  398. |> Assrt.equals_string __LOC__
  399. {|{
  400. "type": "Note",
  401. "id": "http://example.com/foo/a/b/#c",
  402. "attributedTo": "http://example.com/foo/activitypub/actor.jsa",
  403. "to": "https://www.w3.org/ns/activitystreams#Public",
  404. "cc": "http://example.com/foo/activitypub/subscribers/index.jsa",
  405. "mediaType": "text/plain; charset=utf8",
  406. "contentMap": {
  407. "de": "Das war aber einfach"
  408. },
  409. "sensitive": false,
  410. "summaryMap": {
  411. "de": "uhu"
  412. },
  413. "published": "2023-03-07T01:23:45Z",
  414. "tags": {
  415. "type": "Hashtag",
  416. "href": "http://example.com/foo/t/term/",
  417. "name": "#lbl"
  418. }
  419. }|};
  420. let co : 'a As2_vocab.Types.collection_page = {
  421. id = Uri.of_string "http://example.com/foo/";
  422. current = None;
  423. first = None;
  424. is_ordered = true;
  425. items = [Ap.Note.Create.make n];
  426. last = None;
  427. next = None;
  428. part_of = None;
  429. prev = None;
  430. total_items= None;
  431. } in
  432. let j = As2_vocab.Encode.collection_page ~base
  433. (As2_vocab.Encode.create ~base
  434. (As2_vocab.Encode.note ~base))
  435. co in
  436. j |> Ezjsonm.value_to_string ~minify
  437. |> Assrt.equals_string __LOC__ {|{
  438. "@context": [
  439. "https://www.w3.org/ns/activitystreams",
  440. "https://w3id.org/security/v1",
  441. {
  442. "schema": "http://schema.org#",
  443. "PropertyValue": "schema:PropertyValue",
  444. "value": "schema:value",
  445. "@language": "und"
  446. }
  447. ],
  448. "type": "OrderedCollectionPage",
  449. "id": "http://example.com/foo/",
  450. "orderedItems": [
  451. {
  452. "type": "Create",
  453. "id": "http://example.com/foo/a/b/#c/Create",
  454. "actor": "http://example.com/foo/activitypub/actor.jsa",
  455. "published": "2023-03-07T01:23:45Z",
  456. "to": "https://www.w3.org/ns/activitystreams#Public",
  457. "cc": "http://example.com/foo/activitypub/subscribers/index.jsa",
  458. "directMessage": false,
  459. "object": {
  460. "type": "Note",
  461. "id": "http://example.com/foo/a/b/#c",
  462. "attributedTo": "http://example.com/foo/activitypub/actor.jsa",
  463. "to": "https://www.w3.org/ns/activitystreams#Public",
  464. "cc": "http://example.com/foo/activitypub/subscribers/index.jsa",
  465. "mediaType": "text/plain; charset=utf8",
  466. "contentMap": {
  467. "de": "Das war aber einfach"
  468. },
  469. "sensitive": false,
  470. "summaryMap": {
  471. "de": "uhu"
  472. },
  473. "published": "2023-03-07T01:23:45Z",
  474. "tags": {
  475. "type": "Hashtag",
  476. "href": "http://example.com/foo/t/term/",
  477. "name": "#lbl"
  478. }
  479. }
  480. }
  481. ]
  482. }|};
  483. assert true
  484. )
  485. (* https://www.w3.org/TR/activitystreams-core/#ex17-jsonld *)
  486. let tc_ex15_note = "tc_ex15_note", `Quick, (fun () ->
  487. let j = "data/ap/note/as2_core.ex15.json" |> File.in_channel Ezjsonm.value_from_channel in
  488. (match j with
  489. | `O [
  490. "@context", _ ;
  491. "summary", _ ;
  492. "type", `String "Create" ;
  493. "actor", _ ;
  494. "object", `O [
  495. "type", `String "Note" ;
  496. _]
  497. ] -> assert true
  498. | _ -> assert false);
  499. let p = j
  500. |> As2_vocab.Decode.(create note)
  501. |> Result.is_error in
  502. assert p
  503. )
  504. (*
  505. _p.obj.summary |> Option.get |> check string "" "";
  506. *)
  507. (* https://github.com/mattjbray/ocaml-decoders *)
  508. type role = Admin | User
  509. type user =
  510. { lang : string
  511. ; txt : role list
  512. }
  513. let tc_example= "tc_exampl", `Quick, (fun () ->
  514. Logr.info (fun m -> m "%s.%s" "As2_vocab" "example");
  515. let module My_encoders(E : Decoders.Encode.S) = struct
  516. open E
  517. let user : role encoder =
  518. function
  519. | Admin -> string "ADMIN"
  520. | User -> string "USER"
  521. let user : user encoder =
  522. fun u ->
  523. obj
  524. [ ("name", string u.lang)
  525. ; ("roles", list user u.txt)
  526. ]
  527. end in
  528. let module E = Decoders_ezjsonm.Encode in
  529. let module My_ezjson_encoders = My_encoders(Decoders_ezjsonm.Encode) in
  530. let open My_ezjson_encoders in
  531. let users =
  532. [ {lang = "Alice"; txt = [Admin; User]}
  533. ; {lang = "Bob"; txt = [User]}
  534. ] in
  535. E.encode_string E.obj [("users", E.list user users)]
  536. |> check string __LOC__ {|{"users":[{"name":"Alice","roles":["ADMIN","USER"]},{"name":"Bob","roles":["USER"]}]}|}
  537. )
  538. type _i18n =
  539. { lang : string
  540. ; txt : string
  541. }
  542. let tc_encode_content_map = "tc_encode_content_map", `Quick, (fun () ->
  543. Logr.info (fun m -> m "%s.%s" "As2_vocab" "encode_content_map");
  544. let l = [("a","A");("b","B")] in
  545. let module E = Decoders_ezjsonm.Encode in
  546. let j = l |> List.map (fun (k,v) -> (k,E.string v)) in
  547. E.encode_string E.obj j
  548. |> check string __LOC__ {|{"a":"A","b":"B"}|}
  549. )
  550. let tc_decode_content_map = "tc_decode_content_map", `Quick, (fun () ->
  551. Logr.info (fun m -> m "%s.%s" "As2_vocab" "decode_content_map");
  552. let s = Ezjsonm.value_from_string {|{"a":"A","b":"B"}|} in
  553. let module D = Decoders_ezjsonm.Decode in
  554. let l = D.key_value_pairs D.string s
  555. |> Result.get_ok in
  556. (match l with
  557. | [("a","A");("b","B")] -> ()
  558. | _ -> "" |> check string __LOC__ {|{"a":"A","b":"B"}|});
  559. let s = Ezjsonm.value_from_string {|{"contentMap":{"a":"A","b":"B","b":"C"}}|} in
  560. let l = D.field "contentMap" (D.key_value_pairs D.string) s
  561. |> Result.get_ok in
  562. match l with
  563. | [("a","A");("b","B");("b","C")] -> ()
  564. | _ -> "" |> check string __LOC__ {|{"a":"A","b":"B"}|}
  565. )
  566. let tc_decode_natur = "tc_decode_natur", `Quick, (fun () ->
  567. (* https://codeberg.org/seppo/seppo/issues/5 *)
  568. Logr.info (fun m -> m "%s.%s" "As2_vocab" "decode_natur");
  569. let j = "data/ap/actor/natur.0.json" |> File.in_channel Ezjsonm.from_channel in
  570. let e = j |> As2_vocab.Decode.actor |> Result.get_error in
  571. let s = e |> Decoders_ezjsonm.Decode.string_of_error in
  572. s |> check string __LOC__ {|Expected an object with an attribute "publicKey", but got
  573. {"id":"https://dev.rdf-pub.org/d613b246-8984-4654-903d-8d44143aca40","type":"Person","inboxSparql":"https://dev.rdf-pub.org/d613b246-8984-4654-903d-8d44143aca40/inbox/sparql","rdfpub:oauth2Issuer":"https://login.m4h.network/auth/realms/LOA","rdfpub:oauth2IssuerPreferredUserName":"max@login.m4h.network","rdfpub:oauth2IssuerUserId":"1813bdc1-152c-4c27-92a6-6cdfe401ef3d@login.m4h.network","outboxSparql":"https://dev.rdf-pub.org/d613b246-8984-4654-903d-8d44143aca40/outbox/sparql","identifier":"52ff7eb2-0b7c-4388-9894-b40a27714c1b","version":{"type":"xsd:integer","@value":"1"},"owl:sameAs":{"id":"https://dev.rdf-pub.org/05a75688-c517-4ae1-842c-5da3d8460627"},"inbox":"https://dev.rdf-pub.org/d613b246-8984-4654-903d-8d44143aca40/inbox","endpoints":{"oauthAuthorizationEndpoint":"https://dev.rdf-pub.org/oauth/oauthAuthorizationEndpoint","oauthTokenEndpoint":"https://dev.rdf-pub.org/oauth/oauthTokenEndpoint"},"name":"max","outbox":"https://dev.rdf-pub.org/d613b246-8984-4654-903d-8d44143aca40/outbox","published":"2024-01-14T15:59:42.102+01:00","@context":["https://schema.org/docs/jsonldcontext.json","https://rdf-pub.org/schema/rdf-pub-context.json","https://www.w3.org/ns/activitystreams"]}|}
  574. )
  575. let tc_decode_sharkey= "tc_decode_sharke", `Quick, (fun () ->
  576. (* https://joinsharkey.org *)
  577. Logr.info (fun m -> m "%s.%s" "As2_vocab" "decode_sharkey");
  578. let j = "data/ap/actor/sharkey.0.json" |> File.in_channel Ezjsonm.from_channel in
  579. let p = j |> As2_vocab.Decode.actor |> Result.get_ok in
  580. p.name
  581. |> Option.value ~default:"-"
  582. |> check string __LOC__ {|-|};
  583. ("data/ap/actor/sharkey.1.json" |> File.in_channel Ezjsonm.from_channel
  584. |> As2_vocab.Decode.actor
  585. |> Result.get_ok)
  586. .name
  587. |> Option.value ~default:"-"
  588. |> check string __LOC__ {|wakest the shark possum|}
  589. )
  590. let () =
  591. run
  592. "seppo_suite" [
  593. __FILE__ , [
  594. set_up;
  595. tc_actor_3rd;
  596. tc_actor_decode;
  597. tc_actor_encode;
  598. tc_actor_encode_issue35;
  599. tc_profile_sunshine;
  600. tc_note_decode;
  601. tc_create_article_decode;
  602. tc_err;
  603. tc_person;
  604. tc_reject_decode;
  605. tc_webfinger_sunshine;
  606. tc_encode;
  607. tc_ex15_note;
  608. tc_example;
  609. tc_encode_content_map;
  610. tc_decode_content_map;
  611. tc_decode_natur;
  612. tc_decode_sharkey;
  613. ]
  614. ];
  615. assert true