Skip to content

Commit 1cf7c28

Browse files
authored
Cloner 6 (#9)
some performance bump
1 parent fbd8662 commit 1cf7c28

File tree

7 files changed

+63
-29
lines changed

7 files changed

+63
-29
lines changed

Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ cover:
2525

2626
build:
2727
$(MAKE) fmt
28-
go build -o app ./cmd
28+
export CGO_ENABLED=0 && go build -o app ./cmd
2929

3030
run:
3131
go run ./cmd

README.md

+11
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,17 @@ export PATH="$PATH:/usr/local/go/bin:$HOME/go/bin"
2323
make setup
2424
```
2525

26+
## Run
27+
28+
Edit **config.yml** to specify the settings. Possible options:
29+
30+
```yaml
31+
Debug bool `yaml:"debug" env:"DEBUG"` # additional logging messages
32+
User string `yaml:"user" env:"USER" env-default:"user"` # github username (which likes should be processed)
33+
Dir string `yaml:"dir" env:"DIR" env-default:"."` # local directory where repos should be cloned
34+
Token string `yaml:"token" env:"TOKEN"` # github PAT
35+
```
36+
2637
## Project layout
2738

2839
Directory names and meanings

cmd/main.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,9 @@ import (
1010

1111
"github.com/rs/zerolog/log"
1212

13-
"giclo/internal/adapters/config"
14-
"giclo/internal/application"
15-
"giclo/internal/domain/errors"
13+
"github.com/devalv/giclo/internal/adapters/config"
14+
"github.com/devalv/giclo/internal/application"
15+
"github.com/devalv/giclo/internal/domain/errors"
1616
)
1717

1818
func ParseFlags() (cfgPath string, err error) {

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
module giclo
1+
module github.com/devalv/giclo
22

33
go 1.21
44

internal/adapters/config/config.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ package config
33
import (
44
"github.com/ilyakaznacheev/cleanenv"
55

6-
"giclo/internal/domain/models"
6+
"github.com/devalv/giclo/internal/domain/models"
77
)
88

99
func NewConfig(cfgPath string) (*models.Config, error) {

internal/application/app.go

+42-18
Original file line numberDiff line numberDiff line change
@@ -11,13 +11,14 @@ import (
1111
"regexp"
1212
"strconv"
1313
"strings"
14+
"sync"
1415
"time"
1516

1617
"github.com/go-git/go-git/v5"
1718
"github.com/rs/zerolog/log"
1819

19-
"giclo/internal/domain/errors"
20-
"giclo/internal/domain/models"
20+
"github.com/devalv/giclo/internal/domain/errors"
21+
"github.com/devalv/giclo/internal/domain/models"
2122
)
2223

2324
type Application struct {
@@ -98,7 +99,7 @@ func getRepos(resp *http.Response) (*[]models.GithubAPIRepoResponse, error) {
9899

99100
// get pages count liked by a user
100101
func getTotalPages(ctx context.Context, cfg *models.Config) (int, error) {
101-
resp, err := getAPIResponse(ctx, cfg, 1, 10)
102+
resp, err := getAPIResponse(ctx, cfg, 1, 50)
102103
if err != nil {
103104
return 0, err
104105
}
@@ -144,8 +145,8 @@ func getLikedRepos(ctx context.Context, reposPath string, cfg *models.Config) (*
144145
var reposToClone []models.ReposToClone
145146

146147
for i := 1; i <= totalPages; i++ {
147-
// TODO: goroutines
148-
resp, err := getAPIResponse(ctx, cfg, i, 10)
148+
// чтобы не нарваться на бан - получаем список последовательно
149+
resp, err := getAPIResponse(ctx, cfg, i, 50)
149150
if err != nil {
150151
log.Warn().Err(err)
151152
continue
@@ -171,13 +172,45 @@ func getLikedRepos(ctx context.Context, reposPath string, cfg *models.Config) (*
171172
}
172173

173174
// clone repo to a local fs
174-
func cloneRepo(repoURL, dirPath string) error {
175+
func cloneRepo(wg *sync.WaitGroup, isDebug bool, repoURL, dirPath string) {
176+
defer wg.Done()
177+
defer log.Info().Msgf("Клонирован %s", repoURL)
178+
179+
if isDebug {
180+
log.Debug().Msgf("Собираемся клонировать %s в %s", repoURL, dirPath)
181+
}
182+
175183
_, err := git.PlainClone(dirPath, false, &git.CloneOptions{
176184
URL: repoURL,
177-
Progress: os.Stdout,
185+
Progress: nil,
178186
})
187+
if err != nil {
188+
log.Warn().Err(err)
189+
}
190+
}
191+
192+
// clone repos concurrently to a local fs with a WaitGroup
193+
func cloneRepos(isDebug bool, likedRepos *[]models.ReposToClone) {
194+
// TODO: таймаут на клонирование каждого отдельного репозитория
195+
196+
var lastEl int
197+
for i := 0; i < len(*likedRepos); i += 5 {
198+
tmpRepos := *likedRepos
199+
var waitGroup sync.WaitGroup
200+
201+
if i+5 <= len(*likedRepos) {
202+
lastEl = i + 5
203+
} else {
204+
lastEl = len(*likedRepos)
205+
}
206+
207+
for _, repo := range tmpRepos[i:lastEl] {
208+
waitGroup.Add(1)
209+
go cloneRepo(&waitGroup, isDebug, repo.CloneURL, repo.CloneDir)
210+
}
179211

180-
return err
212+
waitGroup.Wait()
213+
}
181214
}
182215

183216
func (app *Application) Start(ctx context.Context) {
@@ -196,16 +229,7 @@ func (app *Application) Start(ctx context.Context) {
196229
log.Fatal().Err(err).Msgf(errors.APILikedResponseError, err)
197230
}
198231

199-
// TODO: явно нужна горутина
200-
for _, repo := range *likedRepos {
201-
if app.cfg.Debug {
202-
log.Debug().Msgf("Собираемся клонировать %s в %s", repo.CloneURL, repo.CloneDir)
203-
}
204-
err := cloneRepo(repo.CloneURL, repo.CloneDir)
205-
if err != nil {
206-
log.Warn().Err(err)
207-
}
208-
}
232+
cloneRepos(app.cfg.Debug, likedRepos)
209233

210234
app.Stop(ctx)
211235
}

internal/domain/models/models.go

+4-5
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,10 @@ import (
88
)
99

1010
type Config struct {
11-
Debug bool `yaml:"debug" env:"DEBUG"`
12-
User string `yaml:"user" env:"USER" env-default:"user"`
13-
Dir string `yaml:"dir" env:"DIR" env-default:"."`
14-
Token string `yaml:"token" env:"TOKEN"`
15-
Compress bool `yaml:"compress" env:"COMPRESS"`
11+
Debug bool `yaml:"debug" env:"DEBUG"`
12+
User string `yaml:"user" env:"USER" env-default:"user"`
13+
Dir string `yaml:"dir" env:"DIR" env-default:"."`
14+
Token string `yaml:"token" env:"TOKEN"`
1615
}
1716

1817
func (cfg *Config) Check() error {

0 commit comments

Comments
 (0)