Compare commits

..

2 Commits

Author SHA1 Message Date
9b535f621b
also update visibility when syncing 2023-04-05 20:48:25 +02:00
494be7abe2
doc for token scope 2023-04-05 20:47:54 +02:00
4 changed files with 28 additions and 12 deletions

View File

@ -1,10 +1,10 @@
# Backup Github repos to Gitea
# Backup GitHub repos to Gitea
Simple docker image that syncs your Github repos to a given Gitea server. It takes all the repos (public and private) and sets up a mirror. Private repos will stay a private mirror. Repos that are already setup will be skipped.
Simple docker image that syncs your GitHub repos to a given Gitea server. It takes all the repos (public and private) and sets up a mirror. Private repos will stay a private mirror. Repos that are already set up will be skipped.
## Quick Start
Create a `docker-compose.yaml` and `.env` (see `.env.sample`). Create and insert tokens and you are ready to go.
Create a `docker-compose.yaml` and `.env` (see `.env.sample`). Create and insert tokens, and you are ready to go.
```yaml
version: '3.8'
@ -19,13 +19,14 @@ services:
## Configuration
```sh
# Github PAT
# Also works with the new fine grained, read-only tokens
# Github PAT (deprecated)
# or
# Fine Grained token (Metadata and Content read-only scopes required)
GITHUB_TOKEN=
# Host of the Gitea server
GITEA_HOST=
# Gitea token
# Gitea token (scopes: repo)
GITEA_TOKEN=
# OPTIONAL

View File

@ -1,5 +1,5 @@
{
"version": "1.0.0",
"version": "1.1.0",
"type": "module",
"scripts": {
"start": "node .",

View File

@ -3,6 +3,7 @@ import axios, { AxiosError } from 'axios'
import { Config } from '../config.js'
import { logger } from '../logger.js'
import { ListRepositoriesResponse } from './gitea.types.js'
import { Repository } from './github.types.js'
const Base = axios.create({
baseURL: new URL('/api/v1', Config.gitea.host).href,
@ -77,3 +78,12 @@ export async function listAllRepositories() {
logger.debug('Listed all repositories in Gitea', { data: repos })
return repos
}
export async function updateRepository(owner: string, repo: string, body: Partial<Repository>) {
logger.debug('Updating repository', { owner, repo, body })
await Base({
url: `/repos/${owner}/${repo}`,
method: 'PATCH',
data: body,
})
}

View File

@ -1,4 +1,4 @@
import { listAllRepositories as giteaRepos, mirror, MirrorOptions } from './api/gitea.js'
import { listAllRepositories as giteaRepos, mirror, MirrorOptions, updateRepository } from './api/gitea.js'
import { listAllRepositories as githubRepos } from './api/github.js'
import { Config } from './config.js'
import { logger } from './logger.js'
@ -20,7 +20,12 @@ export async function sync() {
const sameName = syncedRepos.find((r) => r.name === repo.name || r.original_url === repo.clone_url)
if (sameName) {
if (sameName.original_url === repo.clone_url) {
logger.info('Already synced, skipping', { name: repo.name })
if (sameName.private === repo.private) logger.info('Already synced, skipping', { name: repo.name })
else {
logger.info('Visibility changed, updating', { name: repo.name })
const [owner, repository] = sameName.full_name.split('/')
await updateRepository(owner, repository, { private: repo.private })
}
} else {
logger.error('Repo with same name but different url', {
name: repo.name,
@ -42,9 +47,9 @@ export async function sync() {
logger.info('Mirrored repository', { name: repo.name })
}
logger.info('Finished sync')
} catch (e) {
logger.error(e)
logger.error('Failed to sync')
} catch (error) {
logger.debug(error)
logger.error('Failed to sync', { error: error instanceof Error ? error.message : 'Unknown error' })
} finally {
running = false
}