236 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			236 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
package client
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"net/http"
 | 
						|
	"net/http/cookiejar"
 | 
						|
	"regexp"
 | 
						|
	"strings"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/bndr/gojenkins"
 | 
						|
	"github.com/pkg/errors"
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	errorNotFound = errors.New("404")
 | 
						|
	regex         = regexp.MustCompile("(<application-desc main-class=\"hudson.remoting.jnlp.Main\"><argument>)(?P<secret>[a-z0-9]*)")
 | 
						|
)
 | 
						|
 | 
						|
// Jenkins defines Jenkins API.
 | 
						|
type Jenkins interface {
 | 
						|
	GenerateToken(userName, tokenName string) (*UserToken, error)
 | 
						|
	Info() (*gojenkins.ExecutorResponse, error)
 | 
						|
	SafeRestart() error
 | 
						|
	CreateNode(name string, numExecutors int, description string, remoteFS string, label string, options ...interface{}) (*gojenkins.Node, error)
 | 
						|
	DeleteNode(name string) (bool, error)
 | 
						|
	CreateFolder(name string, parents ...string) (*gojenkins.Folder, error)
 | 
						|
	CreateJobInFolder(config string, jobName string, parentIDs ...string) (*gojenkins.Job, error)
 | 
						|
	CreateJob(config string, options ...interface{}) (*gojenkins.Job, error)
 | 
						|
	CreateOrUpdateJob(config, jobName string) (*gojenkins.Job, bool, error)
 | 
						|
	RenameJob(job string, name string) *gojenkins.Job
 | 
						|
	CopyJob(copyFrom string, newName string) (*gojenkins.Job, error)
 | 
						|
	DeleteJob(name string) (bool, error)
 | 
						|
	BuildJob(name string, options ...interface{}) (int64, error)
 | 
						|
	GetNode(name string) (*gojenkins.Node, error)
 | 
						|
	GetLabel(name string) (*gojenkins.Label, error)
 | 
						|
	GetBuild(jobName string, number int64) (*gojenkins.Build, error)
 | 
						|
	GetJob(id string, parentIDs ...string) (*gojenkins.Job, error)
 | 
						|
	GetSubJob(parentID string, childID string) (*gojenkins.Job, error)
 | 
						|
	GetFolder(id string, parents ...string) (*gojenkins.Folder, error)
 | 
						|
	GetAllNodes() ([]*gojenkins.Node, error)
 | 
						|
	GetAllBuildIds(job string) ([]gojenkins.JobBuild, error)
 | 
						|
	GetAllJobNames() ([]gojenkins.InnerJob, error)
 | 
						|
	GetAllJobs() ([]*gojenkins.Job, error)
 | 
						|
	GetQueue() (*gojenkins.Queue, error)
 | 
						|
	GetQueueUrl() string
 | 
						|
	GetQueueItem(id int64) (*gojenkins.Task, error)
 | 
						|
	GetArtifactData(id string) (*gojenkins.FingerPrintResponse, error)
 | 
						|
	GetPlugins(depth int) (*gojenkins.Plugins, error)
 | 
						|
	UninstallPlugin(name string) error
 | 
						|
	HasPlugin(name string) (*gojenkins.Plugin, error)
 | 
						|
	InstallPlugin(name string, version string) error
 | 
						|
	ValidateFingerPrint(id string) (bool, error)
 | 
						|
	GetView(name string) (*gojenkins.View, error)
 | 
						|
	GetAllViews() ([]*gojenkins.View, error)
 | 
						|
	CreateView(name string, viewType string) (*gojenkins.View, error)
 | 
						|
	Poll() (int, error)
 | 
						|
	ExecuteScript(groovyScript string) (logs string, err error)
 | 
						|
	GetNodeSecret(name string) (string, error)
 | 
						|
}
 | 
						|
 | 
						|
type jenkins struct {
 | 
						|
	gojenkins.Jenkins
 | 
						|
}
 | 
						|
 | 
						|
// JenkinsAPIConnectionSettings is struct that handle information about Jenkins API connection.
 | 
						|
