Initial commit; nothing works yet
This commit is contained in:
commit
a599d133f4
19
.gitignore
vendored
Normal file
19
.gitignore
vendored
Normal file
|
@ -0,0 +1,19 @@
|
|||
*.class
|
||||
*.jar
|
||||
*~
|
||||
.hg/
|
||||
.hgignore
|
||||
/.calva
|
||||
/.clj-kondo
|
||||
/.lein-*
|
||||
/.lsp
|
||||
/.nrepl-port
|
||||
/.prepl-port
|
||||
/checkouts
|
||||
/classes
|
||||
/keys
|
||||
/resources/activitystreams-test-documents
|
||||
/target
|
||||
pom.xml
|
||||
pom.xml.asc
|
||||
profiles.clj
|
24
CHANGELOG.md
Normal file
24
CHANGELOG.md
Normal file
|
@ -0,0 +1,24 @@
|
|||
# Change Log
|
||||
All notable changes to this project will be documented in this file. This change log follows the conventions of [keepachangelog.com](http://keepachangelog.com/).
|
||||
|
||||
## [Unreleased]
|
||||
### Changed
|
||||
- Add a new arity to `make-widget-async` to provide a different widget shape.
|
||||
|
||||
## [0.1.1] - 2022-12-12
|
||||
### Changed
|
||||
- Documentation on how to make the widgets.
|
||||
|
||||
### Removed
|
||||
- `make-widget-sync` - we're all async, all the time.
|
||||
|
||||
### Fixed
|
||||
- Fixed widget maker to keep working when daylight savings switches over.
|
||||
|
||||
## 0.1.0 - 2022-12-12
|
||||
### Added
|
||||
- Files from the new template.
|
||||
- Widget maker public API - `make-widget-sync`.
|
||||
|
||||
[Unreleased]: https://sourcehost.site/your-name/dog-and-duck/compare/0.1.1...HEAD
|
||||
[0.1.1]: https://sourcehost.site/your-name/dog-and-duck/compare/0.1.0...0.1.1
|
280
LICENSE
Normal file
280
LICENSE
Normal file
|
@ -0,0 +1,280 @@
|
|||
Eclipse Public License - v 2.0
|
||||
|
||||
THE ACCOMPANYING PROGRAM IS PROVIDED UNDER THE TERMS OF THIS ECLIPSE
|
||||
PUBLIC LICENSE ("AGREEMENT"). ANY USE, REPRODUCTION OR DISTRIBUTION
|
||||
OF THE PROGRAM CONSTITUTES RECIPIENT'S ACCEPTANCE OF THIS AGREEMENT.
|
||||
|
||||
1. DEFINITIONS
|
||||
|
||||
"Contribution" means:
|
||||
|
||||
a) in the case of the initial Contributor, the initial content
|
||||
Distributed under this Agreement, and
|
||||
|
||||
b) in the case of each subsequent Contributor:
|
||||
i) changes to the Program, and
|
||||
ii) additions to the Program;
|
||||
where such changes and/or additions to the Program originate from
|
||||
and are Distributed by that particular Contributor. A Contribution
|
||||
"originates" from a Contributor if it was added to the Program by
|
||||
such Contributor itself or anyone acting on such Contributor's behalf.
|
||||
Contributions do not include changes or additions to the Program that
|
||||
are not Modified Works.
|
||||
|
||||
"Contributor" means any person or entity that Distributes the Program.
|
||||
|
||||
"Licensed Patents" mean patent claims licensable by a Contributor which
|
||||
are necessarily infringed by the use or sale of its Contribution alone
|
||||
or when combined with the Program.
|
||||
|
||||
"Program" means the Contributions Distributed in accordance with this
|
||||
Agreement.
|
||||
|
||||
"Recipient" means anyone who receives the Program under this Agreement
|
||||
or any Secondary License (as applicable), including Contributors.
|
||||
|
||||
"Derivative Works" shall mean any work, whether in Source Code or other
|
||||
form, that is based on (or derived from) the Program and for which the
|
||||
editorial revisions, annotations, elaborations, or other modifications
|
||||
represent, as a whole, an original work of authorship.
|
||||
|
||||
"Modified Works" shall mean any work in Source Code or other form that
|
||||
results from an addition to, deletion from, or modification of the
|
||||
contents of the Program, including, for purposes of clarity any new file
|
||||
in Source Code form that contains any contents of the Program. Modified
|
||||
Works shall not include works that contain only declarations,
|
||||
interfaces, types, classes, structures, or files of the Program solely
|
||||
in each case in order to link to, bind by name, or subclass the Program
|
||||
or Modified Works thereof.
|
||||
|
||||
"Distribute" means the acts of a) distributing or b) making available
|
||||
in any manner that enables the transfer of a copy.
|
||||
|
||||
"Source Code" means the form of a Program preferred for making
|
||||
modifications, including but not limited to software source code,
|
||||
documentation source, and configuration files.
|
||||
|
||||
"Secondary License" means either the GNU General Public License,
|
||||
Version 2.0, or any later versions of that license, including any
|
||||
exceptions or additional permissions as identified by the initial
|
||||
Contributor.
|
||||
|
||||
2. GRANT OF RIGHTS
|
||||
|
||||
a) Subject to the terms of this Agreement, each Contributor hereby
|
||||
grants Recipient a non-exclusive, worldwide, royalty-free copyright
|
||||
license to reproduce, prepare Derivative Works of, publicly display,
|
||||
publicly perform, Distribute and sublicense the Contribution of such
|
||||
Contributor, if any, and such Derivative Works.
|
||||
|
||||
b) Subject to the terms of this Agreement, each Contributor hereby
|
||||
grants Recipient a non-exclusive, worldwide, royalty-free patent
|
||||
license under Licensed Patents to make, use, sell, offer to sell,
|
||||
import and otherwise transfer the Contribution of such Contributor,
|
||||
if any, in Source Code or other form. This patent license shall
|
||||
apply to the combination of the Contribution and the Program if, at
|
||||
the time the Contribution is added by the Contributor, such addition
|
||||
of the Contribution causes such combination to be covered by the
|
||||
Licensed Patents. The patent license shall not apply to any other
|
||||
combinations which include the Contribution. No hardware per se is
|
||||
licensed hereunder.
|
||||
|
||||
c) Recipient understands that although each Contributor grants the
|
||||
licenses to its Contributions set forth herein, no assurances are
|
||||
provided by any Contributor that the Program does not infringe the
|
||||
patent or other intellectual property rights of any other entity.
|
||||
Each Contributor disclaims any liability to Recipient for claims
|
||||
brought by any other entity based on infringement of intellectual
|
||||
property rights or otherwise. As a condition to exercising the
|
||||
rights and licenses granted hereunder, each Recipient hereby
|
||||
assumes sole responsibility to secure any other intellectual
|
||||
property rights needed, if any. For example, if a third party
|
||||
patent license is required to allow Recipient to Distribute the
|
||||
Program, it is Recipient's responsibility to acquire that license
|
||||
before distributing the Program.
|
||||
|
||||
d) Each Contributor represents that to its knowledge it has
|
||||
sufficient copyright rights in its Contribution, if any, to grant
|
||||
the copyright license set forth in this Agreement.
|
||||
|
||||
e) Notwithstanding the terms of any Secondary License, no
|
||||
Contributor makes additional grants to any Recipient (other than
|
||||
those set forth in this Agreement) as a result of such Recipient's
|
||||
receipt of the Program under the terms of a Secondary License
|
||||
(if permitted under the terms of Section 3).
|
||||
|
||||
3. REQUIREMENTS
|
||||
|
||||
3.1 If a Contributor Distributes the Program in any form, then:
|
||||
|
||||
a) the Program must also be made available as Source Code, in
|
||||
accordance with section 3.2, and the Contributor must accompany
|
||||
the Program with a statement that the Source Code for the Program
|
||||
is available under this Agreement, and informs Recipients how to
|
||||
obtain it in a reasonable manner on or through a medium customarily
|
||||
used for software exchange; and
|
||||
|
||||
b) the Contributor may Distribute the Program under a license
|
||||
different than this Agreement, provided that such license:
|
||||
i) effectively disclaims on behalf of all other Contributors all
|
||||
warranties and conditions, express and implied, including
|
||||
warranties or conditions of title and non-infringement, and
|
||||
implied warranties or conditions of merchantability and fitness
|
||||
for a particular purpose;
|
||||
|
||||
ii) effectively excludes on behalf of all other Contributors all
|
||||
liability for damages, including direct, indirect, special,
|
||||
incidental and consequential damages, such as lost profits;
|
||||
|
||||
iii) does not attempt to limit or alter the recipients' rights
|
||||
in the Source Code under section 3.2; and
|
||||
|
||||
iv) requires any subsequent distribution of the Program by any
|
||||
party to be under a license that satisfies the requirements
|
||||
of this section 3.
|
||||
|
||||
3.2 When the Program is Distributed as Source Code:
|
||||
|
||||
a) it must be made available under this Agreement, or if the
|
||||
Program (i) is combined with other material in a separate file or
|
||||
files made available under a Secondary License, and (ii) the initial
|
||||
Contributor attached to the Source Code the notice described in
|
||||
Exhibit A of this Agreement, then the Program may be made available
|
||||
under the terms of such Secondary Licenses, and
|
||||
|
||||
b) a copy of this Agreement must be included with each copy of
|
||||
the Program.
|
||||
|
||||
3.3 Contributors may not remove or alter any copyright, patent,
|
||||
trademark, attribution notices, disclaimers of warranty, or limitations
|
||||
of liability ("notices") contained within the Program from any copy of
|
||||
the Program which they Distribute, provided that Contributors may add
|
||||
their own appropriate notices.
|
||||
|
||||
4. COMMERCIAL DISTRIBUTION
|
||||
|
||||
Commercial distributors of software may accept certain responsibilities
|
||||
with respect to end users, business partners and the like. While this
|
||||
license is intended to facilitate the commercial use of the Program,
|
||||
the Contributor who includes the Program in a commercial product
|
||||
offering should do so in a manner which does not create potential
|
||||
liability for other Contributors. Therefore, if a Contributor includes
|
||||
the Program in a commercial product offering, such Contributor
|
||||
("Commercial Contributor") hereby agrees to defend and indemnify every
|
||||
other Contributor ("Indemnified Contributor") against any losses,
|
||||
damages and costs (collectively "Losses") arising from claims, lawsuits
|
||||
and other legal actions brought by a third party against the Indemnified
|
||||
Contributor to the extent caused by the acts or omissions of such
|
||||
Commercial Contributor in connection with its distribution of the Program
|
||||
in a commercial product offering. The obligations in this section do not
|
||||
apply to any claims or Losses relating to any actual or alleged
|
||||
intellectual property infringement. In order to qualify, an Indemnified
|
||||
Contributor must: a) promptly notify the Commercial Contributor in
|
||||
writing of such claim, and b) allow the Commercial Contributor to control,
|
||||
and cooperate with the Commercial Contributor in, the defense and any
|
||||
related settlement negotiations. The Indemnified Contributor may
|
||||
participate in any such claim at its own expense.
|
||||
|
||||
For example, a Contributor might include the Program in a commercial
|
||||
product offering, Product X. That Contributor is then a Commercial
|
||||
Contributor. If that Commercial Contributor then makes performance
|
||||
claims, or offers warranties related to Product X, those performance
|
||||
claims and warranties are such Commercial Contributor's responsibility
|
||||
alone. Under this section, the Commercial Contributor would have to
|
||||
defend claims against the other Contributors related to those performance
|
||||
claims and warranties, and if a court requires any other Contributor to
|
||||
pay any damages as a result, the Commercial Contributor must pay
|
||||
those damages.
|
||||
|
||||
5. NO WARRANTY
|
||||
|
||||
EXCEPT AS EXPRESSLY SET FORTH IN THIS AGREEMENT, AND TO THE EXTENT
|
||||
PERMITTED BY APPLICABLE LAW, THE PROGRAM IS PROVIDED ON AN "AS IS"
|
||||
BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, EITHER EXPRESS OR
|
||||
IMPLIED INCLUDING, WITHOUT LIMITATION, ANY WARRANTIES OR CONDITIONS OF
|
||||
TITLE, NON-INFRINGEMENT, MERCHANTABILITY OR FITNESS FOR A PARTICULAR
|
||||
PURPOSE. Each Recipient is solely responsible for determining the
|
||||
appropriateness of using and distributing the Program and assumes all
|
||||
risks associated with its exercise of rights under this Agreement,
|
||||
including but not limited to the risks and costs of program errors,
|
||||
compliance with applicable laws, damage to or loss of data, programs
|
||||
or equipment, and unavailability or interruption of operations.
|
||||
|
||||
6. DISCLAIMER OF LIABILITY
|
||||
|
||||
EXCEPT AS EXPRESSLY SET FORTH IN THIS AGREEMENT, AND TO THE EXTENT
|
||||
PERMITTED BY APPLICABLE LAW, NEITHER RECIPIENT NOR ANY CONTRIBUTORS
|
||||
SHALL HAVE ANY LIABILITY FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING WITHOUT LIMITATION LOST
|
||||
PROFITS), HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||
ARISING IN ANY WAY OUT OF THE USE OR DISTRIBUTION OF THE PROGRAM OR THE
|
||||
EXERCISE OF ANY RIGHTS GRANTED HEREUNDER, EVEN IF ADVISED OF THE
|
||||
POSSIBILITY OF SUCH DAMAGES.
|
||||
|
||||
7. GENERAL
|
||||
|
||||
If any provision of this Agreement is invalid or unenforceable under
|
||||
applicable law, it shall not affect the validity or enforceability of
|
||||
the remainder of the terms of this Agreement, and without further
|
||||
action by the parties hereto, such provision shall be reformed to the
|
||||
minimum extent necessary to make such provision valid and enforceable.
|
||||
|
||||
If Recipient institutes patent litigation against any entity
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that the
|
||||
Program itself (excluding combinations of the Program with other software
|
||||
or hardware) infringes such Recipient's patent(s), then such Recipient's
|
||||
rights granted under Section 2(b) shall terminate as of the date such
|
||||
litigation is filed.
|
||||
|
||||
All Recipient's rights under this Agreement shall terminate if it
|
||||
fails to comply with any of the material terms or conditions of this
|
||||
Agreement and does not cure such failure in a reasonable period of
|
||||
time after becoming aware of such noncompliance. If all Recipient's
|
||||
rights under this Agreement terminate, Recipient agrees to cease use
|
||||
and distribution of the Program as soon as reasonably practicable.
|
||||
However, Recipient's obligations under this Agreement and any licenses
|
||||
granted by Recipient relating to the Program shall continue and survive.
|
||||
|
||||
Everyone is permitted to copy and distribute copies of this Agreement,
|
||||
but in order to avoid inconsistency the Agreement is copyrighted and
|
||||
may only be modified in the following manner. The Agreement Steward
|
||||
reserves the right to publish new versions (including revisions) of
|
||||
this Agreement from time to time. No one other than the Agreement
|
||||
Steward has the right to modify this Agreement. The Eclipse Foundation
|
||||
is the initial Agreement Steward. The Eclipse Foundation may assign the
|
||||
responsibility to serve as the Agreement Steward to a suitable separate
|
||||
entity. Each new version of the Agreement will be given a distinguishing
|
||||
version number. The Program (including Contributions) may always be
|
||||
Distributed subject to the version of the Agreement under which it was
|
||||
received. In addition, after a new version of the Agreement is published,
|
||||
Contributor may elect to Distribute the Program (including its
|
||||
Contributions) under the new version.
|
||||
|
||||
Except as expressly stated in Sections 2(a) and 2(b) above, Recipient
|
||||
receives no rights or licenses to the intellectual property of any
|
||||
Contributor under this Agreement, whether expressly, by implication,
|
||||
estoppel or otherwise. All rights in the Program not expressly granted
|
||||
under this Agreement are reserved. Nothing in this Agreement is intended
|
||||
to be enforceable by any entity that is not a Contributor or Recipient.
|
||||
No third-party beneficiary rights are created under this Agreement.
|
||||
|
||||
Exhibit A - Form of Secondary Licenses Notice
|
||||
|
||||
"This Source Code may also be made available under the following
|
||||
Secondary Licenses when the conditions for such availability set forth
|
||||
in the Eclipse Public License, v. 2.0 are satisfied: GNU General Public
|
||||
License as published by the Free Software Foundation, either version 2
|
||||
of the License, or (at your option) any later version, with the GNU
|
||||
Classpath Exception which is available at
|
||||
https://www.gnu.org/software/classpath/license.html."
|
||||
|
||||
Simply including a copy of this Agreement, including this Exhibit A
|
||||
is not sufficient to license the Source Code under Secondary Licenses.
|
||||
|
||||
If it is not possible or desirable to put the notice in a particular
|
||||
file, then You may include the notice in a location (such as a LICENSE
|
||||
file in a relevant directory) where a recipient would be likely to
|
||||
look for such a notice.
|
||||
|
||||
You may add additional accurate notices of copyright ownership.
|
71
README.md
Normal file
71
README.md
Normal file
|
@ -0,0 +1,71 @@
|
|||
# The Old Dog and Duck
|
||||
|
||||
A Clojure library designed to implement the ActivityPub protocol, obviously.
|
||||
|
||||
## Introduction
|
||||
|
||||
The Old Dog and Duck is clearly a pub, and it's a pub related to an activity; to whit, hunting ducks with dogs. Yes, of course one could also hunt dogs with ducks, but in practice that doesn't work so well. The point isn't whether or not I approve of hunting ducks with dogs (or vice versa); to be clear, I don't. The point is that it's a pub related to an activity, and is therefore an [ActivityPub](https://www.w3.org/TR/activitypub/).
|
||||
|
||||
Are we clear?
|
||||
|
||||
Good.
|
||||
|
||||
Let us proceed.
|
||||
|
||||
**The Old Dog and Duck** is intended to be a set of libraries to enable people to build stuff which interacts with ActivityPub. It isn't intended to be a replacement for, or clone of, Mastodon. I do think I might implement my own ActivityPub server on top of The Old Dog and Duck, that specifically might allow for user-pluggable feed-sorting algorithms and with my own user interface/user experience take, but that project is not this project.
|
||||
|
||||
## Architecture
|
||||
|
||||
There are a number of separate concerns required to implement ActivityPub. They include
|
||||
|
||||
1. Parsing ActivityStreams messages received from peers and from clients;
|
||||
2. Persisting ActivityStreams objects;
|
||||
3. Delivering ActivityStreams objects to peers;
|
||||
4. Delivering ActivityStreams objects to clients.
|
||||
|
||||
**NOTE THAT** what Mastodon delivers to clients is not actually in ActivityStreams format; this seems to be an ad-hoc hack that's just never been fixed and has therefore become a de-facto standard for communication between ActivityPub hosts and their clients.
|
||||
|
||||
My proposal would be to deliver exactly the same ActivityStreams format to my client as to other servers. There may be a valid reason for not doing this, but if there is I will discover it in due course.
|
||||
|
||||
## Proposed dog-and-duck libraries
|
||||
|
||||
**NOTE THAT** at the present stage all the proposed libraries are in one package, namely this package, but that it is proposed that in future they will form separate libraries in separate packages.
|
||||
|
||||
### Bar
|
||||
|
||||
Where conversations happen. Handle interactions with clients.
|
||||
|
||||
### Cellar
|
||||
|
||||
Where things are stored. Persistance for ActivityStreams objects; I may at least initially simply copy the Mastodon postgres schema, but equally I may not.
|
||||
|
||||
### Pantry
|
||||
|
||||
Where deliveries are ordered and arrive; and from where deliveries onwards are despatched. Handle interactions with peers.
|
||||
|
||||
### Quack
|
||||
|
||||
Duck-typing for ActivityStreams objects.
|
||||
|
||||
### Scratch
|
||||
|
||||
What the dog does when bored. Essentially, a place where I can learn how to make this stuff work, but perhaps eventually an ActivityPub server in its own right.
|
||||
|
||||
## Usage
|
||||
|
||||
FIXME
|
||||
|
||||
## License
|
||||
|
||||
Copyright © 2022 FIXME
|
||||
|
||||
This program and the accompanying materials are made available under the
|
||||
terms of the Eclipse Public License 2.0 which is available at
|
||||
http://www.eclipse.org/legal/epl-2.0.
|
||||
|
||||
This Source Code may also be made available under the following Secondary
|
||||
Licenses when the conditions for such availability set forth in the Eclipse
|
||||
Public License, v. 2.0 are satisfied: GNU General Public License as published by
|
||||
the Free Software Foundation, either version 2 of the License, or (at your
|
||||
option) any later version, with the GNU Classpath Exception which is available
|
||||
at https://www.gnu.org/software/classpath/license.html.
|
110
doc/Using_ActivityPub.md
Normal file
110
doc/Using_ActivityPub.md
Normal file
|
@ -0,0 +1,110 @@
|
|||
# Using ActivityPub
|
||||
|
||||
user=> (require '[clj-activitypub.core :as activitypub])
|
||||
nil
|
||||
user=> (require '[clj-activitypub.webfinger :as webfinger])
|
||||
nil
|
||||
user=> (require '[clojure.walk :refer [keywordize-keys]])
|
||||
nil
|
||||
user=> (require '[clojure.pprint :refer [pprint]])
|
||||
nil
|
||||
user=> (def base-domain "mastodon.scot")
|
||||
#'user/base-domain
|
||||
user=> (def account-handle "@simon_brooke@mastodon.scot")
|
||||
#'user/account-handle
|
||||
user=> (in-ns 'user)
|
||||
#object[clojure.lang.Namespace 0x525575 "user"]
|
||||
user=> (activitypub/parse-account account-handle )
|
||||
{:domain "mastodon.scot", :username "simon_brooke"}
|
||||
user=> (map *1 [:domain :username])
|
||||
("mastodon.scot" "simon_brooke")
|
||||
user=> (apply webfinger/fetch-user-id *1)
|
||||
"https://mastodon.scot/users/simon_brooke"
|
||||
user=> (activitypub/fetch-user *1)
|
||||
{"followers" "https://mastodon.scot/users/simon_brooke/followers", "inbox" "https://mastodon.scot/users/simon_brooke/inbox", "url" "https://mastodon.scot/@simon_brooke", "@context" ["https://www.w3.org/ns/activitystreams" "https://w3id.org/security/v1" {"identityKey" {"@type" "@id", "@id" "toot:identityKey"}, "EncryptedMessage" "toot:EncryptedMessage", "Ed25519Key" "toot:Ed25519Key", "devices" {"@type" "@id", "@id" "toot:devices"}, "manuallyApprovesFollowers" "as:manuallyApprovesFollowers", "schema" "http://schema.org#", "PropertyValue" "schema:PropertyValue", "Curve25519Key" "toot:Curve25519Key", "claim" {"@type" "@id", "@id" "toot:claim"}, "value" "schema:value", "Hashtag" "as:Hashtag", "movedTo" {"@id" "as:movedTo", "@type" "@id"}, "discoverable" "toot:discoverable", "messageType" "toot:messageType", "messageFranking" "toot:messageFranking", "cipherText" "toot:cipherText", "toot" "http://joinmastodon.org/ns#", "alsoKnownAs" {"@id" "as:alsoKnownAs", "@type" "@id"}, "featured" {"@id" "toot:featured", "@type" "@id"}, "featuredTags" {"@id" "toot:featuredTags", "@type" "@id"}, "Ed25519Signature" "toot:Ed25519Signature", "focalPoint" {"@container" "@list", "@id" "toot:focalPoint"}, "fingerprintKey" {"@type" "@id", "@id" "toot:fingerprintKey"}, "Device" "toot:Device", "publicKeyBase64" "toot:publicKeyBase64", "deviceId" "toot:deviceId", "suspended" "toot:suspended"}], "devices" "https://mastodon.scot/users/simon_brooke/collections/devices", "manuallyApprovesFollowers" false, "image" {"type" "Image", "mediaType" "image/jpeg", "url" "https://media.mastodon.scot/mastodon-scot-public/accounts/headers/109/252/274/874/045/781/original/e1f1823c4361fa27.jpg"}, "endpoints" {"sharedInbox" "https://mastodon.scot/inbox"}, "id" "https://mastodon.scot/users/simon_brooke", "publicKey" {"id" "https://mastodon.scot/users/simon_brooke#main-key", "owner" "https://mastodon.scot/users/simon_brooke", "publicKeyPem" "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA2/6GgLJgJlPhhqFm1tUQ\noSLnWxhDwq4HlZIHrBsVjkSvUAnHKqq42Q/hta+fkWB8rmTFpmjLXDj/Fi0uejvT\nBc+KrLwfX/yR8+G87afGCRS3CaumoLJ7zkBIlsFzIKMoIke1D3QuHX95yGGXs+hp\nmyxt/+CXRyZjK7u9NG7SMRUlpwvOlpD12Aei35Nb8NSr03JvY8/WVMIbWrecyI0b\nAlwj6axxHx7J15Yo+aEtKzZ2OFKXf+sh0QF9BEnYcmVKYlR6kiOglLFHKdCBUSYi\ni9Flv00TydqlGvR5fpShBqORiy0M/FVtNXlz2sNBEsGB2meipkjh+cRLzTbYo4KL\nJwIDAQAB\n-----END PUBLIC KEY-----\n"}, "summary" "<p>Anarcho-syndicalist, autistic, crofter, cyclist, depressive, entrepreneur, geek, Zapatista. Politics & environment, especially <a href=\"https://mastodon.scot/tags/LandReform\" class=\"mention hashtag\" rel=\"tag\">#<span>LandReform</span></a>. he/him.</p><p>Twitter: <span class=\"h-card\"><a href=\"https://mastodon.scot/@simon_brooke\" class=\"u-url mention\">@<span>simon_brooke</span></a></span><br />GitHub: simon-brooke<br />FetLife: Simon_Brooke</p><p>Credo: Life is harsh. What we can do - and what we should do - is strive to make it less harsh for the people around us.</p>", "attachment" [{"type" "PropertyValue", "name" "Home Page", "value" "<a href=\"https://www.journeyman.cc/~simon/\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\"><span class=\"invisible\">https://www.</span><span class=\"\">journeyman.cc/~simon/</span><span class=\"invisible\"></span></a>"}], "name" "Simon Brooke", "tag" [{"type" "Hashtag", "href" "https://mastodon.scot/tags/landreform", "name" "#landreform"}], "published" "2022-10-29T00:00:00Z", "preferredUsername" "simon_brooke", "discoverable" true, "alsoKnownAs" ["https://mastodon.social/users/simon_brooke"], "featured" "https://mastodon.scot/users/simon_brooke/collections/featured", "featuredTags" "https://mastodon.scot/users/simon_brooke/collections/tags", "type" "Person", "outbox" "https://mastodon.scot/users/simon_brooke/outbox", "following" "https://mastodon.scot/users/simon_brooke/following", "icon" {"type" "Image", "mediaType" "image/png", "url" "https://media.mastodon.scot/mastodon-scot-public/accounts/avatars/109/252/274/874/045/781/original/172e8f7530627e87.png"}}
|
||||
user=> (def sb (keywordize-keys *1))
|
||||
#'user/sb
|
||||
user=> (:outbox sb)
|
||||
"https://mastodon.scot/users/simon_brooke/outbox"
|
||||
user=> (require '[clojure.data.json :as json])
|
||||
nil
|
||||
user=> (slurp (:outbox sb))
|
||||
Execution error (IOException) at sun.net.www.protocol.http.HttpURLConnection/getInputStream0 (HttpURLConnection.java:1894).
|
||||
Server returned HTTP response code: 403 for URL: https://mastodon.scot/users/simon_brooke/outbox
|
||||
user=> (pprint sb)
|
||||
{:inbox "https://mastodon.scot/users/simon_brooke/inbox",
|
||||
:name "Simon Brooke",
|
||||
:@context
|
||||
["https://www.w3.org/ns/activitystreams"
|
||||
"https://w3id.org/security/v1"
|
||||
{:schema "http://schema.org#",
|
||||
:messageType "toot:messageType",
|
||||
:messageFranking "toot:messageFranking",
|
||||
:identityKey {:@type "@id", :@id "toot:identityKey"},
|
||||
:Hashtag "as:Hashtag",
|
||||
:deviceId "toot:deviceId",
|
||||
:publicKeyBase64 "toot:publicKeyBase64",
|
||||
:value "schema:value",
|
||||
:Ed25519Key "toot:Ed25519Key",
|
||||
:featured {:@id "toot:featured", :@type "@id"},
|
||||
:Curve25519Key "toot:Curve25519Key",
|
||||
:discoverable "toot:discoverable",
|
||||
:focalPoint {:@container "@list", :@id "toot:focalPoint"},
|
||||
:suspended "toot:suspended",
|
||||
:fingerprintKey {:@type "@id", :@id "toot:fingerprintKey"},
|
||||
:Ed25519Signature "toot:Ed25519Signature",
|
||||
:cipherText "toot:cipherText",
|
||||
:EncryptedMessage "toot:EncryptedMessage",
|
||||
:alsoKnownAs {:@id "as:alsoKnownAs", :@type "@id"},
|
||||
:featuredTags {:@id "toot:featuredTags", :@type "@id"},
|
||||
:devices {:@type "@id", :@id "toot:devices"},
|
||||
:toot "http://joinmastodon.org/ns#",
|
||||
:movedTo {:@id "as:movedTo", :@type "@id"},
|
||||
:Device "toot:Device",
|
||||
:PropertyValue "schema:PropertyValue",
|
||||
:manuallyApprovesFollowers "as:manuallyApprovesFollowers",
|
||||
:claim {:@type "@id", :@id "toot:claim"}}],
|
||||
:featured
|
||||
"https://mastodon.scot/users/simon_brooke/collections/featured",
|
||||
:type "Person",
|
||||
:discoverable true,
|
||||
:icon
|
||||
{:type "Image",
|
||||
:mediaType "image/png",
|
||||
:url
|
||||
"https://media.mastodon.scot/mastodon-scot-public/accounts/avatars/109/252/274/874/045/781/original/172e8f7530627e87.png"},
|
||||
:following "https://mastodon.scot/users/simon_brooke/following",
|
||||
:summary
|
||||
"<p>Anarcho-syndicalist, autistic, crofter, cyclist, depressive, entrepreneur, geek, Zapatista. Politics & environment, especially <a href=\"https://mastodon.scot/tags/LandReform\" class=\"mention hashtag\" rel=\"tag\">#<span>LandReform</span></a>. he/him.</p><p>Twitter: <span class=\"h-card\"><a href=\"https://mastodon.scot/@simon_brooke\" class=\"u-url mention\">@<span>simon_brooke</span></a></span><br />GitHub: simon-brooke<br />FetLife: Simon_Brooke</p><p>Credo: Life is harsh. What we can do - and what we should do - is strive to make it less harsh for the people around us.</p>",
|
||||
:publicKey
|
||||
{:id "https://mastodon.scot/users/simon_brooke#main-key",
|
||||
:owner "https://mastodon.scot/users/simon_brooke",
|
||||
:publicKeyPem
|
||||
"-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA2/6GgLJgJlPhhqFm1tUQ\noSLnWxhDwq4HlZIHrBsVjkSvUAnHKqq42Q/hta+fkWB8rmTFpmjLXDj/Fi0uejvT\nBc+KrLwfX/yR8+G87afGCRS3CaumoLJ7zkBIlsFzIKMoIke1D3QuHX95yGGXs+hp\nmyxt/+CXRyZjK7u9NG7SMRUlpwvOlpD12Aei35Nb8NSr03JvY8/WVMIbWrecyI0b\nAlwj6axxHx7J15Yo+aEtKzZ2OFKXf+sh0QF9BEnYcmVKYlR6kiOglLFHKdCBUSYi\ni9Flv00TydqlGvR5fpShBqORiy0M/FVtNXlz2sNBEsGB2meipkjh+cRLzTbYo4KL\nJwIDAQAB\n-----END PUBLIC KEY-----\n"},
|
||||
:endpoints {:sharedInbox "https://mastodon.scot/inbox"},
|
||||
:preferredUsername "simon_brooke",
|
||||
:id "https://mastodon.scot/users/simon_brooke",
|
||||
:alsoKnownAs ["https://mastodon.social/users/simon_brooke"],
|
||||
:outbox "https://mastodon.scot/users/simon_brooke/outbox",
|
||||
:url "https://mastodon.scot/@simon_brooke",
|
||||
:featuredTags
|
||||
"https://mastodon.scot/users/simon_brooke/collections/tags",
|
||||
:devices
|
||||
"https://mastodon.scot/users/simon_brooke/collections/devices",
|
||||
:image
|
||||
{:type "Image",
|
||||
:mediaType "image/jpeg",
|
||||
:url
|
||||
"https://media.mastodon.scot/mastodon-scot-public/accounts/headers/109/252/274/874/045/781/original/e1f1823c4361fa27.jpg"},
|
||||
:tag
|
||||
[{:type "Hashtag",
|
||||
:href "https://mastodon.scot/tags/landreform",
|
||||
:name "#landreform"}],
|
||||
:followers "https://mastodon.scot/users/simon_brooke/followers",
|
||||
:published "2022-10-29T00:00:00Z",
|
||||
:manuallyApprovesFollowers false,
|
||||
:attachment
|
||||
[{:type "PropertyValue",
|
||||
:name "Home Page",
|
||||
:value
|
||||
"<a href=\"https://www.journeyman.cc/~simon/\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\"><span class=\"invisible\">https://www.</span><span class=\"\">journeyman.cc/~simon/</span><span class=\"invisible\"></span></a>"}]}
|
3
doc/intro.md
Normal file
3
doc/intro.md
Normal file
|
@ -0,0 +1,3 @@
|
|||
# Introduction to dog-and-duck
|
||||
|
||||
TODO: write [great documentation](http://jacobian.org/writing/what-to-write/)
|
15
project.clj
Normal file
15
project.clj
Normal file
|
@ -0,0 +1,15 @@
|
|||
(defproject dog-and-duck "0.1.0-SNAPSHOT"
|
||||
:description "FIXME: write description"
|
||||
:url "http://example.com/FIXME"
|
||||
:license {:name "EPL-2.0 OR GPL-2.0-or-later WITH Classpath-exception-2.0"
|
||||
:url "https://www.eclipse.org/legal/epl-2.0/"}
|
||||
:dependencies [[org.clojure/clojure "1.10.3"]
|
||||
[org.clojure/data.json "2.4.0"]
|
||||
[org.clojure/math.numeric-tower "0.0.5"]
|
||||
[org.clojure/spec.alpha "0.3.218"]
|
||||
[mvxcvi/clj-pgp "1.1.0"]
|
||||
[org.bouncycastle/bcpkix-jdk18on "1.72"] ;; required by clj-activitypub
|
||||
[clj-http "3.12.3"] ;; required by clj-activitypub
|
||||
[cheshire "5.11.0"] ;; if this is not present, clj-http/client errors with 'json-enabled?'
|
||||
]
|
||||
:repl-options {:init-ns dog-and-duck.scratch.core})
|
4
resources/README.md
Normal file
4
resources/README.md
Normal file
|
@ -0,0 +1,4 @@
|
|||
# dog-and-duck/resources
|
||||
|
||||
You should clone git@github.com:w3c-social/activitystreams-test-documents.git
|
||||
into this directory.
|
1
resources/feed.json
Normal file
1
resources/feed.json
Normal file
File diff suppressed because one or more lines are too long
3
src/clj_activitypub/README.md
Normal file
3
src/clj_activitypub/README.md
Normal file
|
@ -0,0 +1,3 @@
|
|||
# NOTE
|
||||
|
||||
Files in this directory are copied from [Jahfer's clj-activitypub library](https://github.com/jahfer/clj-activitypub). If and when Jahfer issues a release of that library, this directory will be deleted and a dependency on that library will be added to the project.
|
147
src/clj_activitypub/core.clj
Normal file
147
src/clj_activitypub/core.clj
Normal file
|
@ -0,0 +1,147 @@
|
|||
(ns clj-activitypub.core
|
||||
(:require [clj-activitypub.internal.crypto :as crypto]
|
||||
[clj-activitypub.internal.thread-cache :as thread-cache]
|
||||
[clj-activitypub.internal.http-util :as http]
|
||||
[clj-http.client :as client]
|
||||
[clojure.string :as str]))
|
||||
|
||||
(defn config
|
||||
"Creates hash of computed data relevant for most ActivityPub utilities."
|
||||
[{:keys [domain username username-route public-key private-key]
|
||||
:or {username-route "/users/"
|
||||
public-key nil
|
||||
private-key nil}}]
|
||||
(let [base-url (str "https://" domain)]
|
||||
{:domain domain
|
||||
:base-url base-url
|
||||
:username username
|
||||
:user-id (str base-url username-route username)
|
||||
:public-key public-key
|
||||
:private-key (when private-key
|
||||
(crypto/private-key private-key))}))
|
||||
|
||||
(defn parse-account
|
||||
"Given an ActivityPub handle (e.g. @jahfer@mastodon.social), produces
|
||||
a map containing {:domain ... :username ...}."
|
||||
[handle]
|
||||
(let [[username domain] (filter #(not (str/blank? %))
|
||||
(str/split handle #"@"))]
|
||||
{:domain domain :username username}))
|
||||
|
||||
(def ^:private user-cache (thread-cache/make))
|
||||
(defn fetch-user
|
||||
"Fetches the customer account details located at user-id from a remote
|
||||
server. Will return cached results if they exist in memory."
|
||||
[user-id]
|
||||
((:get-v user-cache)
|
||||
user-id
|
||||
#(:body
|
||||
(client/get user-id {:as :json-string-keys
|
||||
:throw-exceptions false
|
||||
:ignore-unknown-host? true
|
||||
:headers {"Accept" "application/activity+json"}}))))
|
||||
|
||||
(defn actor
|
||||
"Accepts a config, and returns a map in the form expected by the ActivityPub
|
||||
spec. See https://www.w3.org/TR/activitypub/#actor-objects for reference."
|
||||
[{:keys [user-id username public-key]}]
|
||||
{"@context" ["https://www.w3.org/ns/activitystreams"
|
||||
"https://w3id.org/security/v1"]
|
||||
:id user-id
|
||||
:type "Person"
|
||||
:preferredUsername username
|
||||
:inbox (str user-id "/inbox")
|
||||
:outbox (str user-id "/outbox")
|
||||
:publicKey {:id (str user-id "#main-key")
|
||||
:owner user-id
|
||||
:publicKeyPem (or public-key "")}})
|
||||
|
||||
(def signature-headers ["(request-target)" "host" "date" "digest"])
|
||||
|
||||
(defn- str-for-signature [headers]
|
||||
(let [headers-xf (reduce-kv
|
||||
(fn [m k v]
|
||||
(assoc m (str/lower-case k) v)) {} headers)]
|
||||
(->> signature-headers
|
||||
(select-keys headers-xf)
|
||||
(reduce-kv (fn [coll k v] (conj coll (str k ": " v))) [])
|
||||
(interpose "\n")
|
||||
(apply str))))
|
||||
|
||||
(defn gen-signature-header
|
||||
"Generates a HTTP Signature string based on the provided map of headers."
|
||||
[config headers]
|
||||
(let [{:keys [user-id private-key]} config
|
||||
string-to-sign (str-for-signature headers)
|
||||
signature (crypto/base64-encode (crypto/sign string-to-sign private-key))
|
||||
sig-header-keys {"keyId" user-id
|
||||
"headers" (str/join " " signature-headers)
|
||||
"signature" signature}]
|
||||
(->> sig-header-keys
|
||||
(reduce-kv (fn [m k v]
|
||||
(conj m (str k "=" "\"" v "\""))) [])
|
||||
(interpose ",")
|
||||
(apply str))))
|
||||
|
||||
(defn auth-headers
|
||||
"Given a config and request map of {:body ... :headers ...}, returns the
|
||||
original set of headers with Signature and Digest attributes appended."
|
||||
[config {:keys [body headers]}]
|
||||
(let [digest (http/digest body)
|
||||
h (-> headers
|
||||
(assoc "Digest" digest)
|
||||
(assoc "(request-target)" "post /inbox"))]
|
||||
(assoc headers
|
||||
"Signature" (gen-signature-header config h)
|
||||
"Digest" digest)))
|
||||
|
||||
(defmulti obj
|
||||
"Produces a map representing an ActivityPub object which can be serialized
|
||||
directly to JSON in the form expected by the ActivityStreams 2.0 spec.
|
||||
See https://www.w3.org/TR/activitystreams-vocabulary/ for reference."
|
||||
(fn [_config object-data] (:type object-data)))
|
||||
|
||||
(defmethod obj :note
|
||||
[{:keys [user-id]}
|
||||
{:keys [id published inReplyTo content to]
|
||||
:or {published (http/date)
|
||||
inReplyTo ""
|
||||
to "https://www.w3.org/ns/activitystreams#Public"}}]
|
||||
{"id" (str user-id "/notes/" id)
|
||||
"type" "Note"
|
||||
"published" published
|
||||
"attributedTo" user-id
|
||||
"inReplyTo" inReplyTo
|
||||
"content" content
|
||||
"to" to})
|
||||
|
||||
(defmulti activity
|
||||
"Produces a map representing an ActivityPub activity which can be serialized
|
||||
directly to JSON in the form expected by the ActivityStreams 2.0 spec.
|
||||
See https://www.w3.org/TR/activitystreams-vocabulary/ for reference."
|
||||
(fn [_config activity-type _data] activity-type))
|
||||
|
||||
(defmethod activity :create [{:keys [user-id]} _ data]
|
||||
{"@context" ["https://www.w3.org/ns/activitystreams"
|
||||
"https://w3id.org/security/v1"]
|
||||
"type" "Create"
|
||||
"actor" user-id
|
||||
"object" data})
|
||||
|
||||
(defmethod activity :delete [{:keys [user-id]} _ data]
|
||||
{"@context" ["https://www.w3.org/ns/activitystreams"
|
||||
"https://w3id.org/security/v1"]
|
||||
"type" "Delete"
|
||||
"actor" user-id
|
||||
"object" data})
|
||||
|
||||
(defn with-config
|
||||
"Returns curried forms of the #activity and #obj multimethods in the form
|
||||
{:activity ... :obj ...}, with the initial parameter set to config."
|
||||
[config]
|
||||
(let [f (juxt
|
||||
#(partial activity %)
|
||||
#(partial obj %))
|
||||
[activity-fn obj-fn] (f config)]
|
||||
{:activity activity-fn
|
||||
:obj obj-fn}))
|
36
src/clj_activitypub/internal/crypto.clj
Normal file
36
src/clj_activitypub/internal/crypto.clj
Normal file
|
@ -0,0 +1,36 @@
|
|||
(ns clj-activitypub.internal.crypto
|
||||
(:require [clojure.java.io :as io])
|
||||
(:import (java.util Base64)
|
||||
(java.security MessageDigest SecureRandom Signature)))
|
||||
|
||||
(java.security.Security/addProvider
|
||||
(org.bouncycastle.jce.provider.BouncyCastleProvider.))
|
||||
|
||||
(defn- keydata [reader]
|
||||
(->> reader
|
||||
(org.bouncycastle.openssl.PEMParser.)
|
||||
(.readObject)))
|
||||
|
||||
(defn- pem-string->key-pair [string]
|
||||
(let [kd (keydata (io/reader (.getBytes string)))]
|
||||
(.getKeyPair (org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter.) kd)))
|
||||
|
||||
(defn private-key [private-pem-str]
|
||||
(-> private-pem-str
|
||||
(pem-string->key-pair)
|
||||
(.getPrivate)))
|
||||
|
||||
(defn base64-encode [bytes]
|
||||
(.encodeToString (Base64/getEncoder) bytes))
|
||||
|
||||
(defn sha256-base64 [data]
|
||||
(let [digest (.digest (MessageDigest/getInstance "SHA-256") (.getBytes data))]
|
||||
(base64-encode digest)))
|
||||
|
||||
(defn sign [data private-key]
|
||||
(let [bytes (.getBytes data)
|
||||
signer (doto (Signature/getInstance "SHA256withRSA")
|
||||
(.initSign private-key (SecureRandom.))
|
||||
(.update bytes))]
|
||||
(.sign signer)))
|
||||
|
25
src/clj_activitypub/internal/http_util.clj
Normal file
25
src/clj_activitypub/internal/http_util.clj
Normal file
|
@ -0,0 +1,25 @@
|
|||
(ns clj-activitypub.internal.http-util
|
||||
(:require [clj-activitypub.internal.crypto :as crypto])
|
||||
(:import (java.net URLEncoder)
|
||||
(java.time OffsetDateTime ZoneOffset)
|
||||
(java.time.format DateTimeFormatter)))
|
||||
|
||||
(defn encode-url-params [params]
|
||||
(->> params
|
||||
(reduce-kv
|
||||
(fn [coll k v]
|
||||
(conj coll
|
||||
(str (URLEncoder/encode (name k)) "=" (URLEncoder/encode (str v)))))
|
||||
[])
|
||||
(interpose "&")
|
||||
(apply str)))
|
||||
|
||||
(defn date []
|
||||
(-> (OffsetDateTime/now (ZoneOffset/UTC))
|
||||
(.format DateTimeFormatter/RFC_1123_DATE_TIME)))
|
||||
|
||||
(defn digest
|
||||
"Accepts body from HTTP request and generates string
|
||||
for use in HTTP `Digest` request header."
|
||||
[body]
|
||||
(str "sha-256=" (crypto/sha256-base64 body)))
|
44
src/clj_activitypub/internal/thread_cache.clj
Normal file
44
src/clj_activitypub/internal/thread_cache.clj
Normal file
|
@ -0,0 +1,44 @@
|
|||
(ns clj-activitypub.internal.thread-cache)
|
||||
|
||||
(defn- current-time
|
||||
"Returns current time using UNIX epoch."
|
||||
[]
|
||||
(System/currentTimeMillis))
|
||||
|
||||
(defn- update-read-at [store k v]
|
||||
(dosync
|
||||
(commute store assoc k
|
||||
(merge v {:read-at (current-time)}))))
|
||||
|
||||
(defn make
|
||||
"Creates a thread-local cache."
|
||||
([] (make false))
|
||||
([cache-if-nil]
|
||||
(let [store (ref {})]
|
||||
(letfn [(cache-kv ([k v]
|
||||
(dosync
|
||||
(commute store assoc k
|
||||
{:write-at (current-time)
|
||||
:read-at (current-time)
|
||||
:value v})
|
||||
v)))
|
||||
(get-v ([k]
|
||||
(when-let [data (get @store k)]
|
||||
(update-read-at store k data)
|
||||
(:value data)))
|
||||
([k compute-fn]
|
||||
(let [storage @store]
|
||||
(if (contains? storage k)
|
||||
(get-v k)
|
||||
(let [v (compute-fn)]
|
||||
(when (or (not (nil? v)) cache-if-nil)
|
||||
(cache-kv k v)
|
||||
(get-v k)))))))
|
||||
(lru ([]
|
||||
(mapv
|
||||
(fn [[k v]] [k (:value v)])
|
||||
(sort-by #(-> % val :read-at) < @store))))]
|
||||
{:cache-kv cache-kv
|
||||
:get-v get-v
|
||||
:cache-if-nil cache-if-nil
|
||||
:lru lru}))))
|
32
src/clj_activitypub/webfinger.clj
Normal file
32
src/clj_activitypub/webfinger.clj
Normal file
|
@ -0,0 +1,32 @@
|
|||
(ns clj-activitypub.webfinger
|
||||
(:require [clj-http.client :as client]
|
||||
[clj-activitypub.internal.http-util :as http]
|
||||
[clj-activitypub.internal.thread-cache :as thread-cache]))
|
||||
|
||||
(def remote-uri-path "/.well-known/webfinger")
|
||||
|
||||
(defn- resource-str [domain username]
|
||||
(str "acct:" username "@" domain))
|
||||
|
||||
(defn resource-url
|
||||
"Builds a URL pointing to the user's account on the remote server."
|
||||
[domain username & [params]]
|
||||
(let [resource (resource-str domain username)
|
||||
query-str (http/encode-url-params (merge params {:resource resource}))]
|
||||
(str "https://" domain remote-uri-path "?" query-str)))
|
||||
|
||||
(def ^:private user-id-cache
|
||||
(thread-cache/make))
|
||||
|
||||
(defn fetch-user-id
|
||||
"Follows the webfinger request to a remote domain, retrieving the ID of the requested
|
||||
account. Typically returns a string in the form of a URL."
|
||||
[domain username]
|
||||
((:get-v user-id-cache)
|
||||
(str domain "@" username) ;; cache key
|
||||
(fn []
|
||||
(let [response (some-> (resource-url domain username {:rel "self"})
|
||||
(client/get {:as :json :throw-exceptions false :ignore-unknown-host? true}))]
|
||||
(some->> response :body :links
|
||||
(some #(when (= (:type %) "application/activity+json") %))
|
||||
:href)))))
|
63
src/dog_and_duck/quack/quack.clj
Normal file
63
src/dog_and_duck/quack/quack.clj
Normal file
|
@ -0,0 +1,63 @@
|
|||
(ns dog-and-duck.quack.quack
|
||||
"Validator for ActivityPub objects: if it walks like a duck, and it quacks like a duck..."
|
||||
;;(:require [clojure.spec.alpha as s])
|
||||
(:import [java.net URI URISyntaxException]))
|
||||
|
||||
(defn object?
|
||||
"Return `true` iff `x` is recognisably an ActivityStreams object.
|
||||
|
||||
**NOTE THAT** The ActivityStreams spec
|
||||
[says](https://www.w3.org/TR/activitystreams-core/#object):
|
||||
|
||||
> All properties are optional (including the id and type)
|
||||
|
||||
But we are *just not having that*, because otherwise we're flying blind.
|
||||
We *shall* reject objects lacking at least `:type`. Missing `:id` keys are
|
||||
tolerable because they represent transient objects, which we expect to
|
||||
handle."
|
||||
[x]
|
||||
(and (map? x) (:type x) true))
|
||||
|
||||
(object? nil)
|
||||
(object? {:type "test"})
|
||||
|
||||
(defn persistent-object?
|
||||
"`true` iff `x` is a persistent object.
|
||||
|
||||
Transient objects in ActivityPub are not required to have an `id` key, but persistent
|
||||
ones must have a key, and it must be an IRI (but normally a URI)."
|
||||
[x]
|
||||
(try
|
||||
(and (object? x) (uri? (URI. (:id x))))
|
||||
(catch URISyntaxException _ false)))
|
||||
|
||||
(persistent-object? {:type "test" :id "https://mastodon.scot/@barfilfarm"})
|
||||
|
||||
(defn actor?
|
||||
"TODO!"
|
||||
[x]
|
||||
true)
|
||||
|
||||
(def verb?
|
||||
"The set of types we will accept as verbs.
|
||||
|
||||
There's an [explicit set of allowed verbs]
|
||||
(https://www.w3.org/TR/activitystreams-vocabulary/#activity-types)."
|
||||
#{"Accept" "Add" "Announce" "Arrive" "Block" "Create" "Delete" "Dislike"
|
||||
"Flag" "Follow" "Ignore" "Invite" "Join" "Leave" "Like" "Listen" "Move"
|
||||
"Offer" "Question" "Reject" "Read" "Remove" "TentativeAccept"
|
||||
"TentativeReject" "Travel" "Undo" "Update" "View"})
|
||||
|
||||
(defn activity?
|
||||
"`true` iff `x` is an activity, else false.
|
||||
|
||||
see "
|
||||
[x]
|
||||
(try
|
||||
(and (object? x)
|
||||
(uri? (URI. ((keyword "@context") x)))
|
||||
(string? (:summary x))
|
||||
(actor? (:actor x))
|
||||
(verb? (:type x))
|
||||
(or (object? (:object x)) (uri? (URI. x))))
|
||||
(catch URISyntaxException _ false)))
|
6
src/dog_and_duck/scratch/core.clj
Normal file
6
src/dog_and_duck/scratch/core.clj
Normal file
|
@ -0,0 +1,6 @@
|
|||
(ns dog-and-duck.scratch.core)
|
||||
|
||||
(defn foo
|
||||
"I don't do a whole lot."
|
||||
[x]
|
||||
(println x "Hello, World!"))
|
20
src/dog_and_duck/scratch/parser.clj
Normal file
20
src/dog_and_duck/scratch/parser.clj
Normal file
|
@ -0,0 +1,20 @@
|
|||
(ns dog-and-duck.scratch.parser
|
||||
(:require [clojure.walk :refer [keywordize-keys]]
|
||||
[clojure.data.json :as json]
|
||||
[dog-and-duck.quack.quack :as q]))
|
||||
|
||||
(defn clean
|
||||
"Take this `json` input, and return a sequence of ActivityPub objects
|
||||
represented by it."
|
||||
[json]
|
||||
(let [feed (json/read-str json)]
|
||||
(filter
|
||||
q/object?
|
||||
(cond (map? feed) (list (keywordize-keys feed))
|
||||
(coll? feed) (map keywordize-keys feed)))))
|
||||
|
||||
(map :type (map keywordize-keys (json/read-str (slurp "resources/feed.json"))))
|
||||
|
||||
(keys (first (map keywordize-keys (json/read-str (slurp "resources/feed.json")))))
|
||||
|
||||
(q/object? (first (map keywordize-keys (json/read-str (slurp "resources/feed.json")))))
|
44
src/dog_and_duck/scratch/scratch.clj
Normal file
44
src/dog_and_duck/scratch/scratch.clj
Normal file
|
@ -0,0 +1,44 @@
|
|||
(ns dog-and-duck.scratch.scratch
|
||||
"Scratchpad where I try to understand how to do this stuff."
|
||||
(:require [clj-activitypub.core :as activitypub]
|
||||
[clj-activitypub.webfinger :as webfinger]
|
||||
[clj-pgp.core :as pgp]
|
||||
[clj-pgp.keyring :as keyring]
|
||||
[clj-pgp.generate :as pgp-gen]
|
||||
[clojure.walk :refer [keywordize-keys]]
|
||||
[clojure.pprint :refer [pprint]]))
|
||||
|
||||
;;; Use any ActivityPub account handle you like - for example, your own
|
||||
(def account-handle "@simon_brooke@mastodon.scot")
|
||||
|
||||
(def handle (activitypub/parse-account account-handle))
|
||||
(webfinger/fetch-user-id "mastodon.scot" "simon_brooke")
|
||||
(apply webfinger/fetch-user-id (map handle [:domain :username]))
|
||||
|
||||
;;; Retrieve the account details from its home server
|
||||
;;; (`keywordize-keys` is not necessary here but produces a more idiomatic clojure
|
||||
;;; data structure)
|
||||
(def account
|
||||
"Fetch my account to mess with"
|
||||
(let [handle (activitypub/parse-account account-handle)]
|
||||
(keywordize-keys
|
||||
(activitypub/fetch-user
|
||||
(apply webfinger/fetch-user-id (map handle [:domain :username]))))))
|
||||
|
||||
;;; examine what you got back!
|
||||
(:outbox account)
|
||||
|
||||
|
||||
(def rsa (pgp-gen/rsa-keypair-generator 2048))
|
||||
(def kp (pgp-gen/generate-keypair rsa :rsa-general))
|
||||
|
||||
;; how we make a public/private key pair. But this key pair is not the one
|
||||
;; known to mastodon.scot as my key pair, so that doesn't get us very far...
|
||||
;; I think.
|
||||
(let [rsa (pgp-gen/rsa-keypair-generator 2048)
|
||||
kp (pgp-gen/generate-keypair rsa :rsa-general)
|
||||
public (-> kp .getPublicKey .getEncoded)
|
||||
private (-> kp .getPrivateKey .getPrivateKeyDataPacket .getEncoded)]
|
||||
(println (str "Public key: " public))
|
||||
(println (str "Private key: " private))
|
||||
)
|
7
test/dog_and_duck/scratch/core_test.clj
Normal file
7
test/dog_and_duck/scratch/core_test.clj
Normal file
|
@ -0,0 +1,7 @@
|
|||
(ns dog-and-duck.scratch.core-test
|
||||
(:require [clojure.test :refer :all]
|
||||
[dog-and-duck.scratch.parser :refer :all]))
|
||||
|
||||
(deftest a-test
|
||||
(testing "FIXME, I fail."
|
||||
(is (= 0 1))))
|
Loading…
Reference in a new issue