Compare commits
No commits in common. "c5df43b3e7fe1c395cb859d29e32ed3b133038a2" and "3180af5bd4706df7aae932362a6a42a82c991a0e" have entirely different histories.
c5df43b3e7
...
3180af5bd4
|
@ -0,0 +1,6 @@
|
||||||
|
FROM alpine
|
||||||
|
|
||||||
|
COPY ./bin/gitlabfs /usr/bin/gitlabfs
|
||||||
|
|
||||||
|
ENTRYPOINT ["gitlabfs"]
|
||||||
|
|
2
Makefile
2
Makefile
|
@ -1,4 +1,4 @@
|
||||||
PROGRAM := gitforgefs
|
PROGRAM := gitlabfs
|
||||||
TARGET_DIR := ./bin
|
TARGET_DIR := ./bin
|
||||||
VERSION := $(shell git describe --tags --always)
|
VERSION := $(shell git describe --tags --always)
|
||||||
|
|
||||||
|
|
145
README.md
145
README.md
|
@ -1,94 +1,119 @@
|
||||||
# gitforgefs
|
# gitlabfs
|
||||||
|
|
||||||
*Formerly gitlabfs*
|
`gitlabfs` allows you to mount and navigate Gitlab groups and user personal projects as a FUSE filesystem with every groups represented as a folder and every projects represented as a symlink pointing on a local clone of the project.
|
||||||
|
|
||||||
`gitforgefs` allows you to mount and navigate git forges (Github, Gitlab, Gitea, etc.) as a [FUSE](https://github.com/libfuse/libfuse) filesystem with every groups, organization, and users represented as a folder and every repositories represented as a symlink pointing on a local clone of the project. This is helpful to automate the organization of your local clones.
|
Partial output of `tree`, truncated and with a max of 4 levels:
|
||||||
|
|
||||||
To help illustrate, this is the output of `tree` in a filesystem exposing all the repositories of a github user.
|
|
||||||
```
|
```
|
||||||
$ tree
|
$ tree -L 4
|
||||||
.
|
.
|
||||||
└── badjware
|
├── projects
|
||||||
├── aws-cloud-gaming -> /home/marchambault/.local/share/gitforgefs/github.com/257091317
|
│ └── gitlab-org
|
||||||
├── certbot -> /home/marchambault/.local/share/gitforgefs/github.com/122014287
|
│ ├── 5-minute-production-app
|
||||||
├── certbot-dns-cpanel -> /home/marchambault/.local/share/gitforgefs/github.com/131224547
|
│ │ ├── deploy-template -> /home/marchambault/.local/share/gitlabfs/gitlab.com/22487050
|
||||||
├── certbot-dns-ispconfig -> /home/marchambault/.local/share/gitforgefs/github.com/227005814
|
│ │ ├── examples
|
||||||
├── CommonLibVR -> /home/marchambault/.local/share/gitforgefs/github.com/832968971
|
│ │ ├── hipio -> /home/marchambault/.local/share/gitlabfs/gitlab.com/23344605
|
||||||
├── community -> /home/marchambault/.local/share/gitforgefs/github.com/424689724
|
│ │ ├── sandbox
|
||||||
├── docker-postal -> /home/marchambault/.local/share/gitforgefs/github.com/132605640
|
│ │ └── static-template -> /home/marchambault/.local/share/gitlabfs/gitlab.com/23203100
|
||||||
├── dotfiles -> /home/marchambault/.local/share/gitforgefs/github.com/192993195
|
│ ├── allocations -> /home/marchambault/.local/share/gitlabfs/gitlab.com/684698
|
||||||
├── ecommerce-exporter -> /home/marchambault/.local/share/gitforgefs/github.com/562583906
|
│ ├── apilab -> /home/marchambault/.local/share/gitlabfs/gitlab.com/2383700
|
||||||
├── FightClub5eXML -> /home/marchambault/.local/share/gitforgefs/github.com/246177579
|
│ ├── architecture
|
||||||
├── gitforgefs -> /home/marchambault/.local/share/gitforgefs/github.com/324617595
|
│ │ └── tasks -> /home/marchambault/.local/share/gitlabfs/gitlab.com/22351703
|
||||||
├── kustomize-plugins -> /home/marchambault/.local/share/gitforgefs/github.com/263480122
|
│ ├── async-retrospectives -> /home/marchambault/.local/share/gitlabfs/gitlab.com/7937396
|
||||||
├── librechat-mistral -> /home/marchambault/.local/share/gitforgefs/github.com/753193720
|
│ ├── auto-deploy-app -> /home/marchambault/.local/share/gitlabfs/gitlab.com/6329546
|
||||||
├── PapyrusExtenderSSE -> /home/marchambault/.local/share/gitforgefs/github.com/832969611
|
│ ├── auto-deploy-helm -> /home/marchambault/.local/share/gitlabfs/gitlab.com/3651684
|
||||||
├── Parsec-Cloud-Preparation-Tool -> /home/marchambault/.local/share/gitforgefs/github.com/258052650
|
│ ├── auto-devops-v12-10 -> /home/marchambault/.local/share/gitlabfs/gitlab.com/18629149
|
||||||
├── po3-Tweaks -> /home/marchambault/.local/share/gitforgefs/github.com/832969112
|
│ ├── backstage-changelog -> /home/marchambault/.local/share/gitlabfs/gitlab.com/7602162
|
||||||
├── prometheus-ecs-discovery -> /home/marchambault/.local/share/gitforgefs/github.com/187891900
|
│ ├── blob-examples -> /home/marchambault/.local/share/gitlabfs/gitlab.com/3094319
|
||||||
├── simplefuse -> /home/marchambault/.local/share/gitforgefs/github.com/111226611
|
│ ├── build
|
||||||
├── tmux-continuum -> /home/marchambault/.local/share/gitforgefs/github.com/160746043
|
│ │ ├── CNG -> /home/marchambault/.local/share/gitlabfs/gitlab.com/4359271
|
||||||
├── ttyd -> /home/marchambault/.local/share/gitforgefs/github.com/132514236
|
│ │ ├── CNG-mirror -> /home/marchambault/.local/share/gitlabfs/gitlab.com/7682093
|
||||||
├── usb-libvirt-hotplug -> /home/marchambault/.local/share/gitforgefs/github.com/128696299
|
│ │ ├── dsop-scripts -> /home/marchambault/.local/share/gitlabfs/gitlab.com/19310217
|
||||||
└── vfio-win10 -> /home/marchambault/.local/share/gitforgefs/github.com/388475049
|
│ │ ├── omnibus-mirror
|
||||||
|
│ │ └── tr-test-dependency-proxy -> /home/marchambault/.local/share/gitlabfs/gitlab.com/20085049
|
||||||
|
│ ├── charts
|
||||||
|
│ │ ├── apparmor -> /home/marchambault/.local/share/gitlabfs/gitlab.com/18991900
|
||||||
|
│ │ ├── auto-deploy-app -> /home/marchambault/.local/share/gitlabfs/gitlab.com/11915984
|
||||||
|
│ │ ├── components
|
||||||
|
│ │ ├── consul -> /home/marchambault/.local/share/gitlabfs/gitlab.com/18663049
|
||||||
|
│ │ ├── deploy-image-helm-base -> /home/marchambault/.local/share/gitlabfs/gitlab.com/7453181
|
||||||
|
│ │ ├── elastic-stack -> /home/marchambault/.local/share/gitlabfs/gitlab.com/18439881
|
||||||
|
│ │ ├── fluentd-elasticsearch -> /home/marchambault/.local/share/gitlabfs/gitlab.com/17253921
|
||||||
|
│ │ ├── gitlab -> /home/marchambault/.local/share/gitlabfs/gitlab.com/3828396
|
||||||
|
│ │ ├── gitlab-runner -> /home/marchambault/.local/share/gitlabfs/gitlab.com/6329679
|
||||||
|
│ │ ├── knative -> /home/marchambault/.local/share/gitlabfs/gitlab.com/16590122
|
||||||
|
│ │ └── plantuml -> /home/marchambault/.local/share/gitlabfs/gitlab.com/14372596
|
||||||
|
│ [...]
|
||||||
|
└── users
|
||||||
|
└── badjware
|
||||||
|
└── test_project -> /home/marchambault/.local/share/gitlabfs/gitlab.com/23370783
|
||||||
|
|
||||||
24 directories, 0 files
|
696 directories, 0 files
|
||||||
```
|
```
|
||||||
|
|
||||||
## Supported forges
|
|
||||||
|
|
||||||
Currently, the following forges are supported:
|
|
||||||
|
|
||||||
| Forge | Name in configuration | API token permissions, if using an API key |
|
|
||||||
| ------------------------------- | --------------------- | ------------------------------------------------------ |
|
|
||||||
| [Gitlab](https://gitlab.com) | `gitlab` | `read_user`, `read_api` |
|
|
||||||
| [Github](https://github.com) | `github` | `repo` |
|
|
||||||
| [Gitea](https://gitea.com) | `gitea` | organization: `read`, repository: `read`, user: `read` |
|
|
||||||
| [Forgejo](https://forgejo.org/) | `gitea` | organization: `read`, repository: `read`, user: `read` |
|
|
||||||
|
|
||||||
Merge requests to add support to other forges are welcome.
|
|
||||||
|
|
||||||
## Install
|
## Install
|
||||||
|
|
||||||
Install [go](https://golang.org/) and run
|
Install [go](https://golang.org/) and run
|
||||||
``` sh
|
``` sh
|
||||||
go install github.com/badjware/gitforgefs
|
go install github.com/badjware/gitlabfs@latest
|
||||||
```
|
```
|
||||||
|
|
||||||
The executable will be in `$GOPATH/bin/gitforgefs` or `~/go/bin/gitforgefs` by default. For convenience, add `~/go/bin` in your `$PATH` if not done already.
|
The executable will be in `$GOPATH/bin/gitlabfs` or `~/go/bin/gitlabfs` by default. For convenience, copy `gitlabfs` somewhere suitable or add `~/go/bin` in your `PATH`.
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
Download the [example configuration file](./config.example.yaml) and edit the default configuration to suit your needs.
|
Download the [example configuration file](./config.example.yaml) and edit the default configuration to suit your needs.
|
||||||
|
|
||||||
Then, you can run gitforgefs as follows:
|
### Getting an API token
|
||||||
``` sh
|
|
||||||
gitforgefs -config config.yaml /path/to/mountpoint
|
|
||||||
```
|
|
||||||
|
|
||||||
Stopping gitforgefs will unmount the filesystem. In the event the mountpoint is stuck in a bad state (eg: due to receiving a SIGKILL), you may need to manually cleanup using `umount`:
|
To generate an api token, log into your Gitlab instance, and go in your user settings > Access Token. Create a personal access token with the following permissions at the minimum:
|
||||||
|
* `read_user`
|
||||||
|
* `read_api`
|
||||||
|
|
||||||
|
### Getting the group ids
|
||||||
|
|
||||||
|
The group id can be seen just under the name of the group in Gitlab.
|
||||||
|
|
||||||
|
![group-id](media/group_id.jpg)
|
||||||
|
|
||||||
|
### Getting the user ids
|
||||||
|
|
||||||
|
Log into gitlab and go to https://gitlab.com/api/v4/users?username=USERNAME where `USERNAME` is the username of the user you wish to know the id of. The json response will contain the user id.
|
||||||
|
|
||||||
|
See https://forum.gitlab.com/t/where-is-my-user-id-in-gitlab-com/7912
|
||||||
|
|
||||||
|
### Mounting the filesystem
|
||||||
|
|
||||||
|
You can mount the filesystem with the following command:
|
||||||
``` sh
|
``` sh
|
||||||
sudo umount /path/to/mountpoint
|
~/go/bin/gitlabfs -config /path/to/your/config.yaml /path/to/mountpoint
|
||||||
```
|
```
|
||||||
|
Once the filesystem is mounted, you can `cd` into it and navigate it like any other filesystem. The first time `ls` is run the list of groups and projects is fetched from Gitlab. This operation can take a few seconds and the command will appear frozen until it's completed. Subsequent `ls` will fetch from the cache and should be much faster.
|
||||||
|
|
||||||
|
If `on_clone` is set to `init` or `no-checkout`, the locally cloned project will appear empty. Simply running `git pull` manually in the project folder will sync it up with Gitlab.
|
||||||
|
|
||||||
|
### Unmounting the filesystem
|
||||||
|
|
||||||
|
To stop the filesystem, use the command `umount /path/to/mountpoint` to cleanly unmount the filesystem.
|
||||||
|
|
||||||
|
If `gitlabfs` is not cleanly stopped, you might start seeing the error "transport endpoint is not connected" when trying to access the mountpoint, even preventing from mounting back the filesystem on the same mountpoint. To fix this, use `umount` as root user, eg: `sudo umount /path/to/mountpoint`.
|
||||||
|
|
||||||
### Running automatically on user login
|
### Running automatically on user login
|
||||||
|
|
||||||
See [./contrib/systemd](contrib/systemd) for instructions on how to configure a systemd service to automatically run gitforgefs on user login.
|
See [./contrib/systemd](contrib/systemd) for instructions on how to configure a systemd service to automatically run gitlabfs on user login.
|
||||||
|
|
||||||
## Caching
|
## Caching
|
||||||
|
|
||||||
### Filesystem cache
|
To reduce the number of calls to the Gitlab api and improve the responsiveness of the filesystem, `gitlabfs` will cache the content of the group in memory. If a group or project is renamed, created or deleted from Gitlab, these change will not appear in the filesystem. To force `gitlabfs` to refresh its cache, use `touch .refresh` in the folder to refresh to force `gitlabfs` to query Gitlab for the list of groups and projects again.
|
||||||
|
|
||||||
To reduce the number of calls to the APIs and improve the responsiveness of the filesystem, gitforgefs will cache the content of the forge in memory. If a group or project is renamed, created or deleted from the forge, these change will not appear in the filesystem immediately. To force gitforgefs to refresh its cache, use `touch .refresh` in the folder to signal gitforgefs to refresh this folder.
|
While the filesystem lives in memory, the git repositories that are cloned are saved on disk. By default, they are saved in `$XDG_DATA_HOME/gitlabfs` or `$HOME/.local/share/gitlabfs`, if `$XDG_DATA_HOME` is unset. `gitlabfs` symlink to the local clone of that repo. The local clone is unaffected by project rename or archive/unarchive in Gitlab and a given project will always point to the correct local folder.
|
||||||
|
|
||||||
### Local repository cache
|
## Known issues / Future improvements
|
||||||
|
* Cache persists forever until a manual refresh is requested. Some way to automatically refresh would be nice.
|
||||||
|
* The filesystem is currently read-only. Implementing `mkdir` to create groups, `ln` or `touch` to create projects, etc. would be nice.
|
||||||
|
* Code need some cleanup and could maybe be optimized here and there.
|
||||||
|
|
||||||
While the filesystem lives in memory, the git repositories that are cloned are saved on disk. By default, they are saved in `$XDG_DATA_HOME/gitforgefs` or `$HOME/.local/share/gitforgefs`, if `$XDG_DATA_HOME` is unset. `gitforgefs` symlink to the local clone of that repo. The local clone is unaffected by project rename or archive/unarchive in Gitlab and a given project will always point to the correct local folder.
|
## Building
|
||||||
|
|
||||||
## Future improvements
|
|
||||||
* Cache persists forever until a manual refresh is requested. Some way to automatically refresh after a timeout would be nice.
|
|
||||||
|
|
||||||
## Building from the repo
|
|
||||||
|
|
||||||
Simply use `make` to create the executable. The executable will be in `bin/`.
|
Simply use `make` to create the executable. The executable will be in `bin/`.
|
||||||
|
|
||||||
|
|
|
@ -28,8 +28,8 @@ gitlab:
|
||||||
group_ids:
|
group_ids:
|
||||||
- 9970 # gitlab-org
|
- 9970 # gitlab-org
|
||||||
|
|
||||||
# A list of the name of the user to expose their repositories un the filesystem
|
# A list of the user ids to expose their personal projects in the filesystem.
|
||||||
user_names: []
|
user_ids: []
|
||||||
|
|
||||||
# Set how archived projects are handled.
|
# Set how archived projects are handled.
|
||||||
# If set to "show", it will add them to the filesystem and treat them like any other project
|
# If set to "show", it will add them to the filesystem and treat them like any other project
|
||||||
|
@ -102,7 +102,7 @@ gitea:
|
||||||
|
|
||||||
git:
|
git:
|
||||||
# Path to the local repository cache. Repositories in the filesystem will symlink to a folder in this path.
|
# Path to the local repository cache. Repositories in the filesystem will symlink to a folder in this path.
|
||||||
# Default to $XDG_DATA_HOME/gitforgefs, or $HOME/.local/share/gitforgefs if the environment variable $XDG_DATA_HOME is unset.
|
# Default to $XDG_DATA_HOME/gitlabfs, or $HOME/.local/share/gitlabfs if the environment variable $XDG_DATA_HOME is unset.
|
||||||
#clone_location:
|
#clone_location:
|
||||||
|
|
||||||
# The name of the remote in the local clone.
|
# The name of the remote in the local clone.
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
fs:
|
fs:
|
||||||
mountpoint: /tmp/gitforgefs/test/mnt/gitlab
|
mountpoint: /tmp/gitlabfs/test/mnt/gitlab
|
||||||
mountoptions: nodev
|
mountoptions: nodev
|
||||||
forge: gitlab
|
forge: gitlab
|
||||||
|
|
||||||
|
@ -9,8 +9,8 @@ gitlab:
|
||||||
pull_method: ssh
|
pull_method: ssh
|
||||||
group_ids:
|
group_ids:
|
||||||
- 123
|
- 123
|
||||||
user_names:
|
user_ids:
|
||||||
- test-user
|
- 456
|
||||||
archived_project_handling: hide
|
archived_project_handling: hide
|
||||||
include_current_user: true
|
include_current_user: true
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ gitea:
|
||||||
include_current_user: true
|
include_current_user: true
|
||||||
|
|
||||||
git:
|
git:
|
||||||
clone_location: /tmp/gitforgefs/test/cache/gitlab
|
clone_location: /tmp/gitlabfs/test/cache/gitlab
|
||||||
remote: origin
|
remote: origin
|
||||||
on_clone: clone
|
on_clone: clone
|
||||||
auto_pull: false
|
auto_pull: false
|
||||||
|
|
|
@ -39,7 +39,7 @@ type (
|
||||||
Token string `yaml:"token,omitempty"`
|
Token string `yaml:"token,omitempty"`
|
||||||
|
|
||||||
GroupIDs []int `yaml:"group_ids,omitempty"`
|
GroupIDs []int `yaml:"group_ids,omitempty"`
|
||||||
UserNames []string `yaml:"user_ids,omitempty"`
|
UserIDs []int `yaml:"user_ids,omitempty"`
|
||||||
|
|
||||||
ArchivedProjectHandling string `yaml:"archived_project_handling,omitempty"`
|
ArchivedProjectHandling string `yaml:"archived_project_handling,omitempty"`
|
||||||
IncludeCurrentUser bool `yaml:"include_current_user,omitempty"`
|
IncludeCurrentUser bool `yaml:"include_current_user,omitempty"`
|
||||||
|
@ -83,7 +83,7 @@ func LoadConfig(configPath string) (*Config, error) {
|
||||||
if dataHome == "" {
|
if dataHome == "" {
|
||||||
dataHome = filepath.Join(os.Getenv("HOME"), ".local/share")
|
dataHome = filepath.Join(os.Getenv("HOME"), ".local/share")
|
||||||
}
|
}
|
||||||
defaultCloneLocation := filepath.Join(dataHome, "gitforgefs")
|
defaultCloneLocation := filepath.Join(dataHome, "gitlabfs")
|
||||||
|
|
||||||
config := &Config{
|
config := &Config{
|
||||||
FS: FSConfig{
|
FS: FSConfig{
|
||||||
|
@ -96,7 +96,7 @@ func LoadConfig(configPath string) (*Config, error) {
|
||||||
Token: "",
|
Token: "",
|
||||||
PullMethod: "http",
|
PullMethod: "http",
|
||||||
GroupIDs: []int{9970},
|
GroupIDs: []int{9970},
|
||||||
UserNames: []string{},
|
UserIDs: []int{},
|
||||||
ArchivedProjectHandling: "hide",
|
ArchivedProjectHandling: "hide",
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
|
@ -119,6 +119,7 @@ func LoadConfig(configPath string) (*Config, error) {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if configPath != "" {
|
||||||
f, err := os.Open(configPath)
|
f, err := os.Open(configPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to open config file: %v", err)
|
return nil, fmt.Errorf("failed to open config file: %v", err)
|
||||||
|
@ -129,6 +130,7 @@ func LoadConfig(configPath string) (*Config, error) {
|
||||||
if err := d.Decode(config); err != nil {
|
if err := d.Decode(config); err != nil {
|
||||||
return nil, fmt.Errorf("failed to parse config file: %v", err)
|
return nil, fmt.Errorf("failed to parse config file: %v", err)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// validate forge is set
|
// validate forge is set
|
||||||
if config.FS.Forge != ForgeGithub && config.FS.Forge != ForgeGitlab && config.FS.Forge != ForgeGitea {
|
if config.FS.Forge != ForgeGithub && config.FS.Forge != ForgeGitlab && config.FS.Forge != ForgeGitea {
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestLoadConfig(t *testing.T) {
|
func TestLoadConfig(t *testing.T) {
|
||||||
|
@ -16,7 +16,7 @@ func TestLoadConfig(t *testing.T) {
|
||||||
input: "config.test.yaml",
|
input: "config.test.yaml",
|
||||||
expected: &config.Config{
|
expected: &config.Config{
|
||||||
FS: config.FSConfig{
|
FS: config.FSConfig{
|
||||||
Mountpoint: "/tmp/gitforgefs/test/mnt/gitlab",
|
Mountpoint: "/tmp/gitlabfs/test/mnt/gitlab",
|
||||||
MountOptions: "nodev",
|
MountOptions: "nodev",
|
||||||
Forge: "gitlab",
|
Forge: "gitlab",
|
||||||
},
|
},
|
||||||
|
@ -25,7 +25,7 @@ func TestLoadConfig(t *testing.T) {
|
||||||
Token: "12345",
|
Token: "12345",
|
||||||
PullMethod: "ssh",
|
PullMethod: "ssh",
|
||||||
GroupIDs: []int{123},
|
GroupIDs: []int{123},
|
||||||
UserNames: []int{456},
|
UserIDs: []int{456},
|
||||||
ArchivedProjectHandling: "hide",
|
ArchivedProjectHandling: "hide",
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
|
@ -47,7 +47,7 @@ func TestLoadConfig(t *testing.T) {
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
Git: config.GitClientConfig{
|
Git: config.GitClientConfig{
|
||||||
CloneLocation: "/tmp/gitforgefs/test/cache/gitlab",
|
CloneLocation: "/tmp/gitlabfs/test/cache/gitlab",
|
||||||
Remote: "origin",
|
Remote: "origin",
|
||||||
OnClone: "clone",
|
OnClone: "clone",
|
||||||
AutoPull: false,
|
AutoPull: false,
|
||||||
|
@ -148,7 +148,7 @@ func TestMakeGitlabConfig(t *testing.T) {
|
||||||
PullMethod: "http",
|
PullMethod: "http",
|
||||||
Token: "",
|
Token: "",
|
||||||
GroupIDs: []int{9970},
|
GroupIDs: []int{9970},
|
||||||
UserNames: []int{},
|
UserIDs: []int{},
|
||||||
ArchivedProjectHandling: "hide",
|
ArchivedProjectHandling: "hide",
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
|
@ -158,7 +158,7 @@ func TestMakeGitlabConfig(t *testing.T) {
|
||||||
PullMethod: "http",
|
PullMethod: "http",
|
||||||
Token: "",
|
Token: "",
|
||||||
GroupIDs: []int{9970},
|
GroupIDs: []int{9970},
|
||||||
UserNames: []int{},
|
UserIDs: []int{},
|
||||||
ArchivedProjectHandling: "hide",
|
ArchivedProjectHandling: "hide",
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
|
@ -173,7 +173,7 @@ func TestMakeGitlabConfig(t *testing.T) {
|
||||||
PullMethod: "invalid",
|
PullMethod: "invalid",
|
||||||
Token: "",
|
Token: "",
|
||||||
GroupIDs: []int{9970},
|
GroupIDs: []int{9970},
|
||||||
UserNames: []int{},
|
UserIDs: []int{},
|
||||||
ArchivedProjectHandling: "hide",
|
ArchivedProjectHandling: "hide",
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
},
|
},
|
||||||
|
@ -187,7 +187,7 @@ func TestMakeGitlabConfig(t *testing.T) {
|
||||||
PullMethod: "http",
|
PullMethod: "http",
|
||||||
Token: "",
|
Token: "",
|
||||||
GroupIDs: []int{9970},
|
GroupIDs: []int{9970},
|
||||||
UserNames: []int{},
|
UserIDs: []int{},
|
||||||
IncludeCurrentUser: true,
|
IncludeCurrentUser: true,
|
||||||
ArchivedProjectHandling: "invalid",
|
ArchivedProjectHandling: "invalid",
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,15 +1,12 @@
|
||||||
This unit file allows you to automatically start gitforgefs as a systemd unit.
|
This unit file allows you to automatically start gitlabfs as a systemd unit.
|
||||||
|
|
||||||
## Setup
|
## Install
|
||||||
1. Install gitforgefs
|
1. Install gitlabfs using `go get`
|
||||||
2. Copy **gitforgefs@.service** into **$HOME/.config/systemd/user**. Create the folder if it does not exists.
|
2. Run `which gitlabfs` to verify that gitlabfs is present in your PATH. if the command fail, you may need to add **$HOME/go/bin** to your PATH.
|
||||||
``` sh
|
3. Copy **gitlabfs@.service** into **~/.config/systemd/user**. Create the folder if it does not exists.
|
||||||
mkdir -p $HOME/.config/systemd/user
|
4. Reload systemd: `systemctl --user daemon-reload`
|
||||||
curl -o $HOME/.config/systemd/user/gitforgefs@.service https://raw.githubusercontent.com/badjware/gitlabfs/dev/contrib/systemd/gitforgefs%40.service
|
|
||||||
```
|
|
||||||
3. Reload systemd: `systemctl --user daemon-reload`
|
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
1. Create your gitforgefs config file in **$HOME/.config/gitforgefs** eg: **$HOME/.config/gitforgefs/gitlab.com.yaml**. Make sure the config file name ends with **.yaml** and a mountpoint is configured in the file.
|
1. Create your gitlabfs config file in **~/.config/gitlabfs** eg: **~/.config/gitlabfs/gitlab.com.yaml**. Make sure the config file name ends with **.yaml** and a mountpoint is configured in the file.
|
||||||
2. Start your service with `systemctl --user start gitforgefs@<name of your config>.service`. eg: `systemctl --user start gitforgefs@gitlab.com.service`. Omit the **.yaml** extension.
|
2. Start your service with `systemctl --user start gitlabfs@<name of your config>.service`. eg: `systemctl --user start gitlabfs@gitlab.com.service`. Omit the **.yaml** in the name of the service.
|
||||||
3. Enable your service to start on login with `systemctl --user enable gitforgefs@<name of your config>.service`. eg: `systemctl --user enable gitforgefs@gitlab.com.service`
|
3. Enable your service start on login with `systemctl --user enable gitlabfs@<name of your config>.service`. eg: `systemctl --user enable gitlabfs@gitlab.com.service`
|
||||||
|
|
|
@ -4,7 +4,7 @@ Wants=network-online.target
|
||||||
After=network-online.target
|
After=network-online.target
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
ExecStart=%h/go/bin/gitforgefs -config %E/gitforgefs/%i.yaml
|
ExecStart=%h/go/bin/gitlabfs -config %E/gitlabfs/%i.yaml
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=default.target
|
WantedBy=default.target
|
|
@ -6,8 +6,8 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"code.gitea.io/sdk/gitea"
|
"code.gitea.io/sdk/gitea"
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
)
|
)
|
||||||
|
|
||||||
type giteaClient struct {
|
type giteaClient struct {
|
||||||
|
|
|
@ -5,7 +5,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"code.gitea.io/sdk/gitea"
|
"code.gitea.io/sdk/gitea"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Organization struct {
|
type Organization struct {
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
"code.gitea.io/sdk/gitea"
|
"code.gitea.io/sdk/gitea"
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Repository struct {
|
type Repository struct {
|
||||||
|
|
|
@ -5,7 +5,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"code.gitea.io/sdk/gitea"
|
"code.gitea.io/sdk/gitea"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
)
|
)
|
||||||
|
|
||||||
type User struct {
|
type User struct {
|
||||||
|
|
|
@ -6,8 +6,8 @@ import (
|
||||||
"log/slog"
|
"log/slog"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/google/go-github/v63/github"
|
"github.com/google/go-github/v63/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/google/go-github/v63/github"
|
"github.com/google/go-github/v63/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ package github
|
||||||
import (
|
import (
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/google/go-github/v63/github"
|
"github.com/google/go-github/v63/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/google/go-github/v63/github"
|
"github.com/google/go-github/v63/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -6,8 +6,8 @@ import (
|
||||||
"slices"
|
"slices"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/xanzy/go-gitlab"
|
"github.com/xanzy/go-gitlab"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -19,8 +19,6 @@ type gitlabClient struct {
|
||||||
|
|
||||||
rootContent map[string]fstree.GroupSource
|
rootContent map[string]fstree.GroupSource
|
||||||
|
|
||||||
userIDs []int
|
|
||||||
|
|
||||||
// API response cache
|
// API response cache
|
||||||
groupCacheMux sync.RWMutex
|
groupCacheMux sync.RWMutex
|
||||||
groupCache map[int]*Group
|
groupCache map[int]*Group
|
||||||
|
@ -45,8 +43,6 @@ func NewClient(logger *slog.Logger, config config.GitlabClientConfig) (*gitlabCl
|
||||||
|
|
||||||
rootContent: nil,
|
rootContent: nil,
|
||||||
|
|
||||||
userIDs: []int{},
|
|
||||||
|
|
||||||
groupCache: map[int]*Group{},
|
groupCache: map[int]*Group{},
|
||||||
userCache: map[int]*User{},
|
userCache: map[int]*User{},
|
||||||
}
|
}
|
||||||
|
@ -56,17 +52,7 @@ func NewClient(logger *slog.Logger, config config.GitlabClientConfig) (*gitlabCl
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warn("failed to fetch the current user:", "error", err.Error())
|
logger.Warn("failed to fetch the current user:", "error", err.Error())
|
||||||
} else {
|
} else {
|
||||||
gitlabClient.userIDs = append(gitlabClient.userIDs, currentUser.ID)
|
gitlabClient.UserIDs = append(gitlabClient.UserIDs, currentUser.ID)
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch the configured users and add them to the list
|
|
||||||
for _, userName := range config.UserNames {
|
|
||||||
user, _, err := client.Users.ListUsers(&gitlab.ListUsersOptions{Username: &userName})
|
|
||||||
if err != nil || len(user) != 1 {
|
|
||||||
logger.Warn("failed to fetch the user", "userName", userName, "error", err.Error())
|
|
||||||
} else {
|
|
||||||
gitlabClient.userIDs = append(gitlabClient.userIDs, user[0].ID)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return gitlabClient, nil
|
return gitlabClient, nil
|
||||||
|
@ -86,7 +72,7 @@ func (c *gitlabClient) FetchRootGroupContent() (map[string]fstree.GroupSource, e
|
||||||
rootGroupCache[group.Name] = group
|
rootGroupCache[group.Name] = group
|
||||||
}
|
}
|
||||||
// fetch users
|
// fetch users
|
||||||
for _, uid := range c.userIDs {
|
for _, uid := range c.UserIDs {
|
||||||
user, err := c.fetchUser(uid)
|
user, err := c.fetchUser(uid)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -100,7 +86,7 @@ func (c *gitlabClient) FetchRootGroupContent() (map[string]fstree.GroupSource, e
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *gitlabClient) FetchGroupContent(gid uint64) (map[string]fstree.GroupSource, map[string]fstree.RepositorySource, error) {
|
func (c *gitlabClient) FetchGroupContent(gid uint64) (map[string]fstree.GroupSource, map[string]fstree.RepositorySource, error) {
|
||||||
if slices.Contains[[]int, int](c.userIDs, int(gid)) {
|
if slices.Contains[[]int, int](c.UserIDs, int(gid)) {
|
||||||
// gid is a user
|
// gid is a user
|
||||||
user, err := c.fetchUser(int(gid))
|
user, err := c.fetchUser(int(gid))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/xanzy/go-gitlab"
|
"github.com/xanzy/go-gitlab"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ package gitlab
|
||||||
import (
|
import (
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/xanzy/go-gitlab"
|
"github.com/xanzy/go-gitlab"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/xanzy/go-gitlab"
|
"github.com/xanzy/go-gitlab"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -10,9 +10,9 @@ import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/badjware/gitforgefs/utils"
|
"github.com/badjware/gitlabfs/utils"
|
||||||
"github.com/vmihailenco/taskq/v3"
|
"github.com/vmihailenco/taskq/v3"
|
||||||
"github.com/vmihailenco/taskq/v3/memqueue"
|
"github.com/vmihailenco/taskq/v3/memqueue"
|
||||||
)
|
)
|
||||||
|
@ -41,7 +41,7 @@ func NewClient(logger *slog.Logger, p config.GitClientConfig) (*gitClient, error
|
||||||
|
|
||||||
logger: logger,
|
logger: logger,
|
||||||
|
|
||||||
hostnameProg: regexp.MustCompile(`([a-z0-1\-]+\.)+[a-z0-1\-]+`),
|
hostnameProg: regexp.MustCompile(`([a-z0-1:\-]+\.)+[a-z0-1:\-]+`),
|
||||||
|
|
||||||
queue: queueFactory.RegisterQueue(&taskq.QueueOptions{
|
queue: queueFactory.RegisterQueue(&taskq.QueueOptions{
|
||||||
Name: "git-queue",
|
Name: "git-queue",
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/utils"
|
"github.com/badjware/gitlabfs/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *gitClient) clone(url string, defaultBranch string, dst string) error {
|
func (c *gitClient) clone(url string, defaultBranch string, dst string) error {
|
||||||
|
|
|
@ -4,7 +4,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/utils"
|
"github.com/badjware/gitlabfs/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *gitClient) pull(repoPath string, defaultBranch string) error {
|
func (c *gitClient) pull(repoPath string, defaultBranch string) error {
|
||||||
|
|
2
go.mod
2
go.mod
|
@ -1,4 +1,4 @@
|
||||||
module github.com/badjware/gitforgefs
|
module github.com/badjware/gitlabfs
|
||||||
|
|
||||||
go 1.21
|
go 1.21
|
||||||
|
|
||||||
|
|
14
main.go
14
main.go
|
@ -7,16 +7,16 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/badjware/gitforgefs/config"
|
"github.com/badjware/gitlabfs/config"
|
||||||
"github.com/badjware/gitforgefs/forges/gitea"
|
"github.com/badjware/gitlabfs/forges/gitea"
|
||||||
"github.com/badjware/gitforgefs/forges/github"
|
"github.com/badjware/gitlabfs/forges/github"
|
||||||
"github.com/badjware/gitforgefs/forges/gitlab"
|
"github.com/badjware/gitlabfs/forges/gitlab"
|
||||||
"github.com/badjware/gitforgefs/fstree"
|
"github.com/badjware/gitlabfs/fstree"
|
||||||
"github.com/badjware/gitforgefs/git"
|
"github.com/badjware/gitlabfs/git"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
configPath := flag.String("config", "config.yaml", "The config file")
|
configPath := flag.String("config", "", "The config file")
|
||||||
mountoptionsFlag := flag.String("o", "", "Filesystem mount options. See mount.fuse(8)")
|
mountoptionsFlag := flag.String("o", "", "Filesystem mount options. See mount.fuse(8)")
|
||||||
debug := flag.Bool("debug", false, "Enable debug logging")
|
debug := flag.Bool("debug", false, "Enable debug logging")
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue