2021-08-27 08:03:24 -05:00
|
|
|
/*!
|
2023-03-16 04:16:01 -05:00
|
|
|
share2fedi - Instance-agnostic share page for the Fediverse.
|
|
|
|
Copyright (C) 2020-2023 Nikita Karamov <me@kytta.dev>
|
2021-08-27 08:03:24 -05:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as published
|
|
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2023-03-16 04:16:01 -05:00
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-or-later
|
2021-08-27 08:03:24 -05:00
|
|
|
*/
|
|
|
|
|
2023-03-16 06:16:21 -05:00
|
|
|
import http from "http";
|
2023-03-16 12:59:43 -05:00
|
|
|
import https from "https";
|
|
|
|
|
|
|
|
const pathsMap = {
|
|
|
|
mastodon: {
|
2023-03-17 08:08:08 -05:00
|
|
|
checkUrl: "/api/v1/instance/rules",
|
2023-03-16 12:59:43 -05:00
|
|
|
postUrl: "share",
|
|
|
|
textParam: "text",
|
|
|
|
},
|
|
|
|
gnuSocial: {
|
|
|
|
checkUrl: "/api/statusnet/version.xml",
|
|
|
|
postUrl: "/notice/new",
|
|
|
|
textParam: "status_textarea",
|
|
|
|
},
|
2023-03-17 08:08:08 -05:00
|
|
|
pleroma: {
|
|
|
|
checkUrl: "/api/v1/pleroma/federation_status",
|
|
|
|
postUrl: "share",
|
|
|
|
textParam: "message",
|
|
|
|
},
|
2023-03-16 12:59:43 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
const queryUrl = (url, service) => {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const get = url.protocol === "https:" ? https.get : http.get;
|
|
|
|
get(url, ({ statusCode }) => {
|
|
|
|
if (statusCode === 200) {
|
|
|
|
console.debug(url.href, "is", service);
|
|
|
|
resolve(service);
|
|
|
|
} else {
|
|
|
|
reject(url);
|
|
|
|
}
|
|
|
|
}).on("error", (error) => {
|
|
|
|
reject(error);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const detectService = async (instanceURL) => {
|
|
|
|
const checkPromises = Object.entries(pathsMap).map(
|
|
|
|
([service, { checkUrl }]) =>
|
2023-03-17 08:06:30 -05:00
|
|
|
queryUrl(new URL(checkUrl, instanceURL), service),
|
2023-03-16 12:59:43 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
return await Promise.any(checkPromises);
|
|
|
|
};
|
2023-03-16 06:16:21 -05:00
|
|
|
|
|
|
|
const requestListener = async (request, response) => {
|
2023-03-16 12:56:32 -05:00
|
|
|
if (request.method !== "POST") {
|
|
|
|
response.writeHead(405).end();
|
|
|
|
return;
|
2023-03-16 06:16:21 -05:00
|
|
|
}
|
|
|
|
|
2023-03-16 12:56:32 -05:00
|
|
|
let data = "";
|
|
|
|
request.on("data", (chunk) => {
|
|
|
|
data += chunk.toString();
|
|
|
|
});
|
2023-03-16 06:16:21 -05:00
|
|
|
|
2023-03-16 12:56:32 -05:00
|
|
|
request.on("end", () => {
|
|
|
|
const requestBody = new URLSearchParams(data);
|
|
|
|
const postText = requestBody.get("text") || "";
|
|
|
|
const instanceURL =
|
|
|
|
requestBody.get("instance") || "https://mastodon.social/";
|
2023-03-16 06:16:21 -05:00
|
|
|
|
2023-03-16 12:59:43 -05:00
|
|
|
detectService(instanceURL)
|
|
|
|
.then((service) => {
|
|
|
|
const publishUrl = new URL(pathsMap[service].postUrl, instanceURL);
|
|
|
|
publishUrl.search = new URLSearchParams([
|
|
|
|
[pathsMap[service].textParam, postText],
|
|
|
|
]);
|
|
|
|
response.writeHead(303, { Location: publishUrl.toString() }).end();
|
|
|
|
})
|
|
|
|
.catch((error) => {
|
|
|
|
response.writeHead(400).end(error);
|
|
|
|
});
|
2023-03-16 12:56:32 -05:00
|
|
|
});
|
2023-03-16 06:16:21 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
if (!import.meta.env || import.meta.env.PROD) {
|
|
|
|
http.createServer(requestListener).listen(8080);
|
|
|
|
}
|
|
|
|
|
|
|
|
export const viteNodeApp = requestListener;
|