2020-09-28 17:01:49 +03:00
|
|
|
|
/*
|
|
|
|
|
* Copyright 2020 Fluence Labs Limited
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
import { v4 as uuidv4 } from 'uuid';
|
2021-01-19 15:47:49 +03:00
|
|
|
|
import { fromByteArray, toByteArray } from 'base64-js';
|
2020-12-23 17:24:22 +03:00
|
|
|
|
import PeerId from 'peer-id';
|
|
|
|
|
import { encode } from 'bs58';
|
2021-01-19 15:47:49 +03:00
|
|
|
|
import { injectDataIntoParticle } from './ParticleProcessor';
|
2021-02-25 18:36:10 +03:00
|
|
|
|
import { PeerIdB58 } from './commonTypes';
|
2020-09-28 17:01:49 +03:00
|
|
|
|
|
|
|
|
|
const DEFAULT_TTL = 7000;
|
|
|
|
|
|
2021-01-29 16:48:27 +03:00
|
|
|
|
/**
|
|
|
|
|
* The class representing Particle - a data structure used to perform operations on Fluence Network. It originates on some peer in the network, travels the network through a predefined path, triggering function execution along its way.
|
|
|
|
|
*/
|
2021-01-19 15:47:49 +03:00
|
|
|
|
export class Particle {
|
|
|
|
|
script: string;
|
|
|
|
|
data: Map<string, any>;
|
|
|
|
|
ttl: number;
|
|
|
|
|
|
2021-01-29 16:48:27 +03:00
|
|
|
|
/**
|
|
|
|
|
* Creates a particle with specified parameters.
|
|
|
|
|
* @param { String }script - Air script which defines the execution of a particle – its path, functions it triggers on peers, and so on.
|
|
|
|
|
* @param { Map<string, any> | Record<string, any> } data - Variables passed to the particle in the form of either JS Map or JS object with keys representing variable names and values representing values correspondingly
|
|
|
|
|
* @param { [Number]=7000 } ttl - Time to live, a timout after which the particle execution is stopped by Aquamarine.
|
|
|
|
|
*/
|
2021-01-19 15:47:49 +03:00
|
|
|
|
constructor(script: string, data?: Map<string, any> | Record<string, any>, ttl?: number) {
|
|
|
|
|
this.script = script;
|
|
|
|
|
if (data === undefined) {
|
|
|
|
|
this.data = new Map();
|
|
|
|
|
} else if (data instanceof Map) {
|
|
|
|
|
this.data = data;
|
|
|
|
|
} else {
|
|
|
|
|
this.data = new Map();
|
|
|
|
|
for (let k in data) {
|
|
|
|
|
this.data.set(k, data[k]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.ttl = ttl ?? DEFAULT_TTL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface ParticleDto {
|
2020-12-23 17:24:22 +03:00
|
|
|
|
id: string;
|
|
|
|
|
init_peer_id: string;
|
|
|
|
|
timestamp: number;
|
|
|
|
|
ttl: number;
|
|
|
|
|
script: string;
|
2020-09-28 17:01:49 +03:00
|
|
|
|
// sign upper fields
|
2020-12-23 17:24:22 +03:00
|
|
|
|
signature: string;
|
2020-12-24 19:11:10 +03:00
|
|
|
|
data: Uint8Array;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Represents particle action to send to a node
|
|
|
|
|
*/
|
2021-01-19 15:47:49 +03:00
|
|
|
|
interface ParticlePayload {
|
|
|
|
|
action: 'Particle';
|
2020-12-24 19:11:10 +03:00
|
|
|
|
id: string;
|
|
|
|
|
init_peer_id: string;
|
|
|
|
|
timestamp: number;
|
|
|
|
|
ttl: number;
|
|
|
|
|
script: string;
|
|
|
|
|
signature: number[];
|
|
|
|
|
data: string;
|
2020-09-28 17:01:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-19 15:47:49 +03:00
|
|
|
|
function wrapWithVariableInjectionScript(script: string, fields: string[]): string {
|
2020-11-04 00:03:22 +03:00
|
|
|
|
fields.forEach((v) => {
|
2020-12-23 17:24:22 +03:00
|
|
|
|
script = `
|
2020-11-04 00:03:22 +03:00
|
|
|
|
(seq
|
2021-01-19 15:47:49 +03:00
|
|
|
|
(call %init_peer_id% ("__magic" "load") ["${v}"] ${v})
|
2020-11-04 00:03:22 +03:00
|
|
|
|
${script}
|
|
|
|
|
)
|
2020-12-23 17:24:22 +03:00
|
|
|
|
`;
|
2020-11-04 00:03:22 +03:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return script;
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-25 18:36:10 +03:00
|
|
|
|
function wrapWithXor(script: string): string {
|
|
|
|
|
return `
|
|
|
|
|
(xor
|
|
|
|
|
${script}
|
|
|
|
|
(seq
|
|
|
|
|
(call __magic_relay ("op" "identity") [])
|
|
|
|
|
(call %init_peer_id% ("__magic" "handle_xor") [%last_error%])
|
|
|
|
|
)
|
|
|
|
|
)`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function wrapWithXorLocal(script: string): string {
|
|
|
|
|
return `
|
|
|
|
|
(xor
|
|
|
|
|
${script}
|
|
|
|
|
(call %init_peer_id% ("__magic" "handle_xor") [%last_error%])
|
|
|
|
|
)`;
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-19 15:47:49 +03:00
|
|
|
|
export async function build(
|
|
|
|
|
peerId: PeerId,
|
2021-02-25 18:36:10 +03:00
|
|
|
|
relay: PeerIdB58 | undefined,
|
2021-01-19 15:47:49 +03:00
|
|
|
|
script: string,
|
|
|
|
|
data?: Map<string, any>,
|
|
|
|
|
ttl?: number,
|
|
|
|
|
customId?: string,
|
|
|
|
|
): Promise<ParticleDto> {
|
|
|
|
|
const id = customId ?? genUUID();
|
2020-12-23 17:24:22 +03:00
|
|
|
|
let currentTime = new Date().getTime();
|
2020-09-28 17:01:49 +03:00
|
|
|
|
|
2021-01-19 15:47:49 +03:00
|
|
|
|
if (data === undefined) {
|
|
|
|
|
data = new Map();
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-12 14:07:28 +03:00
|
|
|
|
if (ttl === undefined) {
|
2020-12-23 17:24:22 +03:00
|
|
|
|
ttl = DEFAULT_TTL;
|
2020-10-05 17:17:04 +03:00
|
|
|
|
}
|
2020-09-28 17:01:49 +03:00
|
|
|
|
|
2021-02-25 18:36:10 +03:00
|
|
|
|
if (relay) {
|
|
|
|
|
data.set('__magic_relay', relay);
|
|
|
|
|
}
|
2021-01-19 15:47:49 +03:00
|
|
|
|
injectDataIntoParticle(id, data, ttl);
|
|
|
|
|
script = wrapWithVariableInjectionScript(script, Array.from(data.keys()));
|
2021-02-25 18:36:10 +03:00
|
|
|
|
if (relay) {
|
|
|
|
|
script = wrapWithXor(script);
|
|
|
|
|
} else {
|
|
|
|
|
script = wrapWithXorLocal(script);
|
|
|
|
|
}
|
2020-11-04 00:03:22 +03:00
|
|
|
|
|
2021-01-19 15:47:49 +03:00
|
|
|
|
let particle: ParticleDto = {
|
2020-09-28 17:01:49 +03:00
|
|
|
|
id: id,
|
|
|
|
|
init_peer_id: peerId.toB58String(),
|
|
|
|
|
timestamp: currentTime,
|
|
|
|
|
ttl: ttl,
|
|
|
|
|
script: script,
|
2021-03-03 17:27:00 +03:00
|
|
|
|
// TODO: sign particle
|
2020-12-23 17:24:22 +03:00
|
|
|
|
signature: '',
|
2020-12-24 19:11:10 +03:00
|
|
|
|
data: Buffer.from([]),
|
2020-12-23 17:24:22 +03:00
|
|
|
|
};
|
2020-09-28 17:01:49 +03:00
|
|
|
|
|
|
|
|
|
return particle;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2020-12-24 19:11:10 +03:00
|
|
|
|
* Creates an action to send to a node.
|
2020-09-28 17:01:49 +03:00
|
|
|
|
*/
|
2021-01-19 15:47:49 +03:00
|
|
|
|
export function toPayload(particle: ParticleDto): ParticlePayload {
|
2020-12-24 19:11:10 +03:00
|
|
|
|
return {
|
2021-01-19 15:47:49 +03:00
|
|
|
|
action: 'Particle',
|
2020-12-24 19:11:10 +03:00
|
|
|
|
id: particle.id,
|
|
|
|
|
init_peer_id: particle.init_peer_id,
|
|
|
|
|
timestamp: particle.timestamp,
|
|
|
|
|
ttl: particle.ttl,
|
|
|
|
|
script: particle.script,
|
|
|
|
|
// TODO: copy signature from a particle after signatures will be implemented on nodes
|
|
|
|
|
signature: [],
|
2021-01-19 15:47:49 +03:00
|
|
|
|
data: fromByteArray(particle.data),
|
2020-12-24 19:11:10 +03:00
|
|
|
|
};
|
2020-09-28 17:01:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-19 15:47:49 +03:00
|
|
|
|
export function parseParticle(str: string): ParticleDto {
|
2020-09-28 17:01:49 +03:00
|
|
|
|
let json = JSON.parse(str);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
id: json.id,
|
|
|
|
|
init_peer_id: json.init_peer_id,
|
|
|
|
|
timestamp: json.timestamp,
|
|
|
|
|
ttl: json.ttl,
|
|
|
|
|
script: json.script,
|
|
|
|
|
signature: json.signature,
|
2020-12-24 19:36:38 +03:00
|
|
|
|
data: toByteArray(json.data),
|
2020-12-23 17:24:22 +03:00
|
|
|
|
};
|
2020-09-28 17:01:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function genUUID() {
|
|
|
|
|
return uuidv4();
|
|
|
|
|
}
|