mirror of
https://github.com/Mintplex-Labs/anything-llm.git
synced 2024-11-15 19:00:33 +01:00
dc4ad6b5a9
* wip bg workers for live document sync * Add ability to re-embed specific documents across many workspaces via background queue bgworkser is gated behind expieremental system setting flag that needs to be explictly enabled UI for watching/unwatching docments that are embedded. TODO: UI to easily manage all bg tasks and see run results TODO: UI to enable this feature and background endpoints to manage it * create frontend views and paths Move elements to correct experimental scope * update migration to delete runs on removal of watched document * Add watch support to YouTube transcripts (#1716) * Add watch support to YouTube transcripts refactor how sync is done for supported types * Watch specific files in Confluence space (#1718) Add failure-prune check for runs * create tmp workflow modifications for beta image * create tmp workflow modifications for beta image * create tmp workflow modifications for beta image * dual build update copy of alert modals * update job interval * Add support for live-sync of Github files * update copy for document sync feature * hide Experimental features from UI * update docs links * [FEAT] Implement new settings menu for experimental features (#1735) * implement new settings menu for experimental features * remove unused context save bar --------- Co-authored-by: timothycarambat <rambat1010@gmail.com> * dont run job on boot * unset workflow changes * Add persistent encryption service Relay key to collector so persistent encryption can be used Encrypt any private data in chunkSources used for replay during resync jobs * update jsDOC * Linting and organization * update modal copy for feature --------- Co-authored-by: Sean Hatfield <seanhatfield5@gmail.com>
179 lines
5.2 KiB
JavaScript
179 lines
5.2 KiB
JavaScript
const { EncryptionManager } = require("../EncryptionManager");
|
|
|
|
// When running locally will occupy the 0.0.0.0 hostname space but when deployed inside
|
|
// of docker this endpoint is not exposed so it is only on the Docker instances internal network
|
|
// so no additional security is needed on the endpoint directly. Auth is done however by the express
|
|
// middleware prior to leaving the node-side of the application so that is good enough >:)
|
|
class CollectorApi {
|
|
constructor() {
|
|
const { CommunicationKey } = require("../comKey");
|
|
this.comkey = new CommunicationKey();
|
|
this.endpoint = `http://0.0.0.0:${process.env.COLLECTOR_PORT || 8888}`;
|
|
}
|
|
|
|
log(text, ...args) {
|
|
console.log(`\x1b[36m[CollectorApi]\x1b[0m ${text}`, ...args);
|
|
}
|
|
|
|
#attachOptions() {
|
|
return {
|
|
whisperProvider: process.env.WHISPER_PROVIDER || "local",
|
|
WhisperModelPref: process.env.WHISPER_MODEL_PREF,
|
|
openAiKey: process.env.OPEN_AI_KEY || null,
|
|
};
|
|
}
|
|
|
|
async online() {
|
|
return await fetch(this.endpoint)
|
|
.then((res) => res.ok)
|
|
.catch(() => false);
|
|
}
|
|
|
|
async acceptedFileTypes() {
|
|
return await fetch(`${this.endpoint}/accepts`)
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("failed to GET /accepts");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return null;
|
|
});
|
|
}
|
|
|
|
async processDocument(filename = "") {
|
|
if (!filename) return false;
|
|
|
|
const data = JSON.stringify({
|
|
filename,
|
|
options: this.#attachOptions(),
|
|
});
|
|
|
|
return await fetch(`${this.endpoint}/process`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Integrity": this.comkey.sign(data),
|
|
"X-Payload-Signer": this.comkey.encrypt(
|
|
new EncryptionManager().xPayload
|
|
),
|
|
},
|
|
body: data,
|
|
})
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("Response could not be completed");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return { success: false, reason: e.message, documents: [] };
|
|
});
|
|
}
|
|
|
|
async processLink(link = "") {
|
|
if (!link) return false;
|
|
|
|
const data = JSON.stringify({ link });
|
|
return await fetch(`${this.endpoint}/process-link`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Integrity": this.comkey.sign(data),
|
|
"X-Payload-Signer": this.comkey.encrypt(
|
|
new EncryptionManager().xPayload
|
|
),
|
|
},
|
|
body: data,
|
|
})
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("Response could not be completed");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return { success: false, reason: e.message, documents: [] };
|
|
});
|
|
}
|
|
|
|
async processRawText(textContent = "", metadata = {}) {
|
|
const data = JSON.stringify({ textContent, metadata });
|
|
return await fetch(`${this.endpoint}/process-raw-text`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Integrity": this.comkey.sign(data),
|
|
"X-Payload-Signer": this.comkey.encrypt(
|
|
new EncryptionManager().xPayload
|
|
),
|
|
},
|
|
body: data,
|
|
})
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("Response could not be completed");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return { success: false, reason: e.message, documents: [] };
|
|
});
|
|
}
|
|
|
|
// We will not ever expose the document processor to the frontend API so instead we relay
|
|
// all requests through the server. You can use this function to directly expose a specific endpoint
|
|
// on the document processor.
|
|
async forwardExtensionRequest({ endpoint, method, body }) {
|
|
return await fetch(`${this.endpoint}${endpoint}`, {
|
|
method,
|
|
body, // Stringified JSON!
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Integrity": this.comkey.sign(body),
|
|
"X-Payload-Signer": this.comkey.encrypt(
|
|
new EncryptionManager().xPayload
|
|
),
|
|
},
|
|
})
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("Response could not be completed");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return { success: false, data: {}, reason: e.message };
|
|
});
|
|
}
|
|
|
|
async getLinkContent(link = "") {
|
|
if (!link) return false;
|
|
|
|
const data = JSON.stringify({ link });
|
|
return await fetch(`${this.endpoint}/util/get-link`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Integrity": this.comkey.sign(data),
|
|
"X-Payload-Signer": this.comkey.encrypt(
|
|
new EncryptionManager().xPayload
|
|
),
|
|
},
|
|
body: data,
|
|
})
|
|
.then((res) => {
|
|
if (!res.ok) throw new Error("Response could not be completed");
|
|
return res.json();
|
|
})
|
|
.then((res) => res)
|
|
.catch((e) => {
|
|
this.log(e.message);
|
|
return { success: false, content: null };
|
|
});
|
|
}
|
|
}
|
|
|
|
module.exports.CollectorApi = CollectorApi;
|