mirror of
https://github.com/filecoin-project/slate.git
synced 2024-12-21 08:01:33 +03:00
555 lines
16 KiB
JavaScript
555 lines
16 KiB
JavaScript
// TODO(jim): The claim is that we can remove this
|
|
// and the package.json depdencies at some later time.
|
|
import { grpc } from "@improbable-eng/grpc-web";
|
|
import { WebsocketTransport } from "@textile/grpc-transport";
|
|
grpc.setDefaultTransport(WebsocketTransport());
|
|
|
|
import * as Environment from "~/node_common/environment";
|
|
import * as Utilities from "~/node_common/utilities";
|
|
import * as Data from "~/node_common/data";
|
|
import * as Constants from "~/node_common/constants";
|
|
import * as Serializers from "~/node_common/serializers";
|
|
import * as Social from "~/node_common/social";
|
|
import * as Strings from "~/common/strings";
|
|
import * as Websocket from "~/node_common/nodejs-websocket";
|
|
|
|
const STAGING_DEAL_BUCKET = "stage-deal";
|
|
|
|
const delay = async (waitMs) => {
|
|
return await new Promise((resolve) => setTimeout(resolve, waitMs));
|
|
};
|
|
|
|
Websocket.create();
|
|
|
|
//NOTE(martina): type = "UPDATE" will be processed by slate and lens. type = "SEARCH" will only be processed by lens
|
|
const websocketSend = async (type, data) => {
|
|
if (Strings.isEmpty(Environment.PUBSUB_SECRET)) {
|
|
return;
|
|
}
|
|
|
|
const ws = Websocket.get();
|
|
const encryptedData = await Utilities.encryptWithSecret(
|
|
JSON.stringify(data),
|
|
Environment.PUBSUB_SECRET
|
|
);
|
|
|
|
// NOTE(jim): Only allow this to be passed around encrypted.
|
|
ws.send(
|
|
JSON.stringify({
|
|
type,
|
|
iv: encryptedData.iv,
|
|
data: encryptedData.hex,
|
|
})
|
|
);
|
|
};
|
|
|
|
export const hydratePartialViewer = async (user) => {
|
|
const data = {
|
|
id: user.id,
|
|
username: user.username,
|
|
data: {
|
|
name: user.data.name ? user.data.name : "",
|
|
photo: user.data.photo ? user.data.photo : "",
|
|
body: user.data.body ? user.data.body : "",
|
|
},
|
|
type: "PARTIAL_VIEWER",
|
|
library: user.data.library,
|
|
onboarding: user.data.onboarding || {},
|
|
|
|
// TODO(jim): Move this elsewhere.
|
|
allow_filecoin_directory_listing: user.data.allow_filecoin_directory_listing
|
|
? user.data.allow_filecoin_directory_listing
|
|
: null,
|
|
allow_automatic_data_storage: user.data.allow_automatic_data_storage
|
|
? user.data.allow_automatic_data_storage
|
|
: null,
|
|
allow_encrypted_data_storage: user.data.allow_encrypted_data_storage
|
|
? user.data.allow_encrypted_data_storage
|
|
: null,
|
|
};
|
|
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
export const hydratePartialOnboarding = async (onboarding, userId) => {
|
|
const data = {
|
|
id: userId,
|
|
onboarding,
|
|
};
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
export const hydratePartialSubscriptions = async (updated, userId) => {
|
|
const data = {
|
|
id: userId,
|
|
};
|
|
const user = await Data.getUserById({ id: userId });
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
if (user.error) {
|
|
return null;
|
|
}
|
|
let mostRecent;
|
|
if (updated.subscriptions) {
|
|
const subscriptions = await Data.getSubscriptionsByUserId({ userId });
|
|
let r1 = await Serializers.doSubscriptions({
|
|
users: [],
|
|
slates: [],
|
|
subscriptions,
|
|
serializedUsersMap: { [user.id]: Serializers.user(user) },
|
|
serializedSlatesMap: {},
|
|
});
|
|
data.subscriptions = r1.serializedSubscriptions;
|
|
mostRecent = r1;
|
|
}
|
|
if (updated.subscribers) {
|
|
const subscribers = await Data.getSubscribersByUserId({ userId });
|
|
let r2 = await Serializers.doSubscribers({
|
|
users: [],
|
|
slates: [],
|
|
subscribers,
|
|
serializedUsersMap: mostRecent
|
|
? mostRecent.serializedUsersMap
|
|
: { [user.id]: Serializers.user(user) },
|
|
serializedSlatesMap: mostRecent ? mostRecent.serializedSlatesMap : {},
|
|
});
|
|
data.subscribers = r2.serializedSubscribers;
|
|
mostRecent = r2;
|
|
}
|
|
if (updated.trusted) {
|
|
console.log("update trusted");
|
|
const trusted = await Data.getTrustedRelationshipsByUserId({ userId });
|
|
let r3 = await Serializers.doTrusted({
|
|
users: [],
|
|
trusted,
|
|
serializedUsersMap: mostRecent
|
|
? mostRecent.serializedUsersMap
|
|
: { [user.id]: Serializers.user(user) },
|
|
serializedSlatesMap: mostRecent ? mostRecent.serializedSlatesMap : {},
|
|
});
|
|
data.trusted = r3.serializedTrusted;
|
|
mostRecent = r3;
|
|
}
|
|
if (updated.pendingTrusted) {
|
|
console.log("update pending trusted");
|
|
const pendingTrusted = await Data.getPendingTrustedRelationshipsByUserId({
|
|
userId,
|
|
});
|
|
let r4 = await Serializers.doPendingTrusted({
|
|
users: [userId],
|
|
pendingTrusted,
|
|
serializedUsersMap: mostRecent
|
|
? mostRecent.serializedUsersMap
|
|
: { [user.id]: Serializers.user(user) },
|
|
serializedSlatesMap: mostRecent ? mostRecent.serializedSlatesMap : {},
|
|
});
|
|
data.pendingTrusted = r4.serializedPendingTrusted;
|
|
}
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
export const hydratePartialKeys = async (keys, userId) => {
|
|
const data = {
|
|
id: userId,
|
|
keys,
|
|
};
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
export const hydratePartialLibrary = async (library, userId) => {
|
|
const data = {
|
|
id: userId,
|
|
library,
|
|
};
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
export const hydratePartialSlates = async (slates, userId) => {
|
|
console.log("hydrate slates");
|
|
const data = {
|
|
id: userId,
|
|
slates,
|
|
};
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
// export const hydratePartialActivity = async (activity, userId) => {
|
|
// this one will need to be more complex like what is in hydrate
|
|
// websocketSend("UPDATE", data);
|
|
// };
|
|
|
|
export const hydrate = async (id) => {
|
|
const user = await Data.getUserById({
|
|
id,
|
|
});
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
if (user.error) {
|
|
return null;
|
|
}
|
|
|
|
// TODO(jim): You can serialize this last because you will have all the information
|
|
// from subscriptions, trusted, and pendingTrusted most likely.
|
|
const activity = await Data.getActivityForUserId({ userId: id });
|
|
const slates = await Data.getSlatesByUserId({ userId: id });
|
|
const keys = await Data.getAPIKeysByUserId({ userId: id });
|
|
const subscriptions = await Data.getSubscriptionsByUserId({ userId: id });
|
|
const subscribers = await Data.getSubscribersByUserId({ userId: id });
|
|
|
|
let serializedUsersMap = { [user.id]: Serializers.user(user) };
|
|
let serializedSlatesMap = {};
|
|
|
|
// NOTE(jim): The most expensive call first.
|
|
const r1 = await Serializers.doSubscriptions({
|
|
users: [],
|
|
slates: [],
|
|
subscriptions,
|
|
serializedUsersMap,
|
|
serializedSlatesMap,
|
|
});
|
|
|
|
const r2 = await Serializers.doSubscribers({
|
|
users: [],
|
|
slates: [],
|
|
subscribers,
|
|
serializedUsersMap: r1.serializedUsersMap,
|
|
serializedSlatesMap: r1.serializedSlatesMap,
|
|
});
|
|
|
|
// NOTE(jim): If any trusted users are subscription users, this ends up being cheaper.
|
|
const trusted = await Data.getTrustedRelationshipsByUserId({ userId: id });
|
|
const r3 = await Serializers.doTrusted({
|
|
users: [],
|
|
trusted,
|
|
serializedUsersMap: r2.serializedUsersMap,
|
|
serializedSlatesMap: r2.serializedSlatesMap,
|
|
});
|
|
|
|
// NOTE(jim): This should be the cheapest call.
|
|
const pendingTrusted = await Data.getPendingTrustedRelationshipsByUserId({
|
|
userId: id,
|
|
});
|
|
const r4 = await Serializers.doPendingTrusted({
|
|
users: [id],
|
|
pendingTrusted,
|
|
serializedUsersMap: r3.serializedUsersMap,
|
|
serializedSlatesMap: r3.serializedSlatesMap,
|
|
});
|
|
|
|
let bytes = 0;
|
|
let imageBytes = 0;
|
|
let videoBytes = 0;
|
|
let audioBytes = 0;
|
|
let epubBytes = 0;
|
|
let pdfBytes = 0;
|
|
user.data.library[0].children.forEach((each) => {
|
|
if (each.type && each.type.startsWith("image/")) {
|
|
imageBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("video/")) {
|
|
videoBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("audio/")) {
|
|
audioBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("application/epub")) {
|
|
epubBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("application/pdf")) {
|
|
pdfBytes += each.size;
|
|
}
|
|
bytes += each.size;
|
|
});
|
|
|
|
let data = {
|
|
...Serializers.user(user),
|
|
type: "VIEWER",
|
|
library: user.data.library,
|
|
onboarding: user.data.onboarding || {},
|
|
|
|
// TODO(jim): Move this elsewhere.
|
|
allow_filecoin_directory_listing: user.data.allow_filecoin_directory_listing
|
|
? user.data.allow_filecoin_directory_listing
|
|
: null,
|
|
allow_automatic_data_storage: user.data.allow_automatic_data_storage
|
|
? user.data.allow_automatic_data_storage
|
|
: null,
|
|
allow_encrypted_data_storage: user.data.allow_encrypted_data_storage
|
|
? user.data.allow_encrypted_data_storage
|
|
: null,
|
|
|
|
// NOTE(jim): Remaining data.
|
|
stats: {
|
|
bytes,
|
|
maximumBytes: Constants.TEXTILE_ACCOUNT_BYTE_LIMIT,
|
|
imageBytes,
|
|
videoBytes,
|
|
audioBytes,
|
|
epubBytes,
|
|
pdfBytes,
|
|
},
|
|
keys,
|
|
activity,
|
|
slates,
|
|
subscriptions: r1.serializedSubscriptions,
|
|
subscribers: r2.serializedSubscribers,
|
|
trusted: r3.serializedTrusted,
|
|
pendingTrusted: r4.serializedPendingTrusted,
|
|
};
|
|
websocketSend("UPDATE", data);
|
|
};
|
|
|
|
// TODO(jim): Work on better serialization when adoption starts occuring.
|
|
export const getById = async ({ id }) => {
|
|
const user = await Data.getUserById({
|
|
id,
|
|
});
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
if (user.error) {
|
|
return null;
|
|
}
|
|
|
|
// TODO(jim): You can serialize this last because you will have all the information
|
|
// from subscriptionsed, trusted, and pendingTrusted most likely.
|
|
const activity = await Data.getActivityForUserId({ userId: id });
|
|
const slates = await Data.getSlatesByUserId({ userId: id });
|
|
const keys = await Data.getAPIKeysByUserId({ userId: id });
|
|
const subscriptions = await Data.getSubscriptionsByUserId({ userId: id });
|
|
const subscribers = await Data.getSubscribersByUserId({ userId: id });
|
|
|
|
let serializedUsersMap = { [user.id]: Serializers.user(user) };
|
|
let serializedSlatesMap = {};
|
|
|
|
// NOTE(jim): The most expensive call first.
|
|
const r1 = await Serializers.doSubscriptions({
|
|
users: [],
|
|
slates: [],
|
|
subscriptions,
|
|
serializedUsersMap,
|
|
serializedSlatesMap,
|
|
});
|
|
|
|
const r2 = await Serializers.doSubscribers({
|
|
users: [],
|
|
slates: [],
|
|
subscribers,
|
|
serializedUsersMap: r1.serializedUsersMap,
|
|
serializedSlatesMap: r1.serializedSlatesMap,
|
|
});
|
|
|
|
// NOTE(jim): If any trusted users are subscription users, this ends up being cheaper.
|
|
const trusted = await Data.getTrustedRelationshipsByUserId({ userId: id });
|
|
const r3 = await Serializers.doTrusted({
|
|
users: [],
|
|
trusted,
|
|
serializedUsersMap: r2.serializedUsersMap,
|
|
serializedSlatesMap: r2.serializedSlatesMap,
|
|
});
|
|
|
|
// NOTE(jim): This should be the cheapest call.
|
|
const pendingTrusted = await Data.getPendingTrustedRelationshipsByUserId({
|
|
userId: id,
|
|
});
|
|
const r4 = await Serializers.doPendingTrusted({
|
|
users: [id],
|
|
pendingTrusted,
|
|
serializedUsersMap: r3.serializedUsersMap,
|
|
serializedSlatesMap: r3.serializedSlatesMap,
|
|
});
|
|
|
|
let bytes = 0;
|
|
let imageBytes = 0;
|
|
let videoBytes = 0;
|
|
let audioBytes = 0;
|
|
let epubBytes = 0;
|
|
let pdfBytes = 0;
|
|
user.data.library[0].children.forEach((each) => {
|
|
if (each.type && each.type.startsWith("image/")) {
|
|
imageBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("video/")) {
|
|
videoBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("audio/")) {
|
|
audioBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("application/epub")) {
|
|
epubBytes += each.size;
|
|
} else if (each.type && each.type.startsWith("application/pdf")) {
|
|
pdfBytes += each.size;
|
|
}
|
|
bytes += each.size;
|
|
});
|
|
|
|
return {
|
|
...Serializers.user(user),
|
|
type: "VIEWER",
|
|
library: user.data.library,
|
|
onboarding: user.data.onboarding || {},
|
|
|
|
// TODO(jim): Move this elsewhere.
|
|
allow_filecoin_directory_listing: user.data.allow_filecoin_directory_listing
|
|
? user.data.allow_filecoin_directory_listing
|
|
: null,
|
|
allow_automatic_data_storage: user.data.allow_automatic_data_storage
|
|
? user.data.allow_automatic_data_storage
|
|
: null,
|
|
allow_encrypted_data_storage: user.data.allow_encrypted_data_storage
|
|
? user.data.allow_encrypted_data_storage
|
|
: null,
|
|
|
|
// NOTE(jim): Remaining data.
|
|
stats: {
|
|
bytes,
|
|
maximumBytes: Constants.TEXTILE_ACCOUNT_BYTE_LIMIT,
|
|
imageBytes,
|
|
videoBytes,
|
|
audioBytes,
|
|
epubBytes,
|
|
pdfBytes,
|
|
},
|
|
keys,
|
|
activity,
|
|
slates,
|
|
subscriptions: r1.serializedSubscriptions,
|
|
subscribers: r2.serializedSubscribers,
|
|
trusted: r3.serializedTrusted,
|
|
pendingTrusted: r4.serializedPendingTrusted,
|
|
};
|
|
};
|
|
|
|
export const getDealHistory = async ({ id }) => {
|
|
const user = await Data.getUserById({
|
|
id,
|
|
});
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
if (user.error) {
|
|
return null;
|
|
}
|
|
|
|
let deals = [];
|
|
|
|
try {
|
|
const PowergateSingleton = await Utilities.getPowergateAPIFromUserToken({
|
|
user,
|
|
});
|
|
const { power } = PowergateSingleton;
|
|
|
|
const result = await power.listStorageDealRecords({
|
|
ascending: false,
|
|
includePending: true,
|
|
includeFinal: true,
|
|
});
|
|
|
|
for (let i = 0; i < result.recordsList.length; i++) {
|
|
const o = result.recordsList[i];
|
|
|
|
deals.push({
|
|
dealId: o.dealInfo.dealId,
|
|
rootCid: o.rootCid,
|
|
proposalCid: o.dealInfo.proposalCid,
|
|
pieceCid: o.dealInfo.pieceCid,
|
|
addr: o.addr,
|
|
miner: o.dealInfo.miner,
|
|
size: o.dealInfo.size,
|
|
// NOTE(jim): formatted size.
|
|
formattedSize: Strings.bytesToSize(o.dealInfo.size),
|
|
pricePerEpoch: o.dealInfo.pricePerEpoch,
|
|
startEpoch: o.dealInfo.startEpoch,
|
|
// NOTE(jim): just for point of reference on the total cost.
|
|
totalSpeculatedCost: Strings.formatAsFilecoinConversion(
|
|
o.dealInfo.pricePerEpoch * o.dealInfo.duration
|
|
),
|
|
duration: o.dealInfo.duration,
|
|
formattedDuration: Strings.getDaysFromEpoch(o.dealInfo.duration),
|
|
activationEpoch: o.dealInfo.activationEpoch,
|
|
time: o.time,
|
|
createdAt: Strings.toDateSinceEpoch(o.time),
|
|
pending: o.pending,
|
|
user: Serializers.user(user),
|
|
});
|
|
}
|
|
} catch (e) {
|
|
console.log(e);
|
|
Social.sendTextileSlackMessage({
|
|
file: "/node_common/managers/viewer.js",
|
|
user,
|
|
message: e.message,
|
|
code: e.code,
|
|
functionName: `power.listStorageDealRecords`,
|
|
});
|
|
}
|
|
|
|
return { type: "VIEWER_FILECOIN_DEALS", deals };
|
|
};
|
|
|
|
export const getTextileById = async ({ id }) => {
|
|
const user = await Data.getUserById({
|
|
id,
|
|
});
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
if (user.error) {
|
|
return null;
|
|
}
|
|
|
|
let dealJobs = [];
|
|
|
|
const { power, powerInfo, powerHealth } = await Utilities.getPowergateAPIFromUserToken({ user });
|
|
|
|
// NOTE(jim): This bucket is purely for staging data for other deals.
|
|
const stagingData = await Utilities.getBucketAPIFromUserToken({
|
|
user,
|
|
bucketName: STAGING_DEAL_BUCKET,
|
|
encrypted: false,
|
|
});
|
|
|
|
let r = null;
|
|
try {
|
|
r = await stagingData.buckets.list();
|
|
} catch (e) {
|
|
Social.sendTextileSlackMessage({
|
|
file: "/node_common/managers/viewer.js",
|
|
user,
|
|
message: e.message,
|
|
code: e.code,
|
|
functionName: `buckets.list`,
|
|
});
|
|
}
|
|
|
|
let items = null;
|
|
const dealBucket = r.find((bucket) => bucket.name === STAGING_DEAL_BUCKET);
|
|
try {
|
|
const path = await stagingData.buckets.listPath(dealBucket.key, "/");
|
|
items = path.item.items;
|
|
} catch (e) {
|
|
Social.sendTextileSlackMessage({
|
|
file: "/node_common/managers/viewer.js",
|
|
user,
|
|
message: e.message,
|
|
code: e.code,
|
|
functionName: `buckets.listPath`,
|
|
});
|
|
}
|
|
|
|
return {
|
|
type: "VIEWER_FILECOIN",
|
|
settings: {
|
|
deals_auto_approve: user.data.settings_deals_auto_approve,
|
|
},
|
|
powerInfo,
|
|
powerHealth,
|
|
deal: items ? items.filter((f) => f.name !== ".textileseed") : [],
|
|
};
|
|
};
|