type JenkinsAPIConnectionSettings struct {
 | 
						|
	Hostname    string
 | 
						|
	Port        int
 | 
						|
	UseNodePort bool
 | 
						|
}
 | 
						|
 | 
						|
type setBearerToken struct {
 | 
						|
	rt    http.RoundTripper
 | 
						|
	token string
 | 
						|
}
 | 
						|
 | 
						|
func (t *setBearerToken) transport() http.RoundTripper {
 | 
						|
	if t.rt != nil {
 | 
						|
		return t.rt
 | 
						|
	}
 | 
						|
	return http.DefaultTransport
 | 
						|
}
 | 
						|
 | 
						|
func (t *setBearerToken) RoundTrip(r *http.Request) (*http.Response, error) {
 | 
						|
	r.Header.Set("Authorization", fmt.Sprintf("Bearer %s", t.token))
 | 
						|
	return t.transport().RoundTrip(r)
 | 
						|
}
 | 
						|
 | 
						|
// CreateOrUpdateJob creates or updates a job from config.
 | 
						|
func (jenkins *jenkins) CreateOrUpdateJob(config, jobName string) (job *gojenkins.Job, created bool, err error) {
 | 
						|
	// create or update
 | 
						|
	job, err = jenkins.GetJob(jobName)
 | 
						|
	if isNotFoundError(err) {
 | 
						|
		job, err = jenkins.CreateJob(config, jobName)
 | 
						|
		created = true
 | 
						|
		return job, true, errors.WithStack(err)
 | 
						|
	} else if err != nil {
 | 
						|
		return job, false, errors.WithStack(err)
 | 
						|
	}
 | 
						|
 | 
						|
	err = job.UpdateConfig(config)
 | 
						|
	return job, false, errors.WithStack(err)
 | 
						|
}
 | 
						|
 | 
						|
// BuildJenkinsAPIUrl returns Jenkins API URL.
 | 
						|
func (j JenkinsAPIConnectionSettings) BuildJenkinsAPIUrl(serviceName string, serviceNamespace string, servicePort int32, serviceNodePort int32) string {
 | 
						|
	if j.Hostname == "" && j.Port == 0 {
 | 
						|
		return fmt.Sprintf("http://%s.%s:%d", serviceName, serviceNamespace, servicePort)
 | 
						|
	}
 | 
						|
 | 
						|
	if j.Hostname != "" && j.UseNodePort {
 | 
						|
		return fmt.Sprintf("http://%s:%d", j.Hostname, serviceNodePort)
 | 
						|
	}
 | 
						|
 | 
						|
	return fmt.Sprintf("http://%s:%d", j.Hostname, j.Port)
 | 
						|
}
 | 
						|
 | 
						|
// Validate validates jenkins API connection settings.
 | 
						|
