@@ -11,13 +11,14 @@ import (
11
11
"regexp"
12
12
"strconv"
13
13
"strings"
14
+ "sync"
14
15
"time"
15
16
16
17
"github.com/go-git/go-git/v5"
17
18
"github.com/rs/zerolog/log"
18
19
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"
21
22
)
22
23
23
24
type Application struct {
@@ -98,7 +99,7 @@ func getRepos(resp *http.Response) (*[]models.GithubAPIRepoResponse, error) {
98
99
99
100
// get pages count liked by a user
100
101
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 )
102
103
if err != nil {
103
104
return 0 , err
104
105
}
@@ -144,8 +145,8 @@ func getLikedRepos(ctx context.Context, reposPath string, cfg *models.Config) (*
144
145
var reposToClone []models.ReposToClone
145
146
146
147
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 )
149
150
if err != nil {
150
151
log .Warn ().Err (err )
151
152
continue
@@ -171,13 +172,45 @@ func getLikedRepos(ctx context.Context, reposPath string, cfg *models.Config) (*
171
172
}
172
173
173
174
// 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
+
175
183
_ , err := git .PlainClone (dirPath , false , & git.CloneOptions {
176
184
URL : repoURL ,
177
- Progress : os . Stdout ,
185
+ Progress : nil ,
178
186
})
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
+ }
179
211
180
- return err
212
+ waitGroup .Wait ()
213
+ }
181
214
}
182
215
183
216
func (app * Application ) Start (ctx context.Context ) {
@@ -196,16 +229,7 @@ func (app *Application) Start(ctx context.Context) {
196
229
log .Fatal ().Err (err ).Msgf (errors .APILikedResponseError , err )
197
230
}
198
231
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 )
209
233
210
234
app .Stop (ctx )
211
235
}
0 commit comments