gogitlabber/main.go
2025-01-01 23:29:16 +01:00

244 lines
6.1 KiB
Go

package main
import (
"encoding/json"
"fmt"
"github.com/k0kubun/go-ansi"
"github.com/schollz/progressbar/v3"
"log"
"net/http"
"os"
"os/exec"
"strings"
)
type Repository struct {
Name string `json:"name"`
PathWithNamespace string `json:"path_with_namespace"`
}
var repoDestinationPre string
var includeArchived string
var gitlabToken string
var gitlabHost string
var pullError []string
func main() {
// load environment variables first, they will be overridden
// by argument flags if specified.
if err := loadEnvironmentVariables(); err != nil {
log.Fatalf("Error loading environment variables: %v", err)
}
// manage all argument magic
manageArguments()
// fetch repository information from gitlab
repositories, err := fetchRepositories()
if err != nil {
log.Fatalf("Error fetching repositories: %v", err)
}
// manage found repositories
checkoutRepositories(repositories)
printPullerror(pullError)
}
func loadEnvironmentVariables() error {
gitlabToken = os.Getenv("GITLAB_API_KEY")
if gitlabToken == "" {
return fmt.Errorf("GITLAB_API_KEY environment variable is not set")
}
gitlabHost = os.Getenv("GITLAB_HOSTNAME")
if gitlabHost == "" {
return fmt.Errorf("GITLAB_HOSTNAME environment variable is not set")
}
return nil
}
func manageArguments() {
// require at least the destination argument
if len(os.Args) <= 1 {
fmt.Println("Usage: gogitlabber --destination=<directory>")
fmt.Println("Example: gogitlabber --destination=$HOME/repos")
os.Exit(1)
}
// parse arguments
for _, arg := range os.Args[1:] {
switch {
case strings.HasPrefix(arg, "--archived="):
includeArchived = strings.TrimPrefix(arg, "--archived=")
case strings.HasPrefix(arg, "--destination="):
repoDestinationPre = strings.TrimPrefix(arg, "--destination=")
case strings.HasPrefix(arg, "--gitlab-api-token="):
gitlabToken = strings.TrimPrefix(arg, "--gitlab-api-token=")
case strings.HasPrefix(arg, "--gitlab-url="):
gitlabHost = strings.TrimPrefix(arg, "--gitlab-url=")
default:
fmt.Println("Usage: gogitlabber --destination=<directory>")
fmt.Println("Example: gogitlabber --destination=$HOME/repos")
os.Exit(1)
}
}
// --archive options:
// - any (fetch both)
// - only (fetch archived only)
// - excluded (fetch non-archived only - default)
if includeArchived == "" {
includeArchived = "excluded"
}
if includeArchived != "any" &&
includeArchived != "only" &&
includeArchived != "excluded" {
fmt.Println("Usage: gogitlabber --archived=(any|excluded|only)")
os.Exit(1)
}
// fail if destination is unknown
if repoDestinationPre == "" {
fmt.Println("Fatal: No destination found.")
fmt.Println("Example: gogitlabber --destination=$HOME/repos")
fmt.Println("Usage: gogitlabber --destination=$HOME/repos")
os.Exit(1)
}
// add slash if not provided to directory
if !strings.HasSuffix(repoDestinationPre, "/") {
repoDestinationPre += "/"
}
}
func fetchRepositories() ([]Repository, error) {
// default options
membership := "membership=true"
perpage := "per_page=100"
order := "order_by=name"
// configure archived options
var archived string
switch {
case includeArchived == "excluded":
archived = "&archived=false"
case includeArchived == "only":
archived = "&archived=true"
default:
archived = ""
}
url := fmt.Sprintf("https://%s/api/v4/projects?%s&%s&%s%s",
gitlabHost, membership, order, perpage, archived)
req, err := http.NewRequest("GET", url, nil)
if err != nil {
return nil, fmt.Errorf("error creating request: %v", err)
}
req.Header.Set("PRIVATE-TOKEN", gitlabToken)
client := &http.Client{}
resp, err := client.Do(req)
if err != nil {
return nil, fmt.Errorf("error making request: %v", err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
return nil, fmt.Errorf("API request failed with status: %d", resp.StatusCode)
}
var repositories []Repository
if err := json.NewDecoder(resp.Body).Decode(&repositories); err != nil {
return nil, fmt.Errorf("error decoding response: %v", err)
}
return repositories, nil
}
func checkoutRepositories(repositories []Repository) {
repoCount := len(repositories)
fmt.Printf("Found %d repositories", repoCount)
// make progressbar using:
// - github.com/k0kubun/go-ansi
// - github.com/schollz/progressbar/v3
bar := progressbar.NewOptions(
repoCount,
progressbar.OptionSetWriter(ansi.NewAnsiStdout()),
progressbar.OptionEnableColorCodes(true),
progressbar.OptionShowCount(),
progressbar.OptionSetElapsedTime(true),
progressbar.OptionSetPredictTime(false),
progressbar.OptionSetWidth(20),
progressbar.OptionSetDescription("Getting your repositories..."),
progressbar.OptionSetTheme(progressbar.Theme{
Saucer: "[green]=[reset]",
SaucerHead: "[green]>[reset]",
SaucerPadding: " ",
BarStart: "[",
BarEnd: "]",
}),
)
for _, repo := range repositories {
repoName := string(repo.PathWithNamespace)
gitlabUrl := fmt.Sprintf("https://gitlab-token:%s@%s/%s.git",
gitlabToken, gitlabHost, repoName)
repoDestination := repoDestinationPre + repoName
cloneCmd := exec.Command("git", "clone", gitlabUrl, repoDestination)
cloneOutput, err := cloneCmd.CombinedOutput()
if err != nil {
// if repo already exists, try to pull the latest changes
if strings.Contains(string(cloneOutput),
"already exists and is not an empty directory") {
pullRepositories(repoDestination)
bar.Add(1)
continue
}
log.Printf("❌ Error cloning %s: %v\n%s", repoName, err, string(cloneOutput))
bar.Add(1)
continue
}
bar.Add(1)
}
// print empty line as the bar does not do that
fmt.Println("")
}
func pullRepositories(repoDestination string) {
pullCmd := exec.Command("git", "-C", repoDestination, "pull", "origin")
pullOutput, err := pullCmd.CombinedOutput()
if err != nil {
if strings.Contains(string(pullOutput),
"You have unstaged changes") {
pullError = append(pullError, repoDestination)
}
}
}
func printPullerror(pullError []string) {
if len(pullError) > 0 {
for _, repo := range pullError {
fmt.Printf("❕%s has unstaged changes.\n", repo)
}
}
}