func (j JenkinsAPIConnectionSettings) Validate() error {
 | 
						|
	if j.Port > 0 && j.UseNodePort {
 | 
						|
		return errors.New("can't use service port and nodePort both. Please use port or nodePort")
 | 
						|
	}
 | 
						|
 | 
						|
	if j.Port < 0 {
 | 
						|
		return errors.New("service port cannot be lower than 0")
 | 
						|
	}
 | 
						|
 | 
						|
	if (j.Hostname == "" && j.Port > 0) || (j.Hostname == "" && j.UseNodePort) {
 | 
						|
		return errors.New("empty hostname is now allowed. Please provide hostname")
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// NewUserAndPasswordAuthorization creates Jenkins API client with user and password authorization.
 | 
						|
func NewUserAndPasswordAuthorization(url, userName, passwordOrToken string) (Jenkins, error) {
 | 
						|
	return newClient(url, userName, passwordOrToken)
 | 
						|
}
 | 
						|
 | 
						|
// NewBearerTokenAuthorization creates Jenkins API client with bearer token authorization.
 | 
						|
func NewBearerTokenAuthorization(url, token string) (Jenkins, error) {
 | 
						|
	return newClient(url, "", token)
 | 
						|
}
 | 
						|
 | 
						|
func newClient(url, userName, passwordOrToken string) (Jenkins, error) {
 | 
						|
	if strings.HasSuffix(url, "/") {
 | 
						|
		url = url[:len(url)-1]
 | 
						|
	}
 | 
						|
 | 
						|
	jenkinsClient := &jenkins{}
 | 
						|
	jenkinsClient.Server = url
 | 
						|
 | 
						|
	var basicAuth *gojenkins.BasicAuth
 | 
						|
	jar, err := cookiejar.New(nil)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errors.Wrap(err, "couldn't create a cookie jar")
 | 
						|
	}
 | 
						|
 | 
						|
	httpClient := &http.Client{
 | 
						|
		Jar:     jar,
 | 
						|
		Timeout: 10 * time.Second,
 | 
						|
	}
 | 
						|
 | 
						|
	if len(userName) > 0 && len(passwordOrToken) > 0 {
 | 
						|
		basicAuth = &gojenkins.BasicAuth{Username: userName, Password: passwordOrToken}
 | 
						|
	} else {
 | 
						|
		httpClient.Transport = &setBearerToken{token: passwordOrToken, rt: httpClient.Transport}
 | 
						|
	}
 | 
						|
 | 
						|
	jenkinsClient.Requester = &gojenkins.Requester{
 | 
						|
		Base:      url,
 | 
						|
		SslVerify: true,
 | 
						|
		Client:    httpClient,
 | 
						|
		BasicAuth: basicAuth,
 | 
						|
	}
 | 
						|
	if _, err := jenkinsClient.Init(); err != nil {
 | 
						|
		return nil, errors.Wrap(err, "couldn't init Jenkins API client")
 | 
						|
	}
 | 
						|
 | 
						|
	status, err := jenkinsClient.Poll()
 | 
						|
	if err != nil {
 | 
						|
		return nil, errors.Wrap(err, "couldn't poll data from Jenkins API")
 | 
						|
	}
 | 
						|
	if status != http.StatusOK {
 | 
						|
		return nil, errors.Errorf("couldn't poll data from Jenkins API, invalid status code returned: %d", status)
 | 
						|
	}
 | 
						|
 | 
						|
	return jenkinsClient, nil
 | 
						|
}
 | 
						|
 | 
						|
func isNotFoundError(err error) bool {
 | 
						|
	if err != nil {
 | 
						|
		return err.Error() == errorNotFound.Error()
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
func (jenkins *jenkins) GetNodeSecret(name string) (string, error) {
 | 
						|
	var content string
 | 
						|
	r, err := jenkins.Requester.GetXML(fmt.Sprintf("/computer/%s/slave-agent.jnlp", name), &content, nil)
 | 
						|
	if err != nil {
 | 
						|
		return "", errors.WithStack(err)
 | 
						|
	}
 | 
						|
	defer r.Body.Close()
 | 
						|
 | 
						|
	match := regex.FindStringSubmatch(content)
 | 
						|
	if match == nil {
 | 
						|
		return "", errors.New("Node secret cannot be parsed")
 | 
						|
	}
 | 
						|
 | 
						|
	result := make(map[string]string)
 | 
						|
 | 
						|
	for i, name := range regex.SubexpNames() {
 | 
						|
		if i != 0 && name != "" {
 | 
						|
			result[name] = match[i]
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return result["secret"], nil
 | 
						|
}
 | 
						|
 | 
						|
// Returns the list of all plugins installed on the Jenkins server.
 | 
						|
// You can supply depth parameter, to limit how much data is returned.
 | 
						|
func (jenkins *jenkins) GetPlugins(depth int) (*gojenkins.Plugins, error) {
 | 
						|
	p := gojenkins.Plugins{Jenkins: &jenkins.Jenkins, Raw: new(gojenkins.PluginResponse), Base: "/pluginManager", Depth: depth}
 | 
						|
	statusCode, err := p.Poll()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if statusCode != http.StatusOK {
 | 
						|
		return nil, fmt.Errorf("invalid status code returned: %d", statusCode)
 | 
						|
	}
 | 
						|
	return &p, nil
 | 
						|
}
 |