2015-05-26 06:44:47 +03:00
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// repoService is the service that runs in background and periodically
|
|
|
|
// pull from the repository.
|
|
|
|
type repoService struct {
|
2015-05-26 06:44:47 +03:00
|
|
|
repo *Repo
|
2015-05-26 22:18:19 +03:00
|
|
|
ticker *time.Ticker // ticker to tick at intervals
|
2015-05-26 06:44:47 +03:00
|
|
|
running bool // whether service is running.
|
|
|
|
halt chan struct{} // channel to notify service to halt and stop pulling.
|
|
|
|
}
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// Start starts a new background service to pull periodically.
|
2015-05-26 06:44:47 +03:00
|
|
|
func Start(repo *Repo) {
|
2015-05-26 22:18:19 +03:00
|
|
|
service := &repoService{
|
2015-05-26 06:44:47 +03:00
|
|
|
repo,
|
2015-05-26 22:18:19 +03:00
|
|
|
time.NewTicker(repo.Interval),
|
2015-05-26 06:44:47 +03:00
|
|
|
true,
|
|
|
|
make(chan struct{}),
|
|
|
|
}
|
2015-05-26 22:18:19 +03:00
|
|
|
go func(s *repoService) {
|
2015-05-26 06:44:47 +03:00
|
|
|
for {
|
2015-05-26 22:18:19 +03:00
|
|
|
select {
|
|
|
|
case <-s.ticker.C:
|
|
|
|
err := repo.Pull()
|
|
|
|
if err != nil {
|
|
|
|
logger().Println(err)
|
|
|
|
}
|
|
|
|
case <-s.halt:
|
|
|
|
s.ticker.Stop()
|
|
|
|
return
|
2015-05-26 06:44:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}(service)
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// add to services to make it stoppable
|
|
|
|
Services.add(service)
|
2015-05-26 06:44:47 +03:00
|
|
|
}
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// services stores all repoServices
|
|
|
|
type services struct {
|
|
|
|
services []*repoService
|
2015-05-26 06:44:47 +03:00
|
|
|
sync.Mutex
|
|
|
|
}
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// add adds a new service to list of services.
|
|
|
|
func (s *services) add(r *repoService) {
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
2015-05-26 06:44:47 +03:00
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
s.services = append(s.services, r)
|
2015-05-26 06:44:47 +03:00
|
|
|
}
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// Stop stops at most `limit` running services pulling from git repo at
|
|
|
|
// repoURL. It waits until the service is terminated before returning.
|
|
|
|
// If limit is less than zero, it is ignored.
|
2015-05-26 06:44:47 +03:00
|
|
|
// TODO find better ways to identify repos
|
2015-05-26 22:18:19 +03:00
|
|
|
func (s *services) Stop(repoURL string, limit int) {
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
2015-05-26 06:44:47 +03:00
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// locate repos
|
|
|
|
for i, j := 0, 0; i < len(s.services) && ((limit >= 0 && j < limit) || limit < 0); i++ {
|
|
|
|
service := s.services[i]
|
|
|
|
if service.repo.URL == repoURL {
|
2015-05-26 06:44:47 +03:00
|
|
|
// send halt signal
|
2015-05-26 22:18:19 +03:00
|
|
|
service.halt <- struct{}{}
|
|
|
|
s.services[i] = nil
|
2015-05-26 06:44:47 +03:00
|
|
|
j++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-26 22:18:19 +03:00
|
|
|
// remove them from repos list
|
|
|
|
services := s.services[:0]
|
|
|
|
for _, s := range s.services {
|
2015-05-26 06:44:47 +03:00
|
|
|
if s != nil {
|
|
|
|
services = append(services, s)
|
|
|
|
}
|
|
|
|
}
|
2015-05-26 22:18:19 +03:00
|
|
|
s.services = services
|
2015-05-26 06:44:47 +03:00
|
|
|
}
|