P2P Framework Beyond Apps

An agent-centric, distributed app development framework and social network engine introducing low-code Social DNA for ultra-interoperability and Social Organisms as a replacement for DAOs
Mix Holochain, blockchains, IPFS, centralized services
and many more.

Getting started...

// Add the following to your UI markup:
import '@perspect3vism/ad4m-connect'

  capabilities='[{"with":{"domain":"*","pointers":["*"]},"can": ["*"]}]'

// Retrieve AD4M client when it's ready:

import { Ad4mClient} from  '@perspect3vism/ad4m'

let ad4m: Ad4mClient = await new Promise((resolve, reject) => {
    onAuthStateChanged(async (status) => {
        if (status === "connected_with_capabilities") {
            alert("Connected  to Ad4m!");
        } else {
            alert("Not connected to Ad4m!");

let myPerspective = await ad4m.perspective.add("My Perspective")

const me = await ad4m.agent.me()
const source = me.did
const predicate = Literal.from("thinks").toUrl()
const target = Literal.from("AD4M will be the last social network").toUrl()

const linkExpresion = await myPerspective.add({source, predicate, target})

const uniqueLinkLanguage = await ad4m.languages.applyTemplateAndPublish(
        "uuid": "84a329-77384c-1510fb", 
        "name": "Perspective Diff Sync clone for demo Neighbourhood"

const neighbourhoodUrl = await ad4m.neighbourhood.publishFromPerspective(
    new Perspective()

const joinedNeighbourhood = await ad4m.neighbourhood.joinFromUrl(neighbourhoodUrl)
const myPerspective = await ad4m.perspective.byUUID(joinedNeighbourhood.uuid)

myPerspective.addListener('link-added', (addedLink: LinkExpression) => {
    console.log("Got a new link:", addedLink)

myPerspective.addListener('link-removed', (removedLink: LinkExpression) => {
    console.log("A link was removed:", removedLink)

Core Concepts


…are build around DID - Decentralized Identifier . Users can bring their existing identity or have AD4M create a new one.

Conceptually, AD4M agents are modelled as something that can speak and that can listen. Agents speak by creating Expressions of AD4M Languages, whereby these Expression get signed by the agent's DID key.

AD4M agents also have a publicly shared Perspective, that other agents can see just by resolving their DID URI. This Perspective is like the agents semantic web page, consisting of statements the agent chooses to share with the world. Statements either about themselves (acting as public profile used by various apps), or about anything else.

Finally, AD4M agents declare a direct message Language, an AD4M Language they choose to be contacted with for receiving messages.

AD4M's built-in Agent-Language resolves DID URIs to AD4M Expressions that look like this:

    did: "did:key:zQ3shNWd4bg67ktTVg9EMnnrsRjhkH6cRNCjRRxfTaTqBniAf",
    perspective: {
        links: []
    directMessageLanguage: "lang://QmZ9Z9Z5yZsegxArToww5zmwtPpojXN6zXJsi7WwMUa8"
(see API docs about Agent)


…encapsulate the actual technology used to communicate, like Holochain or IPFS and enable Agents to create and share Expressions.

Expressions are referenced via a URI of the kind:
<language>://<language specific expression address>
(with special cases like DID URIs being parsed as such and resolved through the Agent Language).

AD4M resolves these URIs by first looking up the Language via its hash (and potentially downloading the Language through the built-in Language of Languages) and then asking the Language about the Expression with given address.

Languages are distributed and interpreted as JavaScript modules. AD4M passes in proxy-object to the managed Holochain, IPFS, etc. instances so Language developers can use these technologies without having to set them up or manage themselves.

// Example of a Language that uses the Holochain proxy object                        
export default async function create(context: LanguageContext): Promise {
    const Holochain = context.Holochain as HolochainLanguageDelegate;
    await Holochain.registerDNAs([{ file: DNA, nick: DNA_NICK }]);

    // ...

    async get(expressionAddress: Address): Promise {
        const expression = await this.#DNA.call(
(Read section in docs about how to write AD4M Languages)


…are local and private graph databases. They represent context and association between expressions.

They consist of a list of RDF/semantic web like triplets (subject-predicate-obejct) called links because all three items are just Expression URIs pointing to Expressions of arbitrary Languages.

Perspectives are like Solid’s pods, but they are agent-centric:
  • Atomic Perspectives belong to and are stored with a single Agent.
  • Links inside Perspectives are Link Expressions, so they include their provenance and cryptographic signature
While Expressions are objective (every agent resolving their URI renders the same data), Perspectives represent subjective associations between objective Expressions.

(See Gettint Started section above for how to deal with Perspectives)