Skip to content

Commit

Permalink
add FleetDM plugin (#363)
Browse files Browse the repository at this point in the history
  • Loading branch information
wasaga authored Sep 12, 2024
1 parent 02b09ea commit 57123b4
Show file tree
Hide file tree
Showing 15 changed files with 718 additions and 18 deletions.
19 changes: 1 addition & 18 deletions .golangci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -7,19 +7,15 @@ linters:
- bodyclose
- decorder
- dogsled
- dupl
- durationcheck
- errcheck
- errname
- errorlint
- exhaustive
- gci
- goconst
- godox
# - gci # https://github.com/daixiang0/gci/issues/209
- gofumpt
- goheader
- goimports
- gomoddirectives
- gomodguard
- goprintffuncname
- gosec
Expand All @@ -28,42 +24,29 @@ linters:
- grouper
- importas
- ineffassign
- interfacebloat
- makezero
- misspell
- nakedret
- nestif
- nilerr
- nilnil
- noctx
- nolintlint
- nosprintfhostport
- paralleltest
- predeclared
- promlinter
- reassign
- revive
- staticcheck
- stylecheck
- tenv
- thelper
- tparallel
- typecheck
- unconvert
- unused
- whitespace

linters-settings:
gci:
custom-order: true
sections:
- standard # Standard section: captures all standard packages.
- default # Default section: contains all imports that could not be matched to another section type.
- prefix(github.com/pomerium/datasource)

issues:
exclude-rules:
# Exclude some linters from running on test files.
- path: _test\.go$
linters:
- gosec
70 changes: 70 additions & 0 deletions cmd/pomerium-datasource/fleetdm.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
package main

import (
"net/http"

"github.com/go-playground/validator/v10"
"github.com/rs/zerolog"
"github.com/spf13/cobra"

"github.com/pomerium/datasource/internal/fleetdm"
"github.com/pomerium/datasource/internal/server"
)

type fleetDMCmd struct {
APIToken string `validate:"required"`
APIURL string `validate:"required,url"`
Address string `validate:"required"`
CertQueryID uint `validate:"required"`

cobra.Command `validate:"-"`
zerolog.Logger `validate:"-"`
}

func fleetDMCommand(log zerolog.Logger) *cobra.Command {
cmd := &fleetDMCmd{
Command: cobra.Command{
Use: "fleetdm",
Short: "run FleetDM connector",
},
Logger: log,
}
cmd.RunE = cmd.exec

cmd.setupFlags()
return &cmd.Command
}

func (cmd *fleetDMCmd) setupFlags() {
flags := cmd.Flags()
flags.StringVar(&cmd.APIToken, "api-token", "", "FleetDM API token")
flags.StringVar(&cmd.APIURL, "api-url", "", "FleetDM API URL")
flags.UintVar(&cmd.CertQueryID, "cert-query-id", 0, "FleetDM certificate query ID")
flags.StringVar(&cmd.Address, "address", ":8080", "tcp address to listen to")
}

func (cmd *fleetDMCmd) exec(c *cobra.Command, _ []string) error {
if err := validator.New().Struct(cmd); err != nil {
return err
}

srv, err := cmd.newServer()
if err != nil {
return err
}

return server.RunHTTPServer(c.Context(), cmd.Address, srv)
}

func (cmd *fleetDMCmd) newServer() (http.Handler, error) {
srv, err := fleetdm.NewServer(
fleetdm.WithAPIToken(cmd.APIToken),
fleetdm.WithAPIURL(cmd.APIURL),
fleetdm.WithCertificateQueryID(cmd.CertQueryID),
)
if err != nil {
return nil, err
}

return srv, nil
}
1 change: 1 addition & 0 deletions cmd/pomerium-datasource/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ func main() {
zenefitsCommand(logger),
ip2LocationCmd,
wellKnownIPsCmd,
fleetDMCommand(logger),
)
if err := rootCmd.ExecuteContext(signalContext(logger)); err != nil {
logger.Fatal().Err(err).Msg("exit")
Expand Down
1 change: 1 addition & 0 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,7 @@ require (
github.com/gostaticanalysis/forcetypeassert v0.1.0 // indirect
github.com/gostaticanalysis/nilerr v0.1.1 // indirect
github.com/hashicorp/errwrap v1.1.0 // indirect
github.com/hashicorp/go-set/v3 v3.0.0-alpha.1 // indirect
github.com/hashicorp/go-version v1.7.0 // indirect
github.com/hashicorp/hcl v1.0.0 // indirect
github.com/hexops/gotextdiff v1.0.3 // indirect
Expand Down
2 changes: 2 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -378,6 +378,8 @@ github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY
github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo=
github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
github.com/hashicorp/go-set/v3 v3.0.0-alpha.1 h1:dPUtuqKJGgxtF7YO42oE+NdUONXi5nfLMKH2NpBffIM=
github.com/hashicorp/go-set/v3 v3.0.0-alpha.1/go.mod h1:7bJRgsF3EL3AtRTzcKXdjAFbYGSef+1gHXhglGGO52k=
github.com/hashicorp/go-version v1.2.1/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
github.com/hashicorp/go-version v1.7.0 h1:5tqGy27NaOTB8yJKUZELlFAS/LTKJkrmONwQKeRZfjY=
github.com/hashicorp/go-version v1.7.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
Expand Down
233 changes: 233 additions & 0 deletions internal/fleetdm/client/client.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,233 @@
package client

import (
"context"
"fmt"
"iter"
"net/http"
"net/url"

"github.com/pomerium/datasource/internal/jsonutil"

"github.com/hashicorp/go-set/v3"
)

const (
maxHostPerPage = 500
)

type Client struct {
cfg *config
}

// New creates a new FleetDM API client
// see https://fleetdm.com/docs/rest-api/rest-api
func New(opts ...Option) (*Client, error) {
cfg := newConfig(opts...)
return &Client{
cfg: cfg,
}, nil
}

func (c *Client) ListHosts(
ctx context.Context,
) iter.Seq2[Host, error] {
var args []string
if c.cfg.withPolicies {
args = append(args, "populate_policies", "true")
}
if c.cfg.withVulnerabilities {
args = append(args, "populate_software", "true")
}
return fetchItemsPaged(ctx, c, convertHostRecord, "hosts", "/api/v1/fleet/hosts", maxHostPerPage, args...)
}

func (c *Client) listTeams(ctx context.Context) ([]uint, error) {
iter, err := fetchItems(ctx, c,
func(tm struct {
ID uint `json:"id"`
},
) (uint, error) {
return tm.ID, nil
},
"teams", "/api/v1/fleet/teams")
if err != nil {
return nil, err
}

var ids []uint
for id, err := range iter {
if err != nil {
return nil, err
}
ids = append(ids, id)
}

return ids, nil
}

func (c *Client) ListPolicies(ctx context.Context) (iter.Seq2[Policy, error], error) {
teams, err := c.listTeams(ctx)
if err != nil {
return nil, fmt.Errorf("list teams: %w", err)
}

global, err := fetchItems(ctx, c, convertPolicy, "policies", "/api/latest/fleet/policies")
if err != nil {
return nil, fmt.Errorf("list global policies: %w", err)
}

policies := []iter.Seq2[Policy, error]{global}
for _, teamID := range teams {
p, err := fetchItems(ctx, c, convertPolicy, "policies", fmt.Sprintf("/api/latest/fleet/teams/%d/policies", teamID))
if err != nil {
return nil, fmt.Errorf("list team policies: %w", err)
}
policies = append(policies, p)
}

return dedup(policies...), nil
}

func (c *Client) QueryCertificates(
ctx context.Context,
queryID uint,
) (iter.Seq2[CertificateSHA1QueryItem, error], error) {
return fetchItems(ctx, c, convertCertificateQuery, "results", fmt.Sprintf("/api/v1/fleet/queries/%d/report", queryID))
}

func fetchItemsPaged[InternalRecord, ExternalRecord any](
ctx context.Context,
c *Client,
convert func(InternalRecord) (ExternalRecord, error),
key string,
path string,
itemsPerPage int,
args ...string,
) iter.Seq2[ExternalRecord, error] {
return func(yield func(ExternalRecord, error) bool) {
page := 0
for {
iter, err := fetchItems(ctx, c, convert, key, path, append(args, "page", fmt.Sprint(page), "per_page", fmt.Sprint(itemsPerPage))...)
if err != nil {
var v ExternalRecord
if !yield(v, fmt.Errorf("fetch page %d: %w", page, err)) {
return
}
return
}

itemCount := 0
for v, err := range iter {
if err != nil {
err = fmt.Errorf("page %d: %w", page, err)
}
if !yield(v, err) {
return
}

if err != nil {
return
}

itemCount++
}

if itemCount < itemsPerPage {
return
}

page++
}
}
}

func fetchItems[InternalRecord, ExternalRecord any](
ctx context.Context,
c *Client,
convert func(InternalRecord) (ExternalRecord, error),
key string,
path string,
args ...string,
) (iter.Seq2[ExternalRecord, error], error) {
req, err := c.newRequest(ctx, "GET", path, args...)
if err != nil {
return nil, err
}

resp, err := c.cfg.httpClient.Do(req)
if err != nil {
return nil, fmt.Errorf("failed to perform request: %w", err)
}

if resp.StatusCode != http.StatusOK {
_ = resp.Body.Close()
return nil, fmt.Errorf("unexpected status code: %d", resp.StatusCode)
}

return convertIter2(
jsonutil.StreamArrayReadAndClose[InternalRecord](resp.Body, []string{key}),
convert,
), nil
}

func (c *Client) newRequest(
ctx context.Context,
method string,
path string,
kv ...string,
) (*http.Request, error) {
u, err := url.Parse(c.cfg.url)
if err != nil {
return nil, fmt.Errorf("failed to parse api endpoint URL: %w", err)
}
if u.Scheme != "https" && u.Scheme != "http" {
return nil, fmt.Errorf("api endpoint URL scheme must be http or https")
}
u.Path = path

if len(kv)%2 != 0 {
return nil, fmt.Errorf("key-value pairs must be even")
}

query := make(url.Values)
for i := 0; i < len(kv); i += 2 {
query.Add(kv[i], kv[i+1])
}
u.RawQuery = query.Encode()

req, err := http.NewRequest(method, u.String(), nil)
if err != nil {
return nil, fmt.Errorf("failed to create request: %w", err)
}

req.Header.Set("Accept", "application/json")
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", c.cfg.token))
return req.WithContext(ctx), nil
}

func dedup[ID comparable, T interface{ GetID() ID }](
iters ...iter.Seq2[T, error],
) iter.Seq2[T, error] {
return func(yield func(T, error) bool) {
seen := set.New[ID](0)
for _, iter := range iters {
for v, err := range iter {
if err != nil {
if !yield(v, err) {
return
}
continue
}
id := v.GetID()
if seen.Contains(id) {
continue
}
seen.Insert(id)
if !yield(v, nil) {
return
}
}
}
}
}
Loading

0 comments on commit 57123b4

Please sign in to comment.