1
0
Fork 0
Anvil/src/lib/server/ap.js
André Jaenisch 1e52798e6a
chore: introduce REUSE
This required adding annotation to the whole project.
I need to double-check whether I have the license banner of AGPL in
place everywhere.

Signed-off-by: André Jaenisch <andre.jaenisch@posteo.de>
2024-03-06 16:03:12 +01:00

185 lines
4.5 KiB
JavaScript

/* Library to talk to Vervis.
* Copyright (C) 2024 André Jaenisch
* SPDX-FileCopyrightText: 2024 André Jaenisch
* SPDX-License-Identifier: AGPL-3.0-or-later
*
* 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 <http://www.gnu.org/licenses/>.
*/
import { Agent } from 'node:https';
import axios, { AxiosHeaders } from 'axios';
import * as cheerio from 'cheerio';
import pkg from '../../../package.json';
const loginResponse = {
id: '87bcb6de-bb70-11ee-b719-6756da82e80f',
username: 'hax0r',
acct: 'hax0r',
display_name: 'Jane Doe',
bot: false,
created_at: '2024-01-13T01:23:45.000Z',
note: '<p>Hackse for life</p>',
url: 'https://example.com/@hax0r',
avatar: 'https://avatars.example.com/hax0r',
role: { name: 'user' }
};
let username = loginResponse.username;
export const requests = {
get: (url) => {
if (url === '/profile') {
return Promise.resolve({
...loginResponse,
username
});
}
return Promise.reject(new Error('Unauthorized'));
},
post: async (url, params) => {
if (url === '/login') {
let username;
try {
const response = await getHomepage({ ...params });
return {
...loginResponse,
...response
};
} catch (e) {
console.error(e);
}
username = params.account;
return {
...loginResponse,
username: params.account
};
}
return Promise.reject(new Error('Invalid Login'));
}
};
async function getRemoteLogin(vervis) {
const httpsAgent = new Agent({ rejectUnauthorized: false });
const headers = new AxiosHeaders();
headers.setUserAgent(`Anvil/${pkg.version}`);
const instance = axios.create({
baseURL: vervis,
headers,
httpsAgent,
withCredentials: true
});
const response = await instance.get('/auth/login');
const dom = cheerio.load(response.data);
const form = dom('form');
return {
instance,
action: form.attr('action'),
csrf: dom('form input[name="_token"]').attr('value'),
enctype: form.attr('enctype'),
method: form.attr('method'),
cookies: response.headers['set-cookie']
};
}
async function postRemoteLogin({ username, password, loginFormData }) {
const headers = new AxiosHeaders();
headers.setContentType(loginFormData.enctype);
headers.set({ Cookie: loginFormData.cookies.join(';') });
const data = {
_token: loginFormData.csrf,
f1: username,
f2: password
};
return loginFormData.instance({
method: loginFormData.method,
url: loginFormData.action,
headers,
data
});
}
async function getHomepage({ account, passphrase, server }) {
const vervis =
{
1: 'https://fig.fr33domlover.site',
2: 'https://grape.fr33domlover.site',
3: 'https://walnut.fr33domlover.site'
}[server] || null;
const loginFormData = await getRemoteLogin(vervis);
const loggedInResponse = await postRemoteLogin({
username: account,
password: passphrase,
loginFormData
});
if (loggedInResponse.status === 200) {
const headers = new AxiosHeaders();
headers.set({ Cookie: loggedInResponse.headers['set-cookie'].join(';') });
const response = await loginFormData.instance.get('/', {
headers
});
const patches = [];
const projects = [];
const repos = [];
const teams = [];
const tickets = [];
const dom = cheerio.load(response.data);
if (dom('h2:contains("Your teams") + p').text().includes("aren't a member")) {
console.log('No teams');
}
if (dom('h2:contains("Your repos") + ul').text() === '') {
console.log('No repos');
}
if (dom('h2:contains("Your ticket trackers") + ul').text() === '') {
console.log('No ticket trackers');
}
if (dom('h2:contains("Your patch trackers") + ul').text() === '') {
console.log('No patch trackers');
}
if (dom('h2:contains("Your projects") + ul').text() === '') {
console.log('No projects');
} else {
dom('h2:contains("Your projects") + ul li a')
.get()
.map((a) => {
projects.push(dom(a).text());
});
}
return {
cookies: response.headers['set-cookie'].join(';'),
patches,
projects,
repos,
teams,
tickets,
username: account,
vervis
};
}
return null;
}