2016-09-26 18:48:55 +00:00
|
|
|
package service
|
2016-09-04 05:13:42 +00:00
|
|
|
|
|
|
|
import (
|
2017-03-15 15:55:30 +00:00
|
|
|
"context"
|
|
|
|
|
2020-11-11 17:59:12 +00:00
|
|
|
"github.com/fleetdm/fleet/server/kolide"
|
2016-09-04 05:13:42 +00:00
|
|
|
)
|
|
|
|
|
2018-01-10 19:38:20 +00:00
|
|
|
func (svc service) ApplyPackSpecs(ctx context.Context, specs []*kolide.PackSpec) error {
|
|
|
|
return svc.ds.ApplyPackSpecs(specs)
|
2016-09-04 05:13:42 +00:00
|
|
|
}
|
|
|
|
|
2018-01-10 19:38:20 +00:00
|
|
|
func (svc service) GetPackSpecs(ctx context.Context) ([]*kolide.PackSpec, error) {
|
|
|
|
return svc.ds.GetPackSpecs()
|
2016-09-04 05:13:42 +00:00
|
|
|
}
|
|
|
|
|
2018-05-08 01:54:29 +00:00
|
|
|
func (svc service) GetPackSpec(ctx context.Context, name string) (*kolide.PackSpec, error) {
|
|
|
|
return svc.ds.GetPackSpec(name)
|
|
|
|
}
|
|
|
|
|
2018-01-10 19:38:20 +00:00
|
|
|
func (svc service) ListPacks(ctx context.Context, opt kolide.ListOptions) ([]*kolide.Pack, error) {
|
|
|
|
return svc.ds.ListPacks(opt)
|
2016-09-04 05:13:42 +00:00
|
|
|
}
|
|
|
|
|
2018-01-10 19:38:20 +00:00
|
|
|
func (svc service) GetPack(ctx context.Context, id uint) (*kolide.Pack, error) {
|
|
|
|
return svc.ds.Pack(id)
|
2016-09-04 05:13:42 +00:00
|
|
|
}
|
|
|
|
|
2018-06-15 14:13:11 +00:00
|
|
|
func (svc service) NewPack(ctx context.Context, p kolide.PackPayload) (*kolide.Pack, error) {
|
|
|
|
var pack kolide.Pack
|
|
|
|
|
|
|
|
if p.Name != nil {
|
|
|
|
pack.Name = *p.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Description != nil {
|
|
|
|
pack.Description = *p.Description
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Platform != nil {
|
|
|
|
pack.Platform = *p.Platform
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Disabled != nil {
|
|
|
|
pack.Disabled = *p.Disabled
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := svc.ds.NewPack(&pack)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.HostIDs != nil {
|
|
|
|
for _, hostID := range *p.HostIDs {
|
|
|
|
err = svc.AddHostToPack(ctx, hostID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.LabelIDs != nil {
|
|
|
|
for _, labelID := range *p.LabelIDs {
|
|
|
|
err = svc.AddLabelToPack(ctx, labelID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &pack, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (svc service) ModifyPack(ctx context.Context, id uint, p kolide.PackPayload) (*kolide.Pack, error) {
|
|
|
|
pack, err := svc.ds.Pack(id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Name != nil {
|
|
|
|
pack.Name = *p.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Description != nil {
|
|
|
|
pack.Description = *p.Description
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Platform != nil {
|
|
|
|
pack.Platform = *p.Platform
|
|
|
|
}
|
|
|
|
|
|
|
|
if p.Disabled != nil {
|
|
|
|
pack.Disabled = *p.Disabled
|
|
|
|
}
|
|
|
|
|
|
|
|
err = svc.ds.SavePack(pack)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// we must determine what hosts are attached to this pack. then, given
|
|
|
|
// our new set of host_ids, we will mutate the database to reflect the
|
|
|
|
// desired state.
|
|
|
|
if p.HostIDs != nil {
|
|
|
|
|
|
|
|
// first, let's retrieve the total set of hosts
|
|
|
|
hosts, err := svc.ListHostsInPack(ctx, pack.ID, kolide.ListOptions{})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// it will be efficient to create a data structure with constant time
|
|
|
|
// lookups to determine whether or not a host is already added
|
|
|
|
existingHosts := map[uint]bool{}
|
|
|
|
for _, host := range hosts {
|
|
|
|
existingHosts[host] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// we will also make a constant time lookup map for the desired set of
|
|
|
|
// hosts as well.
|
|
|
|
desiredHosts := map[uint]bool{}
|
|
|
|
for _, hostID := range *p.HostIDs {
|
|
|
|
desiredHosts[hostID] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the request declares a host ID but the host is not already
|
|
|
|
// associated with the pack, we add it
|
|
|
|
for _, hostID := range *p.HostIDs {
|
|
|
|
if !existingHosts[hostID] {
|
|
|
|
err = svc.AddHostToPack(ctx, hostID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the request does not declare the ID of a host which currently
|
|
|
|
// exists, we delete the existing relationship
|
|
|
|
for hostID := range existingHosts {
|
|
|
|
if !desiredHosts[hostID] {
|
|
|
|
err = svc.RemoveHostFromPack(ctx, hostID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we must determine what labels are attached to this pack. then, given
|
|
|
|
// our new set of label_ids, we will mutate the database to reflect the
|
|
|
|
// desired state.
|
|
|
|
if p.LabelIDs != nil {
|
|
|
|
|
|
|
|
// first, let's retrieve the total set of labels
|
|
|
|
labels, err := svc.ListLabelsForPack(ctx, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// it will be efficient to create a data structure with constant time
|
|
|
|
// lookups to determine whether or not a label is already added
|
|
|
|
existingLabels := map[uint]bool{}
|
|
|
|
for _, label := range labels {
|
|
|
|
existingLabels[label.ID] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// we will also make a constant time lookup map for the desired set of
|
|
|
|
// labels as well.
|
|
|
|
desiredLabels := map[uint]bool{}
|
|
|
|
for _, labelID := range *p.LabelIDs {
|
|
|
|
desiredLabels[labelID] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the request declares a label ID but the label is not already
|
|
|
|
// associated with the pack, we add it
|
|
|
|
for _, labelID := range *p.LabelIDs {
|
|
|
|
if !existingLabels[labelID] {
|
|
|
|
err = svc.AddLabelToPack(ctx, labelID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the request does not declare the ID of a label which currently
|
|
|
|
// exists, we delete the existing relationship
|
|
|
|
for labelID := range existingLabels {
|
|
|
|
if !desiredLabels[labelID] {
|
|
|
|
err = svc.RemoveLabelFromPack(ctx, labelID, pack.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pack, err
|
|
|
|
}
|
|
|
|
|
2018-05-04 18:05:55 +00:00
|
|
|
func (svc service) DeletePack(ctx context.Context, name string) error {
|
|
|
|
return svc.ds.DeletePack(name)
|
2016-10-03 03:14:35 +00:00
|
|
|
}
|
|
|
|
|
2018-06-15 14:13:11 +00:00
|
|
|
func (svc service) DeletePackByID(ctx context.Context, id uint) error {
|
|
|
|
pack, err := svc.ds.Pack(id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return svc.ds.DeletePack(pack.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (svc service) AddLabelToPack(ctx context.Context, lid, pid uint) error {
|
|
|
|
return svc.ds.AddLabelToPack(lid, pid)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (svc service) RemoveLabelFromPack(ctx context.Context, lid, pid uint) error {
|
|
|
|
return svc.ds.RemoveLabelFromPack(lid, pid)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (svc service) AddHostToPack(ctx context.Context, hid, pid uint) error {
|
|
|
|
return svc.ds.AddHostToPack(hid, pid)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (svc service) RemoveHostFromPack(ctx context.Context, hid, pid uint) error {
|
|
|
|
return svc.ds.RemoveHostFromPack(hid, pid)
|
|
|
|
}
|
|
|
|
|
2016-10-14 15:59:27 +00:00
|
|
|
func (svc service) ListLabelsForPack(ctx context.Context, pid uint) ([]*kolide.Label, error) {
|
Add host_ids and label_ids fields to the packs API (#737)
This PR adds the `host_ids` and `label_ids` field to the packs HTTP API so that one can operate on the hosts/labels which a pack is scheduled to be executed on. This replaces (and deletes) the `/api/v1/kolide/packs/123/labels/456` API in favor of `PATCH /api/v1/packs/123` and specifying the `label_ids` field. This also allows for bulk operations.
Consider the following API examples:
## Creating a pack with a known set of hosts and labels
The key addition is the `host_ids` and `label_ids` field in both the request and the response.
### Request
```
POST /api/v1/kolide/packs
```
```json
{
"name": "My new pack",
"description": "The newest of the packs",
"host_ids": [1, 2, 3],
"label_ids": [1, 3, 5]
}
```
### Response
```json
{
"pack": {
"id": 123,
"name": "My new pack",
"description": "The newest of the packs",
"platform": "",
"created_by": 1,
"disabled": false,
"query_count": 0,
"total_hosts_count": 5,
"host_ids": [1, 2, 3],
"label_ids": [1, 3, 5]
}
}
```
## Modifying the hosts and/or labels that a pack is scheduled to execute on
### Request
```
PATCH /api/v1/kolide/packs/123
```
```json
{
"host_ids": [1, 2, 3, 4, 5],
"label_ids": [1, 3, 5, 7]
}
```
### Response
```json
{
"pack": {
"id": 123,
"name": "My new pack",
"description": "The newest of the packs",
"platform": "",
"created_by": 1,
"disabled": false,
"query_count": 0,
"total_hosts_count": 5,
"host_ids": [1, 2, 3, 4, 5],
"label_ids": [1, 3, 5, 7]
}
}
```
close #633
2017-01-03 17:32:06 +00:00
|
|
|
return svc.ds.ListLabelsForPack(pid)
|
2016-09-04 05:13:42 +00:00
|
|
|
}
|
|
|
|
|
2017-03-01 19:56:13 +00:00
|
|
|
func (svc service) ListHostsInPack(ctx context.Context, pid uint, opt kolide.ListOptions) ([]uint, error) {
|
2016-11-22 21:56:05 +00:00
|
|
|
return svc.ds.ListHostsInPack(pid, opt)
|
|
|
|
}
|
|
|
|
|
2017-03-01 19:56:13 +00:00
|
|
|
func (svc service) ListExplicitHostsInPack(ctx context.Context, pid uint, opt kolide.ListOptions) ([]uint, error) {
|
2017-01-11 20:33:30 +00:00
|
|
|
return svc.ds.ListExplicitHostsInPack(pid, opt)
|
|
|
|
}
|
|
|
|
|
2016-10-17 19:30:47 +00:00
|
|
|
func (svc service) ListPacksForHost(ctx context.Context, hid uint) ([]*kolide.Pack, error) {
|
2018-01-10 19:38:20 +00:00
|
|
|
return svc.ds.ListPacksForHost(hid)
|
2016-10-17 19:30:47 +00:00
|
|
|
}
|