mirror of
https://github.com/Mintplex-Labs/anything-llm.git
synced 2024-11-19 12:40:09 +01:00
256 lines
6.2 KiB
JavaScript
256 lines
6.2 KiB
JavaScript
const fs = require("fs");
|
|
const path = require("path");
|
|
const { default: slugify } = require("slugify");
|
|
const { v4 } = require("uuid");
|
|
const { writeToServerDocuments, sanitizeFileName } = require("../../files");
|
|
const { tokenizeString } = require("../../tokenizer");
|
|
const { ConfluencePagesLoader } = require("./ConfluenceLoader");
|
|
|
|
/**
|
|
* Load Confluence documents from a spaceID and Confluence credentials
|
|
* @param {object} args - forwarded request body params
|
|
* @param {import("../../../middleware/setDataSigner").ResponseWithSigner} response - Express response object with encryptionWorker
|
|
* @returns
|
|
*/
|
|
async function loadConfluence(
|
|
{
|
|
baseUrl = null,
|
|
spaceKey = null,
|
|
username = null,
|
|
accessToken = null,
|
|
cloud = true,
|
|
},
|
|
response
|
|
) {
|
|
if (!baseUrl || !spaceKey || !username || !accessToken) {
|
|
return {
|
|
success: false,
|
|
reason:
|
|
"You need either a username and access token, or a personal access token (PAT), to use the Confluence connector.",
|
|
};
|
|
}
|
|
|
|
if (!validBaseUrl(baseUrl)) {
|
|
return {
|
|
success: false,
|
|
reason: "Provided base URL is not a valid URL.",
|
|
};
|
|
}
|
|
|
|
if (!spaceKey) {
|
|
return {
|
|
success: false,
|
|
reason: "You need to provide a Confluence space key.",
|
|
};
|
|
}
|
|
|
|
const { origin, hostname } = new URL(baseUrl);
|
|
console.log(`-- Working Confluence ${origin} --`);
|
|
const loader = new ConfluencePagesLoader({
|
|
baseUrl: origin, // Use the origin to avoid issues with subdomains, ports, protocols, etc.
|
|
spaceKey,
|
|
username,
|
|
accessToken,
|
|
cloud,
|
|
});
|
|
|
|
const { docs, error } = await loader
|
|
.load()
|
|
.then((docs) => {
|
|
return { docs, error: null };
|
|
})
|
|
.catch((e) => {
|
|
return {
|
|
docs: [],
|
|
error: e.message?.split("Error:")?.[1] || e.message,
|
|
};
|
|
});
|
|
|
|
if (!docs.length || !!error) {
|
|
return {
|
|
success: false,
|
|
reason: error ?? "No pages found for that Confluence space.",
|
|
};
|
|
}
|
|
const outFolder = slugify(
|
|
`confluence-${hostname}-${v4().slice(0, 4)}`
|
|
).toLowerCase();
|
|
|
|
const outFolderPath =
|
|
process.env.NODE_ENV === "development"
|
|
? path.resolve(
|
|
__dirname,
|
|
`../../../../server/storage/documents/${outFolder}`
|
|
)
|
|
: path.resolve(process.env.STORAGE_DIR, `documents/${outFolder}`);
|
|
|
|
if (!fs.existsSync(outFolderPath))
|
|
fs.mkdirSync(outFolderPath, { recursive: true });
|
|
|
|
docs.forEach((doc) => {
|
|
if (!doc.pageContent) return;
|
|
|
|
const data = {
|
|
id: v4(),
|
|
url: doc.metadata.url + ".page",
|
|
title: doc.metadata.title || doc.metadata.source,
|
|
docAuthor: origin,
|
|
description: doc.metadata.title,
|
|
docSource: `${origin} Confluence`,
|
|
chunkSource: generateChunkSource(
|
|
{ doc, baseUrl: origin, spaceKey, accessToken, username, cloud },
|
|
response.locals.encryptionWorker
|
|
),
|
|
published: new Date().toLocaleString(),
|
|
wordCount: doc.pageContent.split(" ").length,
|
|
pageContent: doc.pageContent,
|
|
token_count_estimate: tokenizeString(doc.pageContent).length,
|
|
};
|
|
|
|
console.log(
|
|
`[Confluence Loader]: Saving ${doc.metadata.title} to ${outFolder}`
|
|
);
|
|
|
|
const fileName = sanitizeFileName(
|
|
`${slugify(doc.metadata.title)}-${data.id}`
|
|
);
|
|
writeToServerDocuments(data, fileName, outFolderPath);
|
|
});
|
|
|
|
return {
|
|
success: true,
|
|
reason: null,
|
|
data: {
|
|
spaceKey,
|
|
destination: outFolder,
|
|
},
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Gets the page content from a specific Confluence page, not all pages in a workspace.
|
|
* @returns
|
|
*/
|
|
async function fetchConfluencePage({
|
|
pageUrl,
|
|
baseUrl,
|
|
spaceKey,
|
|
username,
|
|
accessToken,
|
|
cloud = true,
|
|
}) {
|
|
if (!pageUrl || !baseUrl || !spaceKey || !username || !accessToken) {
|
|
return {
|
|
success: false,
|
|
content: null,
|
|
reason:
|
|
"You need either a username and access token, or a personal access token (PAT), to use the Confluence connector.",
|
|
};
|
|
}
|
|
|
|
if (!validBaseUrl(baseUrl)) {
|
|
return {
|
|
success: false,
|
|
content: null,
|
|
reason: "Provided base URL is not a valid URL.",
|
|
};
|
|
}
|
|
|
|
if (!spaceKey) {
|
|
return {
|
|
success: false,
|
|
content: null,
|
|
reason: "You need to provide a Confluence space key.",
|
|
};
|
|
}
|
|
|
|
console.log(`-- Working Confluence Page ${pageUrl} --`);
|
|
const loader = new ConfluencePagesLoader({
|
|
baseUrl, // Should be the origin of the baseUrl
|
|
spaceKey,
|
|
username,
|
|
accessToken,
|
|
cloud,
|
|
});
|
|
|
|
const { docs, error } = await loader
|
|
.load()
|
|
.then((docs) => {
|
|
return { docs, error: null };
|
|
})
|
|
.catch((e) => {
|
|
return {
|
|
docs: [],
|
|
error: e.message?.split("Error:")?.[1] || e.message,
|
|
};
|
|
});
|
|
|
|
if (!docs.length || !!error) {
|
|
return {
|
|
success: false,
|
|
reason: error ?? "No pages found for that Confluence space.",
|
|
content: null,
|
|
};
|
|
}
|
|
|
|
const targetDocument = docs.find(
|
|
(doc) => doc.pageContent && doc.metadata.url === pageUrl
|
|
);
|
|
if (!targetDocument) {
|
|
return {
|
|
success: false,
|
|
reason: "Target page could not be found in Confluence space.",
|
|
content: null,
|
|
};
|
|
}
|
|
|
|
return {
|
|
success: true,
|
|
reason: null,
|
|
content: targetDocument.pageContent,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Validates if the provided baseUrl is a valid URL at all.
|
|
* @param {string} baseUrl
|
|
* @returns {boolean}
|
|
*/
|
|
function validBaseUrl(baseUrl) {
|
|
try {
|
|
new URL(baseUrl);
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate the full chunkSource for a specific Confluence page so that we can resync it later.
|
|
* This data is encrypted into a single `payload` query param so we can replay credentials later
|
|
* since this was encrypted with the systems persistent password and salt.
|
|
* @param {object} chunkSourceInformation
|
|
* @param {import("../../EncryptionWorker").EncryptionWorker} encryptionWorker
|
|
* @returns {string}
|
|
*/
|
|
function generateChunkSource(
|
|
{ doc, baseUrl, spaceKey, accessToken, username, cloud },
|
|
encryptionWorker
|
|
) {
|
|
const payload = {
|
|
baseUrl,
|
|
spaceKey,
|
|
token: accessToken,
|
|
username,
|
|
cloud,
|
|
};
|
|
return `confluence://${doc.metadata.url}?payload=${encryptionWorker.encrypt(
|
|
JSON.stringify(payload)
|
|
)}`;
|
|
}
|
|
|
|
module.exports = {
|
|
loadConfluence,
|
|
fetchConfluencePage,
|
|
};
|