init
This commit is contained in:
commit
855e8c81e8
7
.dockerignore
Normal file
7
.dockerignore
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
node_modules/
|
||||||
|
temp/
|
||||||
|
package-lock.json
|
||||||
|
Dockerfile
|
||||||
|
docker-compose.yml
|
||||||
|
data.json
|
||||||
|
.gitignore
|
16
.drone.yml
Normal file
16
.drone.yml
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
kind: pipeline
|
||||||
|
name: default
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: docker
|
||||||
|
image: plugins/docker
|
||||||
|
settings:
|
||||||
|
username: bradbot1
|
||||||
|
password:
|
||||||
|
from_secret: access_token
|
||||||
|
repo: bradbot1/gitea-forwarder
|
||||||
|
tags: latest
|
||||||
|
|
||||||
|
trigger:
|
||||||
|
branch:
|
||||||
|
- master
|
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
node_modules/
|
||||||
|
package-lock.json
|
24
Dockerfile
Normal file
24
Dockerfile
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
FROM node:alpine
|
||||||
|
|
||||||
|
RUN apk update
|
||||||
|
RUN apk add git
|
||||||
|
RUN apk add doas
|
||||||
|
RUN apk add --update --no-cache openssh-client
|
||||||
|
|
||||||
|
RUN mkdir -p -m 0700 ~/.ssh
|
||||||
|
RUN ssh-keyscan github.com >> ~/.ssh/known_hosts
|
||||||
|
RUN ssh-keyscan git.bb1.fun >> ~/.ssh/known_hosts
|
||||||
|
RUN ssh-keyscan gitlab.com >> ~/.ssh/known_hosts
|
||||||
|
RUN ssh-keyscan bitbucket.org >> ~/.ssh/known_hosts
|
||||||
|
RUN ssh-keyscan codeberg.org >> ~/.ssh/known_hosts
|
||||||
|
#RUN eval `ssh-agent -s`
|
||||||
|
WORKDIR /usr/app
|
||||||
|
|
||||||
|
COPY ./ ./
|
||||||
|
|
||||||
|
RUN npm install
|
||||||
|
RUN npm run build
|
||||||
|
RUN rm ./src/ -fr
|
||||||
|
RUN rm tsconfig.json
|
||||||
|
|
||||||
|
ENTRYPOINT [ "./start.sh" ]
|
24
data.json
Normal file
24
data.json
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"origin": "https://git.example.com/user/repo",
|
||||||
|
"webhook": "CustomWebhookSlug",
|
||||||
|
"recipients": [
|
||||||
|
{
|
||||||
|
"url": "https://gitlab.com/user/repo",
|
||||||
|
"humanName": "An optional human name for logging",
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"old": "user@example.com",
|
||||||
|
"email": "00000000+user@users.noreply.github.com",
|
||||||
|
"name": "user"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"old": "user2@example.com",
|
||||||
|
"email": "00000001+user2@users.noreply.github.com",
|
||||||
|
"name": "user2"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
16
docker-compose.yml
Normal file
16
docker-compose.yml
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
version: '3'
|
||||||
|
|
||||||
|
services:
|
||||||
|
forwarder:
|
||||||
|
build: .
|
||||||
|
image: bradbot1/gitea-forwarder
|
||||||
|
ports:
|
||||||
|
- '80:80'
|
||||||
|
environment:
|
||||||
|
PORT: 80
|
||||||
|
# if you want ssl
|
||||||
|
#SSL_CERT_PATH:
|
||||||
|
#SSL_KEY_PATH:
|
||||||
|
#SSL_PASSWORD:
|
||||||
|
volumes:
|
||||||
|
- ./data.json:./data.json
|
17
package.json
Normal file
17
package.json
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
{
|
||||||
|
"scripts": {
|
||||||
|
"dev": "ts-node-dev --clear ./src/index.ts",
|
||||||
|
"build": "tsc",
|
||||||
|
"start": "node dist/index.js"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"0http": "^3.4.3",
|
||||||
|
"body-parser": "^1.20.1",
|
||||||
|
"low-http-server": "^4.1.0",
|
||||||
|
"simple-git": "^3.16.1"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"ts-node-dev": "^2.0.0",
|
||||||
|
"typescript": "^4.9.5"
|
||||||
|
}
|
||||||
|
}
|
24
src/Forward/Forward.ts
Normal file
24
src/Forward/Forward.ts
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
import { Recipient } from './Recipient'
|
||||||
|
|
||||||
|
export class Forward {
|
||||||
|
|
||||||
|
public readonly webhook: string;
|
||||||
|
public readonly origin: string;
|
||||||
|
private readonly recipients: Recipient[] = [];
|
||||||
|
|
||||||
|
constructor(webhookId: string, origin: string) {
|
||||||
|
this.webhook = webhookId;
|
||||||
|
this.origin = origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public createRecipient(url: string, humanName: string|undefined):Recipient {
|
||||||
|
const recipient = new Recipient(url, humanName);
|
||||||
|
this.recipients.push(recipient);
|
||||||
|
return recipient;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getRecipients(): Recipient[] {
|
||||||
|
return this.recipients;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
32
src/Forward/ForwardManager.ts
Normal file
32
src/Forward/ForwardManager.ts
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
import { Forward } from './Forward';
|
||||||
|
|
||||||
|
const known_forwards: Forward[] = [];
|
||||||
|
/**
|
||||||
|
* Generates a 32 character long random string
|
||||||
|
*/
|
||||||
|
function generateRandomWebhookId(): string {
|
||||||
|
return (Math.random().toString(36).substring(2)
|
||||||
|
+Math.random().toString(36).substring(2)
|
||||||
|
+Math.random().toString(36).substring(2)
|
||||||
|
+Math.random().toString(36).substring(2, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getForwardByWebhook(webhookId: string): Forward|null {
|
||||||
|
for (var forward of known_forwards)
|
||||||
|
if (forward.webhook === webhookId)
|
||||||
|
return forward;
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getForwards(): Forward[] {
|
||||||
|
return known_forwards;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function createForward(origin: string, webhook: string|undefined): Forward {
|
||||||
|
if (webhook === undefined) {
|
||||||
|
do {
|
||||||
|
webhook = generateRandomWebhookId();
|
||||||
|
} while (getForwardByWebhook(webhook) != null);
|
||||||
|
}
|
||||||
|
return new Forward(webhook, origin);
|
||||||
|
};
|
24
src/Forward/Recipient.ts
Normal file
24
src/Forward/Recipient.ts
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
import { GitAuthor } from '../Git/GitAuthor';
|
||||||
|
|
||||||
|
export class Recipient {
|
||||||
|
|
||||||
|
public readonly humanName: string;
|
||||||
|
public readonly url:string;
|
||||||
|
public authorMap: Map<string, GitAuthor> = new Map<string, GitAuthor>();
|
||||||
|
|
||||||
|
constructor(url: string, humanName: string|undefined) {
|
||||||
|
this.url = url;
|
||||||
|
this.humanName = humanName||url;
|
||||||
|
}
|
||||||
|
|
||||||
|
public setAuthor(gitAuthor: string, newName: string, newEmail: string):void {
|
||||||
|
this.authorMap.set(gitAuthor, new GitAuthor(newName, newEmail));
|
||||||
|
}
|
||||||
|
|
||||||
|
public formatAuthor(gitAuthor: string): string {
|
||||||
|
const author: GitAuthor|undefined = this.authorMap.get(gitAuthor);
|
||||||
|
if (author === undefined) return gitAuthor;
|
||||||
|
return author.format();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
15
src/Git/GitAuthor.ts
Normal file
15
src/Git/GitAuthor.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
export class GitAuthor {
|
||||||
|
|
||||||
|
public readonly name: string;
|
||||||
|
public readonly email: string;
|
||||||
|
|
||||||
|
constructor(name: string, email: string) {
|
||||||
|
this.name = name;
|
||||||
|
this.email = email;
|
||||||
|
}
|
||||||
|
|
||||||
|
public format(): string {
|
||||||
|
return `${this.name} <${this.email}>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
75
src/Git/GitManager.ts
Normal file
75
src/Git/GitManager.ts
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
import { Forward } from '../Forward/Forward'
|
||||||
|
import { simpleGit as Git } from 'simple-git';
|
||||||
|
import { GitAuthor } from './GitAuthor';
|
||||||
|
import { Recipient } from '../Forward/Recipient';
|
||||||
|
import { execSync } from 'child_process';
|
||||||
|
import { writeFileSync } from 'fs'
|
||||||
|
|
||||||
|
export async function cloneRepo(repo: string, out: string = __dirname): Promise<void> {
|
||||||
|
try {
|
||||||
|
await Git().clone(repo, out);
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
await Git(out).pull();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function push(repo: string, out: string = __dirname): Promise<void> {
|
||||||
|
const git = Git(out);
|
||||||
|
git.addRemote("fumo", repo);
|
||||||
|
git.push("fumo");
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getCommitAuthors(out: string = __dirname): Promise<string[]> {
|
||||||
|
const git = Git(out);
|
||||||
|
const oldAuthors: Set<string> = new Set<string>();
|
||||||
|
for (const commit of (await git.log()).all)
|
||||||
|
oldAuthors.add(commit.author_email);
|
||||||
|
return Array.from(oldAuthors);
|
||||||
|
}
|
||||||
|
|
||||||
|
const _changeCommitAuthorsReplace: string = `#!/bin/sh
|
||||||
|
|
||||||
|
git filter-branch --env-filter '
|
||||||
|
|
||||||
|
an="$GIT_AUTHOR_NAME"
|
||||||
|
am="$GIT_AUTHOR_EMAIL"
|
||||||
|
cn="$GIT_COMMITTER_NAME"
|
||||||
|
cm="$GIT_COMMITTER_EMAIL"
|
||||||
|
|
||||||
|
echo $GIT_COMMITTER_EMAIL
|
||||||
|
|
||||||
|
if [ "$GIT_COMMITTER_EMAIL" = "OLD_EMAIL" ]
|
||||||
|
then
|
||||||
|
cn="NEW_NAME"
|
||||||
|
cm="NEW_EMAIL"
|
||||||
|
fi
|
||||||
|
if [ "$GIT_AUTHOR_EMAIL" = "OLD_EMAIL" ]
|
||||||
|
then
|
||||||
|
an="NEW_NAME"
|
||||||
|
am="NEW_EMAIL"
|
||||||
|
fi
|
||||||
|
|
||||||
|
export GIT_AUTHOR_NAME="$an"
|
||||||
|
export GIT_AUTHOR_EMAIL="$am"
|
||||||
|
export GIT_COMMITTER_NAME="$cn"
|
||||||
|
export GIT_COMMITTER_EMAIL="$cm"
|
||||||
|
'`;
|
||||||
|
|
||||||
|
|
||||||
|
export async function changeCommitAuthors(recipient: Recipient, out: string = __dirname): Promise<void> {
|
||||||
|
for (const author of await getCommitAuthors(out)) {
|
||||||
|
const newAuthor: GitAuthor|undefined = recipient.authorMap.get(author);
|
||||||
|
if (newAuthor === undefined) continue;
|
||||||
|
writeFileSync(out + "git_change.sh", _changeCommitAuthorsReplace.replace(/OLD_EMAIL/g, author).replace(/NEW_NAME/g, newAuthor.name).replace(/NEW_EMAIL/g, newAuthor.email));
|
||||||
|
execSync("/bin/sh " + out + "git_change.sh", {
|
||||||
|
cwd: out
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function handleForward(forward: Forward):Promise<void> {
|
||||||
|
const git = Git().env('GIT_SSH_COMMAND', 'ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no');
|
||||||
|
await git.clone(forward.origin);
|
||||||
|
git.log()
|
||||||
|
}
|
35
src/Server/Server.ts
Normal file
35
src/Server/Server.ts
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
import { setupWebhookRoutes, WebhookRecievedCallback } from './Webhook';
|
||||||
|
|
||||||
|
export function createServer(port: number, webhookRecievedCallback: WebhookRecievedCallback): void {
|
||||||
|
const { router, server } = require('0http')({
|
||||||
|
defaultRoute: (_: any, res: any) => {
|
||||||
|
res.statusCode = 404;
|
||||||
|
res.setHeader('Content-Type', 'application/json; charset=utf-8');
|
||||||
|
res.end('{"error":"route_not_found",message:"There route requested does not exist!"}');
|
||||||
|
},
|
||||||
|
// router: require('find-my-way')(),
|
||||||
|
server: require('low-http-server')({
|
||||||
|
cert_file_name: process.env["SSL_CERT_PATH"],
|
||||||
|
key_file_name: process.env["SSL_KEY_PATH"],
|
||||||
|
password: process.env["SSL_PASSWORD"]
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
router.use(require('body-parser').json({
|
||||||
|
limit: '5kb',
|
||||||
|
strict: true,
|
||||||
|
type: "application/json"
|
||||||
|
}));
|
||||||
|
|
||||||
|
router.get('/', (_:any, res:any) => {
|
||||||
|
res.statusCode = 200;
|
||||||
|
res.setHeader('Content-Type', 'application/json; charset=utf-8');
|
||||||
|
res.end('{"name":"GiteaForwarder","version":"1.0.0","about":"Forwarding git data to various repositories!"}');
|
||||||
|
})
|
||||||
|
|
||||||
|
setupWebhookRoutes(router, webhookRecievedCallback);
|
||||||
|
|
||||||
|
server.listen(port, () => {
|
||||||
|
console.log("[WEB] Now listening on port " + port);
|
||||||
|
});
|
||||||
|
}
|
12
src/Server/Webhook.ts
Normal file
12
src/Server/Webhook.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
export type WebhookRecievedCallback = (webhookId: string, webhook: Promise<any>) => void;
|
||||||
|
|
||||||
|
export function setupWebhookRoutes(router:any, webhookRecievedCallback:WebhookRecievedCallback):void {
|
||||||
|
|
||||||
|
router.post('/webhook/:webhook', (req: any, res: any) => {
|
||||||
|
const webhookId = req.params.webhook;
|
||||||
|
console.log("[WEB] Recieved webhook " + webhookId);
|
||||||
|
webhookRecievedCallback(webhookId, res.body);
|
||||||
|
res.statusCode = 200;
|
||||||
|
res.end();
|
||||||
|
});
|
||||||
|
}
|
91
src/index.ts
Normal file
91
src/index.ts
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
import { Forward } from './Forward/Forward';
|
||||||
|
import { createForward, getForwardByWebhook } from './Forward/ForwardManager';
|
||||||
|
import { createServer } from './Server/Server';
|
||||||
|
import { env, cwd } from 'process';
|
||||||
|
import { cloneRepo, changeCommitAuthors, push } from './Git/GitManager';
|
||||||
|
import { rmSync, existsSync, readFileSync } from 'fs'
|
||||||
|
|
||||||
|
var dataToLoad: string;
|
||||||
|
|
||||||
|
if (env.hasOwnProperty("DATA")) {
|
||||||
|
dataToLoad = env["DATA"]||"";
|
||||||
|
} else if (existsSync("./data.json")) {
|
||||||
|
dataToLoad = readFileSync("./data.json").toString();
|
||||||
|
} else {
|
||||||
|
console.error("Failed to find any data to load!");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
var parsedData: any;
|
||||||
|
|
||||||
|
try {
|
||||||
|
parsedData = JSON.parse(dataToLoad);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to parse the provided data!");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Array.isArray(parsedData)) parsedData = [parsedData];
|
||||||
|
|
||||||
|
for (const forwardData of parsedData) {
|
||||||
|
if (!forwardData.hasOwnProperty("origin")) {
|
||||||
|
console.error("No origin of forward!");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const forward = createForward(forwardData.origin, forwardData.webhook);
|
||||||
|
console.log("Created forward on webhook: " + forward.webhook);
|
||||||
|
if (forwardData.hasOwnProperty("recipients")) {
|
||||||
|
for (const recipientData of forwardData.recipients) {
|
||||||
|
if (!recipientData.hasOwnProperty("url")) {
|
||||||
|
console.error("No url provided for recipient");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const recipient = forward.createRecipient(recipientData.url, recipientData.humanName);
|
||||||
|
if (!recipientData.hasOwnProperty("authors")) {
|
||||||
|
console.error("No authors provided for recipient: " + recipient.humanName||recipient.url);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
for (const author of recipientData.authors) {
|
||||||
|
if (!author.hasOwnProperty("old")) {
|
||||||
|
console.error("No old email provided for recipient: " + author.humanName||recipient.url);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!author.hasOwnProperty("email")) {
|
||||||
|
console.error("No new email provided for recipient: " + author.humanName||recipient.url);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!author.hasOwnProperty("name")) {
|
||||||
|
console.error("No new name provided for recipient: " + author.humanName||recipient.url);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
recipient.setAuthor(author.old, author.name, author.email);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log("No recipients found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
createServer(parseInt(env["PORT"]||"3000"), async (webhookId: string, webhook: Promise<any>) => {
|
||||||
|
const forward: Forward|null = getForwardByWebhook(webhookId);
|
||||||
|
if (forward == null) return;
|
||||||
|
const webhookData: any = await webhook;
|
||||||
|
if (!webhookData.hasOwnProperty("repository")) return;
|
||||||
|
const url: any = webhookData.repository.clone_url || webhookData.repository.ssh_url || webhookData.repository.html_url;
|
||||||
|
if (typeof url !== "string") return;
|
||||||
|
if (forward.origin !== url) {
|
||||||
|
console.log("[VAL] Invalid webhook origin!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log("[VAL] Validated " + url);
|
||||||
|
const outputDir = cwd() + "/git_output_" + Math.random().toString(32).substring(2,8) + "/";
|
||||||
|
for (const sendTo of forward.getRecipients()) {
|
||||||
|
await cloneRepo(forward.origin, outputDir);
|
||||||
|
await changeCommitAuthors(sendTo, outputDir);
|
||||||
|
await push(sendTo.url, outputDir);
|
||||||
|
rmSync(outputDir, {
|
||||||
|
recursive: true,
|
||||||
|
force: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
15
start.sh
Normal file
15
start.sh
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
if ! [ -f "/root/.ssh/id_rsa.pub" ]; then
|
||||||
|
echo "No key found, creating a new one!"
|
||||||
|
ssh-keygen -t rsa -f ~/.ssh/id_rsa -q -P ""
|
||||||
|
chmod 700 /~/.ssh/id_rsa
|
||||||
|
chmod 700 /~/.ssh/id_rsa.pub
|
||||||
|
echo " IdentityFile ~/.ssh/id_rsa" >> /etc/ssh/ssh_config
|
||||||
|
echo "Your new public key for this instance:"
|
||||||
|
cat ~/.ssh/id_rsa.pub
|
||||||
|
fi
|
||||||
|
|
||||||
|
npm run start
|
||||||
|
|
||||||
|
/bin/sh
|
15
tsconfig.json
Normal file
15
tsconfig.json
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "es2016",
|
||||||
|
"module": "commonjs",
|
||||||
|
"esModuleInterop": true,
|
||||||
|
"forceConsistentCasingInFileNames": true,
|
||||||
|
"strict": true,
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"rootDir": "src",
|
||||||
|
"outDir": "dist",
|
||||||
|
"noImplicitAny": true,
|
||||||
|
"allowJs": true
|
||||||
|
},
|
||||||
|
"exclude": [ "dist", "temp" ]
|
||||||
|
}
|
Loading…
Reference in a new issue