mirror of
https://github.com/urbit/shrub.git
synced 2025-01-01 17:16:47 +03:00
509 lines
10 KiB
TypeScript
509 lines
10 KiB
TypeScript
import _ from 'lodash';
|
|
import { GroupPolicy, makeResource, Resource, resourceFromPath } from '../groups';
|
|
|
|
import { decToUd, deSig, unixToDa, Scry } from '../lib';
|
|
import { Enc, Path, Patp, PatpNoSig, Poke, Thread } from '../lib/types';
|
|
import { Content, Graph, GraphChildrenPoke, GraphNode, GraphNodePoke, Post } from './types';
|
|
import { patp2dec } from 'urbit-ob';
|
|
|
|
export const GRAPH_UPDATE_VERSION: number = 2;
|
|
|
|
export const createBlankNodeWithChildPost = (
|
|
ship: PatpNoSig,
|
|
parentIndex: string = '',
|
|
childIndex: string = '',
|
|
contents: Content[]
|
|
): GraphNodePoke => {
|
|
const date = unixToDa(Date.now()).toString();
|
|
const nodeIndex = parentIndex + '/' + date;
|
|
|
|
const childGraph: GraphChildrenPoke = {};
|
|
childGraph[childIndex] = {
|
|
post: {
|
|
author: `~${ship}`,
|
|
index: nodeIndex + '/' + childIndex,
|
|
'time-sent': Date.now(),
|
|
contents,
|
|
hash: null,
|
|
signatures: []
|
|
},
|
|
children: null
|
|
};
|
|
|
|
return {
|
|
post: {
|
|
author: `~${ship}`,
|
|
index: nodeIndex,
|
|
'time-sent': Date.now(),
|
|
contents: [],
|
|
hash: null,
|
|
signatures: []
|
|
},
|
|
children: childGraph
|
|
};
|
|
};
|
|
|
|
export const markPending = (nodes: any): any => {
|
|
Object.keys(nodes).forEach((key) => {
|
|
nodes[key].post.author = deSig(nodes[key].post.author);
|
|
nodes[key].post.pending = true;
|
|
if (nodes[key].children) {
|
|
nodes[key].children = markPending(nodes[key].children);
|
|
}
|
|
});
|
|
return nodes;
|
|
};
|
|
|
|
export const createPost = (
|
|
ship: PatpNoSig,
|
|
contents: Content[],
|
|
parentIndex: string = '',
|
|
childIndex:string = 'DATE_PLACEHOLDER'
|
|
): Post => {
|
|
if (childIndex === 'DATE_PLACEHOLDER') {
|
|
childIndex = unixToDa(Date.now()).toString();
|
|
}
|
|
return {
|
|
author: `~${ship}`,
|
|
index: parentIndex + '/' + childIndex,
|
|
'time-sent': Date.now(),
|
|
contents,
|
|
hash: null,
|
|
signatures: []
|
|
};
|
|
};
|
|
|
|
function moduleToMark(mod: string): string | undefined {
|
|
if(mod === 'link') {
|
|
return 'graph-validator-link';
|
|
}
|
|
if(mod === 'publish') {
|
|
return 'graph-validator-publish';
|
|
}
|
|
if(mod === 'chat') {
|
|
return 'graph-validator-chat';
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
const storeAction = <T>(data: T, version: number = GRAPH_UPDATE_VERSION): Poke<T> => ({
|
|
app: 'graph-store',
|
|
mark: `graph-update-${version}`,
|
|
json: data
|
|
});
|
|
|
|
export { storeAction as graphStoreAction };
|
|
|
|
const viewAction = <T>(threadName: string, action: T): Thread<T> => ({
|
|
inputMark: 'graph-view-action',
|
|
outputMark: 'json',
|
|
threadName,
|
|
body: action
|
|
});
|
|
|
|
export { viewAction as graphViewAction };
|
|
|
|
const hookAction = <T>(data: T, version: number = GRAPH_UPDATE_VERSION): Poke<T> => ({
|
|
app: 'graph-push-hook',
|
|
mark: `graph-update-${version}`,
|
|
json: data
|
|
});
|
|
|
|
const dmAction = <T>(data: T): Poke<T> => ({
|
|
app: 'dm-hook',
|
|
mark: 'dm-hook-action',
|
|
json: data
|
|
});
|
|
|
|
export { hookAction as graphHookAction };
|
|
|
|
|
|
export const createManagedGraph = (
|
|
ship: PatpNoSig,
|
|
name: string,
|
|
title: string,
|
|
description: string,
|
|
group: Path,
|
|
mod: string
|
|
): Thread<any> => {
|
|
const associated = { group: resourceFromPath(group) };
|
|
const resource = makeResource(`~${ship}`, name);
|
|
|
|
return viewAction('graph-create', {
|
|
create: {
|
|
resource,
|
|
title,
|
|
description,
|
|
associated,
|
|
module: mod,
|
|
mark: moduleToMark(mod)
|
|
}
|
|
});
|
|
}
|
|
|
|
export const createUnmanagedGraph = (
|
|
ship: PatpNoSig,
|
|
name: string,
|
|
title: string,
|
|
description: string,
|
|
policy: Enc<GroupPolicy>,
|
|
mod: string
|
|
): Thread<any> => viewAction('graph-create', {
|
|
create: {
|
|
resource: makeResource(`~${ship}`, name),
|
|
title,
|
|
description,
|
|
associated: { policy },
|
|
module: mod,
|
|
mark: moduleToMark(mod)
|
|
}
|
|
});
|
|
|
|
export const joinGraph = (
|
|
ship: Patp,
|
|
name: string
|
|
): Thread<any> => viewAction('graph-join', {
|
|
join: {
|
|
resource: makeResource(ship, name),
|
|
ship,
|
|
}
|
|
});
|
|
|
|
export const deleteGraph = (
|
|
ship: PatpNoSig,
|
|
name: string
|
|
): Thread<any> => viewAction('graph-delete', {
|
|
"delete": {
|
|
resource: makeResource(`~${ship}`, name)
|
|
}
|
|
});
|
|
|
|
export const leaveGraph = (
|
|
ship: Patp,
|
|
name: string
|
|
): Thread<any> => viewAction('graph-leave', {
|
|
"leave": {
|
|
resource: makeResource(ship, name)
|
|
}
|
|
});
|
|
|
|
export const groupifyGraph = (
|
|
ship: Patp,
|
|
name: string,
|
|
toPath?: string
|
|
): Thread<any> => {
|
|
const resource = makeResource(ship, name);
|
|
const to = toPath && resourceFromPath(toPath);
|
|
|
|
return viewAction('graph-groupify', {
|
|
groupify: {
|
|
resource,
|
|
to
|
|
}
|
|
});
|
|
}
|
|
|
|
export const evalCord = (
|
|
cord: string
|
|
): Thread<any> => {
|
|
return ({
|
|
inputMark: 'graph-view-action',
|
|
outputMark: 'tang',
|
|
threadName: 'graph-eval',
|
|
body: {
|
|
eval: cord
|
|
}
|
|
});
|
|
}
|
|
|
|
export const addGraph = (
|
|
ship: Patp,
|
|
name: string,
|
|
graph: any,
|
|
mark: any
|
|
): Poke<any> => {
|
|
return storeAction({
|
|
'add-graph': {
|
|
resource: { ship, name },
|
|
graph,
|
|
mark
|
|
}
|
|
});
|
|
}
|
|
|
|
export const addNodes = (
|
|
ship: Patp,
|
|
name: string,
|
|
nodes: Object
|
|
): Thread<any> => ({
|
|
inputMark: `graph-update-${GRAPH_UPDATE_VERSION}`,
|
|
outputMark: 'graph-view-action',
|
|
threadName: 'graph-add-nodes',
|
|
body: {
|
|
'add-nodes': {
|
|
resource: { ship, name },
|
|
nodes
|
|
}
|
|
}
|
|
});
|
|
|
|
export const addPost = (
|
|
ship: Patp,
|
|
name: string,
|
|
post: Post
|
|
): Thread<any> => {
|
|
let nodes: Record<string, GraphNode> = {};
|
|
nodes[post.index] = {
|
|
post,
|
|
children: null
|
|
};
|
|
return addNodes(ship, name, nodes);
|
|
}
|
|
|
|
export const addNode = (
|
|
ship: Patp,
|
|
name: string,
|
|
node: GraphNodePoke
|
|
): Thread<any> => {
|
|
let nodes: Record<string, GraphNodePoke> = {};
|
|
nodes[node.post.index] = node;
|
|
|
|
return addNodes(ship, name, nodes);
|
|
}
|
|
|
|
export const createGroupFeed = (
|
|
group: Resource,
|
|
vip: any = ''
|
|
): Thread<any> => ({
|
|
inputMark: 'graph-view-action',
|
|
outputMark: 'resource',
|
|
threadName: 'graph-create-group-feed',
|
|
body: {
|
|
'create-group-feed': {
|
|
resource: group,
|
|
vip
|
|
}
|
|
}
|
|
});
|
|
|
|
export const disableGroupFeed = (
|
|
group: Resource
|
|
): Thread<any> => ({
|
|
inputMark: 'graph-view-action',
|
|
outputMark: 'json',
|
|
threadName: 'graph-disable-group-feed',
|
|
body: {
|
|
'disable-group-feed': {
|
|
resource: group
|
|
}
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Set dm-hook to screen new DMs or not
|
|
*
|
|
*/
|
|
export const setScreen = (screen: boolean): Poke<any> => dmAction({ screen });
|
|
|
|
/**
|
|
* Accept a pending DM request
|
|
*
|
|
* @param ship the ship to accept
|
|
*/
|
|
export const acceptDm = (ship: string) => dmAction({
|
|
accept: ship
|
|
});
|
|
|
|
/**
|
|
* Decline a pending DM request
|
|
*
|
|
* @param ship the ship to accept
|
|
*/
|
|
export const declineDm = (ship: string) => dmAction({
|
|
decline: ship
|
|
});
|
|
|
|
/**
|
|
* Remove posts from a set of indices
|
|
*
|
|
*/
|
|
export const removePosts = (
|
|
ship: Patp,
|
|
name: string,
|
|
indices: string[]
|
|
): Poke<any> => hookAction({
|
|
'remove-nodes': {
|
|
resource: { ship, name },
|
|
indices
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Send a DM to a ship
|
|
*
|
|
* @param our sender
|
|
* @param ship recipient
|
|
* @param contents contents of message
|
|
*/
|
|
export const addDmMessage = (our: PatpNoSig, ship: Patp, contents: Content[]): Poke<any> => {
|
|
const post = createPost(our, contents, `/${patp2dec(ship)}`);
|
|
const node: GraphNode = {
|
|
post,
|
|
children: null
|
|
};
|
|
return {
|
|
app: 'dm-hook',
|
|
mark: `graph-update-${GRAPH_UPDATE_VERSION}`,
|
|
json: {
|
|
'add-nodes': {
|
|
resource: { ship: `~${our}`, name: 'dm-inbox' },
|
|
nodes: {
|
|
[post.index]: node
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
|
|
const encodeIndex = (idx: string) => idx.split('/').map(decToUd).join('/');
|
|
|
|
/**
|
|
* Fetch newest (larger keys) nodes in a graph under some index
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param count number of nodes to load
|
|
* @param index index to query
|
|
*/
|
|
export const getNewest = (
|
|
ship: string,
|
|
name: string,
|
|
count: number,
|
|
index = ''
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/newest/${ship}/${name}/${count}${encodeIndex(index)}`
|
|
});
|
|
|
|
/**
|
|
* Fetch nodes in a graph that are older (key is smaller) and direct
|
|
* siblings of some index
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param count number of nodes to load
|
|
* @param index index to query
|
|
*/
|
|
export const getOlderSiblings = (
|
|
ship: string,
|
|
name: string,
|
|
count: number,
|
|
index: string
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/node-siblings/older/${ship}/${name}/${count}${encodeIndex(index)}`
|
|
});
|
|
|
|
/**
|
|
* Fetch nodes in a graph that are younger (key is larger) and direct
|
|
* siblings of some index
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param count number of nodes to load
|
|
* @param index index to query
|
|
*/
|
|
export const getYoungerSiblings = (
|
|
ship: string,
|
|
name: string,
|
|
count: number,
|
|
index: string
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/node-siblings/younger/${ship}/${name}/${count}${encodeIndex(index)}`
|
|
});
|
|
|
|
/**
|
|
* Fetch all nodes in a graph under some index, without loading children
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param index index to query
|
|
*/
|
|
export const getShallowChildren = (ship: string, name: string, index = '') => ({
|
|
app: 'graph-store',
|
|
path: `/shallow-children/${ship}/${name}${encodeIndex(index)}`
|
|
});
|
|
|
|
|
|
/**
|
|
* Fetch newest nodes in a graph as a flat map, including children,
|
|
* optionally starting at a specified key
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param count number of nodes to load
|
|
* @param start key to start at
|
|
*
|
|
*/
|
|
export const getDeepOlderThan = (
|
|
ship: string,
|
|
name: string,
|
|
count: number,
|
|
start?: string
|
|
) => ({
|
|
app: 'graph-store',
|
|
path: `/deep-nodes-older-than/${ship}/${name}/${count}/${start ? decToUd(start) : 'null'}`
|
|
});
|
|
|
|
|
|
/**
|
|
* Fetch a flat map of a nodes ancestors and firstborn children
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param index index to query
|
|
*
|
|
*/
|
|
export const getFirstborn = (
|
|
ship: string,
|
|
name: string,
|
|
index: string
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/firstborn/${ship}/${name}/${encodeIndex(index)}`
|
|
});
|
|
|
|
/**
|
|
* Fetch a single node, and all it's children
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
* @param index index to query
|
|
*
|
|
*/
|
|
export const getNode = (
|
|
ship: string,
|
|
name: string,
|
|
index: string
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/node/${ship}/${name}/${encodeIndex(index)}`
|
|
});
|
|
|
|
/**
|
|
* Fetch entire graph
|
|
*
|
|
* @param ship ship of graph
|
|
* @param name name of graph
|
|
*
|
|
*/
|
|
export const getGraph = (
|
|
ship: string,
|
|
name: string
|
|
): Scry => ({
|
|
app: 'graph-store',
|
|
path: `/graph/${ship}/${name}`
|
|
});
|