Compare commits
2 Commits
ed7df18f83
...
bbf96498aa
| Author | SHA1 | Date | |
|---|---|---|---|
| bbf96498aa | |||
| 954966db58 |
+71
-3
@@ -13,7 +13,7 @@ import (
|
|||||||
|
|
||||||
var log = logging.MustGetLogger("cursorius-server")
|
var log = logging.MustGetLogger("cursorius-server")
|
||||||
|
|
||||||
func createSchema(db database.Database) (graphql.Schema, error) {
|
func createSchema(db database.Database, pollChan chan uuid.UUID) (graphql.Schema, error) {
|
||||||
runnerType := graphql.NewObject(graphql.ObjectConfig{
|
runnerType := graphql.NewObject(graphql.ObjectConfig{
|
||||||
Name: "Runner",
|
Name: "Runner",
|
||||||
Description: "A runner available for use inside of a pipeline.",
|
Description: "A runner available for use inside of a pipeline.",
|
||||||
@@ -454,6 +454,8 @@ func createSchema(db database.Database) (graphql.Schema, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pollChan <- pipeline.Id
|
||||||
|
|
||||||
return pipeline, nil
|
return pipeline, nil
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -558,6 +560,72 @@ func createSchema(db database.Database) (graphql.Schema, error) {
|
|||||||
return runner, nil
|
return runner, nil
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"updatePipeline": &graphql.Field{
|
||||||
|
Type: pipelineType,
|
||||||
|
Description: "Create a new pipeline",
|
||||||
|
Args: graphql.FieldConfigArgument{
|
||||||
|
"pipelineId": &graphql.ArgumentConfig{
|
||||||
|
Type: graphql.NewNonNull(graphql.String),
|
||||||
|
},
|
||||||
|
"name": &graphql.ArgumentConfig{
|
||||||
|
Type: graphql.String,
|
||||||
|
},
|
||||||
|
"url": &graphql.ArgumentConfig{
|
||||||
|
Type: graphql.String,
|
||||||
|
},
|
||||||
|
"pollInterval": &graphql.ArgumentConfig{
|
||||||
|
Type: graphql.Int,
|
||||||
|
},
|
||||||
|
"cloneCredentialId": &graphql.ArgumentConfig{
|
||||||
|
Type: graphql.String,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Resolve: func(params graphql.ResolveParams) (interface{}, error) {
|
||||||
|
pipelineId, err := uuid.Parse(params.Args["pipelineId"].(string))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var name *string
|
||||||
|
var url *string
|
||||||
|
var interval *int
|
||||||
|
|
||||||
|
if nameVal, ok := params.Args["name"]; ok {
|
||||||
|
nameVal := nameVal.(string)
|
||||||
|
name = &nameVal
|
||||||
|
} else {
|
||||||
|
name = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if urlVal, ok := params.Args["url"]; ok {
|
||||||
|
urlVal := urlVal.(string)
|
||||||
|
url = &urlVal
|
||||||
|
} else {
|
||||||
|
url = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if intervalVal, ok := params.Args["pollInterval"]; ok {
|
||||||
|
intervalVal := intervalVal.(int)
|
||||||
|
interval = &intervalVal
|
||||||
|
} else {
|
||||||
|
interval = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pipeline, err := db.UpdatePipeline(
|
||||||
|
pipelineId,
|
||||||
|
name,
|
||||||
|
url,
|
||||||
|
interval,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
pollChan <- pipeline.Id
|
||||||
|
|
||||||
|
return pipeline, nil
|
||||||
|
},
|
||||||
|
},
|
||||||
"setPipelineCloneCredential": &graphql.Field{
|
"setPipelineCloneCredential": &graphql.Field{
|
||||||
Type: pipelineType,
|
Type: pipelineType,
|
||||||
Description: "Set the CloneCredential used by a pipeline to clone the source repo",
|
Description: "Set the CloneCredential used by a pipeline to clone the source repo",
|
||||||
@@ -681,9 +749,9 @@ func createSchema(db database.Database) (graphql.Schema, error) {
|
|||||||
return schema, nil
|
return schema, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func CreateHandler(db database.Database, mux *http.ServeMux) error {
|
func CreateHandler(db database.Database, pollChan chan uuid.UUID, mux *http.ServeMux) error {
|
||||||
|
|
||||||
schema, err := createSchema(db)
|
schema, err := createSchema(db, pollChan)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -77,6 +77,49 @@ RETURNING id, name, url, poll_interval;`
|
|||||||
return pipeline, nil
|
return pipeline, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (db *Database) UpdatePipeline(pipelineId uuid.UUID, name *string, url *string, pollInterval *int) (Pipeline, error) {
|
||||||
|
query := `
|
||||||
|
UPDATE pipelines
|
||||||
|
SET name=$1, url=$2, poll_interval=$3
|
||||||
|
WHERE id=$4
|
||||||
|
RETURNING name, url, poll_interval, clone_credential;`
|
||||||
|
|
||||||
|
pipeline, err := db.GetPipelineById(pipelineId)
|
||||||
|
if err != nil {
|
||||||
|
return pipeline, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var nameNew string
|
||||||
|
var urlNew string
|
||||||
|
var pollIntervalNew int
|
||||||
|
|
||||||
|
if name != nil {
|
||||||
|
nameNew = *name
|
||||||
|
} else {
|
||||||
|
nameNew = pipeline.Name
|
||||||
|
}
|
||||||
|
if url != nil {
|
||||||
|
urlNew = *url
|
||||||
|
} else {
|
||||||
|
urlNew = pipeline.Url
|
||||||
|
}
|
||||||
|
if pollInterval != nil {
|
||||||
|
pollIntervalNew = *pollInterval
|
||||||
|
} else {
|
||||||
|
pollIntervalNew = pipeline.PollInterval
|
||||||
|
}
|
||||||
|
|
||||||
|
err = db.Conn.QueryRow(context.Background(),
|
||||||
|
query, nameNew, urlNew, pollIntervalNew, pipelineId).Scan(
|
||||||
|
&pipeline.Name, &pipeline.Url, &pipeline.PollInterval, &pipeline.CloneCredential,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return pipeline, fmt.Errorf("Could not add credential to pipeline: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pipeline, err
|
||||||
|
}
|
||||||
|
|
||||||
func (db *Database) SetPipelineCloneCredential(pipelineId uuid.UUID, credentialId *uuid.UUID) (Pipeline, error) {
|
func (db *Database) SetPipelineCloneCredential(pipelineId uuid.UUID, credentialId *uuid.UUID) (Pipeline, error) {
|
||||||
query := `
|
query := `
|
||||||
UPDATE pipelines
|
UPDATE pipelines
|
||||||
|
|||||||
+6
-1
@@ -10,6 +10,8 @@ import (
|
|||||||
"git.ohea.xyz/cursorius/server/pipeline_api"
|
"git.ohea.xyz/cursorius/server/pipeline_api"
|
||||||
"git.ohea.xyz/cursorius/server/runnermanager"
|
"git.ohea.xyz/cursorius/server/runnermanager"
|
||||||
"git.ohea.xyz/cursorius/server/webhook"
|
"git.ohea.xyz/cursorius/server/webhook"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
"github.com/op/go-logging"
|
"github.com/op/go-logging"
|
||||||
"golang.org/x/net/http2"
|
"golang.org/x/net/http2"
|
||||||
"golang.org/x/net/http2/h2c"
|
"golang.org/x/net/http2/h2c"
|
||||||
@@ -23,13 +25,14 @@ func setupHTTPServer(
|
|||||||
conf config.PipelineConf,
|
conf config.PipelineConf,
|
||||||
db database.Database,
|
db database.Database,
|
||||||
runnerManagerChans runnermanager.RunnerManagerChans,
|
runnerManagerChans runnermanager.RunnerManagerChans,
|
||||||
|
pollChan chan uuid.UUID,
|
||||||
) error {
|
) error {
|
||||||
|
|
||||||
webhook.CreateWebhookHandler(db, conf, mux)
|
webhook.CreateWebhookHandler(db, conf, mux)
|
||||||
|
|
||||||
pipeline_api.CreateHandler(runnerManagerChans.Allocation, runnerManagerChans.Release, mux)
|
pipeline_api.CreateHandler(runnerManagerChans.Allocation, runnerManagerChans.Release, mux)
|
||||||
|
|
||||||
err := admin_api.CreateHandler(db, mux)
|
err := admin_api.CreateHandler(db, pollChan, mux)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Could not create admin api handler: %w", err)
|
return fmt.Errorf("Could not create admin api handler: %w", err)
|
||||||
}
|
}
|
||||||
@@ -52,6 +55,7 @@ func Listen(
|
|||||||
conf config.PipelineConf,
|
conf config.PipelineConf,
|
||||||
db database.Database,
|
db database.Database,
|
||||||
runnerManagerChans runnermanager.RunnerManagerChans,
|
runnerManagerChans runnermanager.RunnerManagerChans,
|
||||||
|
pollChan chan uuid.UUID,
|
||||||
) error {
|
) error {
|
||||||
|
|
||||||
err := setupHTTPServer(
|
err := setupHTTPServer(
|
||||||
@@ -59,6 +63,7 @@ func Listen(
|
|||||||
conf,
|
conf,
|
||||||
db,
|
db,
|
||||||
runnerManagerChans,
|
runnerManagerChans,
|
||||||
|
pollChan,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Could not setup http endpoints: %w", err)
|
return fmt.Errorf("Could not setup http endpoints: %w", err)
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
_ = poll.StartPolling(configData.Config.PipelineConf, db)
|
pollChan := poll.StartPolling(configData.Config.PipelineConf, db)
|
||||||
|
|
||||||
mux := http.NewServeMux()
|
mux := http.NewServeMux()
|
||||||
|
|
||||||
@@ -57,5 +57,6 @@ func main() {
|
|||||||
configData.Config.PipelineConf,
|
configData.Config.PipelineConf,
|
||||||
db,
|
db,
|
||||||
runnerManagerChans,
|
runnerManagerChans,
|
||||||
|
pollChan,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|||||||
+38
-8
@@ -1,17 +1,18 @@
|
|||||||
package poll
|
package poll
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
|
||||||
"github.com/op/go-logging"
|
|
||||||
|
|
||||||
"git.ohea.xyz/cursorius/server/config"
|
"git.ohea.xyz/cursorius/server/config"
|
||||||
"git.ohea.xyz/cursorius/server/database"
|
"git.ohea.xyz/cursorius/server/database"
|
||||||
"git.ohea.xyz/cursorius/server/pipeline_executor"
|
"git.ohea.xyz/cursorius/server/pipeline_executor"
|
||||||
|
|
||||||
"github.com/go-git/go-git/v5"
|
"github.com/go-git/go-git/v5"
|
||||||
"github.com/go-git/go-git/v5/plumbing"
|
"github.com/go-git/go-git/v5/plumbing"
|
||||||
"github.com/go-git/go-git/v5/storage/memory"
|
"github.com/go-git/go-git/v5/storage/memory"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"github.com/op/go-logging"
|
||||||
)
|
)
|
||||||
|
|
||||||
var log = logging.MustGetLogger("cursorius-server")
|
var log = logging.MustGetLogger("cursorius-server")
|
||||||
@@ -26,13 +27,27 @@ type tag struct {
|
|||||||
commitHash string
|
commitHash string
|
||||||
}
|
}
|
||||||
|
|
||||||
func pollJob(pipeline database.Pipeline, pipelineConf config.PipelineConf, db database.Database) {
|
func pollJob(ctx context.Context, pipeline database.Pipeline, pipelineConf config.PipelineConf, db database.Database) {
|
||||||
firstScan := true
|
firstScan := true
|
||||||
for {
|
for {
|
||||||
// Don't sleep on first scan to ease testing
|
// Don't sleep on first scan to ease testing
|
||||||
// TODO: this should be replaced with a script that mocks a webhook
|
// TODO: this should be replaced with a script that mocks a webhook
|
||||||
if !firstScan {
|
if !firstScan {
|
||||||
time.Sleep(time.Duration(pipeline.PollInterval) * time.Second)
|
|
||||||
|
ctx, cancel := context.WithTimeout(ctx, time.Duration(pipeline.PollInterval)*time.Second)
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
switch ctx.Err() {
|
||||||
|
case context.Canceled:
|
||||||
|
log.Infof("Polling for pipeline %v canceled, stopping", pipeline.Name)
|
||||||
|
cancel()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cancel()
|
||||||
|
|
||||||
log.Infof("Polling repo %v", pipeline.Name)
|
log.Infof("Polling repo %v", pipeline.Name)
|
||||||
} else {
|
} else {
|
||||||
firstScan = false
|
firstScan = false
|
||||||
@@ -136,11 +151,17 @@ func launchPollJobs(conf config.PipelineConf, db database.Database, pollChan cha
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pipelineCancelations := make(map[uuid.UUID]context.CancelFunc)
|
||||||
|
|
||||||
for _, pipeline := range pipelines {
|
for _, pipeline := range pipelines {
|
||||||
if pipeline.PollInterval == 0 {
|
if pipeline.PollInterval == 0 {
|
||||||
continue
|
continue
|
||||||
} else {
|
} else {
|
||||||
go pollJob(pipeline, conf, db)
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
pipelineCancelations[pipeline.Id] = cancel
|
||||||
|
|
||||||
|
log.Infof("Starting polling for pipeline %v with id %v", pipeline.Name, pipeline.Id)
|
||||||
|
go pollJob(ctx, pipeline, conf, db)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -151,8 +172,17 @@ func launchPollJobs(conf config.PipelineConf, db database.Database, pollChan cha
|
|||||||
log.Errorf("Could not get pipeline with id \"%v\" from database: %v", err)
|
log.Errorf("Could not get pipeline with id \"%v\" from database: %v", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// TODO: stop existing polling process for given uuid
|
|
||||||
go pollJob(pipeline, conf, db)
|
// Cancel existing polling job if it exists
|
||||||
|
if cancelFunc, ok := pipelineCancelations[pipeline.Id]; ok {
|
||||||
|
cancelFunc()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start new polling job
|
||||||
|
log.Infof("Starting polling for pipeline %v with id %v", pipeline.Name, pipeline.Id)
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
pipelineCancelations[pipeline.Id] = cancel
|
||||||
|
go pollJob(ctx, pipeline, conf, db)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user