fleet/ee/server/service/orbit.go
Jahziel Villasana-Espinoza aaea56de2b
fix: various setup experience bugs (#23091)
> No issue, just stuff I noticed while testing

# Checklist for submitter

If some of the following don't apply, delete the relevant line.

<!-- Note that API documentation changes are now addressed by the
product design team. -->

- [x] Manual QA for all new/changed functionality
- For Orbit and Fleet Desktop changes:
- [x] Orbit runs on macOS, Linux and Windows. Check if the orbit
feature/bugfix should only apply to one platform (`runtime.GOOS`).
- [ ] Manual QA must be performed in the three main OSs, macOS, Windows
and Linux.
- [ ] Auto-update manual QA, from released version of component to new
version (see [tools/tuf/test](../tools/tuf/test/README.md)).
2024-10-23 13:06:54 -04:00

212 lines
7 KiB
Go

package service
import (
"context"
"strings"
"github.com/fleetdm/fleet/v4/server/contexts/ctxerr"
"github.com/fleetdm/fleet/v4/server/fleet"
"github.com/fleetdm/fleet/v4/server/ptr"
"github.com/go-kit/log/level"
"github.com/google/uuid"
)
func (svc *Service) GetOrbitSetupExperienceStatus(ctx context.Context, orbitNodeKey string, forceRelease bool) (*fleet.SetupExperienceStatusPayload, error) {
// this is not a user-authenticated endpoint
svc.authz.SkipAuthorization(ctx)
host, err := svc.ds.LoadHostByOrbitNodeKey(ctx, orbitNodeKey)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "loading host by orbit node key")
}
appCfg, err := svc.ds.AppConfig(ctx)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "getting app config")
}
// get the status of the bootstrap package deployment
bootstrapPkg, err := svc.ds.GetHostMDMMacOSSetup(ctx, host.ID)
if err != nil && !fleet.IsNotFound(err) {
return nil, ctxerr.Wrap(ctx, err, "get bootstrap package status")
}
// NOTE: bootstrapPkg can be nil if there was none to install.
var bootstrapPkgResult *fleet.SetupExperienceBootstrapPackageResult
if bootstrapPkg != nil {
bootstrapPkgResult = &fleet.SetupExperienceBootstrapPackageResult{
Name: bootstrapPkg.BootstrapPackageName,
Status: bootstrapPkg.BootstrapPackageStatus,
}
}
// get the status of the configuration profiles
cfgProfs, err := svc.ds.GetHostMDMAppleProfiles(ctx, host.UUID)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "get configuration profiles status")
}
var cfgProfResults []*fleet.SetupExperienceConfigurationProfileResult
for _, prof := range cfgProfs {
// NOTE: DDM profiles (declarations) are ignored because while a device is
// awaiting to be released, it cannot process a DDM session (at least
// that's what we noticed during testing).
if strings.HasPrefix(prof.ProfileUUID, fleet.MDMAppleDeclarationUUIDPrefix) {
continue
}
status := fleet.MDMDeliveryPending
if prof.Status != nil {
status = *prof.Status
}
cfgProfResults = append(cfgProfResults, &fleet.SetupExperienceConfigurationProfileResult{
ProfileUUID: prof.ProfileUUID,
Name: prof.Name,
Status: status,
})
}
// AccountConfiguration covers the (optional) command to setup SSO.
adminTeamFilter := fleet.TeamFilter{
User: &fleet.User{GlobalRole: ptr.String(fleet.RoleAdmin)},
}
acctCmds, err := svc.ds.ListMDMCommands(ctx, adminTeamFilter, &fleet.MDMCommandListOptions{
Filters: fleet.MDMCommandFilters{
HostIdentifier: host.UUID,
RequestType: "AccountConfiguration",
},
})
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "list AccountConfiguration commands")
}
var acctCfgResult *fleet.SetupExperienceAccountConfigurationResult
if len(acctCmds) > 0 {
// there may be more than one if e.g. the worker job that sends them had to
// retry, but they would all be processed anyway so we can only care about
// the first one.
acctCfgResult = &fleet.SetupExperienceAccountConfigurationResult{
CommandUUID: acctCmds[0].CommandUUID,
Status: acctCmds[0].Status,
}
}
// get status of software installs and script execution
res, err := svc.ds.ListSetupExperienceResultsByHostUUID(ctx, host.UUID)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "listing setup experience results")
}
payload := &fleet.SetupExperienceStatusPayload{
BootstrapPackage: bootstrapPkgResult,
ConfigurationProfiles: cfgProfResults,
AccountConfiguration: acctCfgResult,
Software: make([]*fleet.SetupExperienceStatusResult, 0),
OrgLogoURL: appCfg.OrgInfo.OrgLogoURLLightBackground,
}
for _, r := range res {
if r.IsForScript() {
payload.Script = r
}
if r.IsForSoftware() {
payload.Software = append(payload.Software, r)
}
}
if forceRelease || isDeviceReadyForRelease(payload) {
manual, err := isDeviceReleasedManually(ctx, svc.ds, host)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "check if device is released manually")
}
if manual {
return payload, nil
}
// otherwise the device is not released manually, proceed with automatic
// release
if forceRelease {
level.Warn(svc.logger).Log("msg", "force-releasing device, DEP enrollment commands, profiles, software installs and script execution may not have all completed", "host_uuid", host.UUID)
} else {
level.Info(svc.logger).Log("msg", "releasing device, all DEP enrollment commands, profiles, software installs and script execution have completed", "host_uuid", host.UUID)
}
// Host will be marked as no longer "awaiting configuration" in the command handler
if err := svc.mdmAppleCommander.DeviceConfigured(ctx, host.UUID, uuid.NewString()); err != nil {
return nil, ctxerr.Wrap(ctx, err, "failed to enqueue DeviceConfigured command")
}
}
_, err = svc.SetupExperienceNextStep(ctx, host.UUID)
if err != nil {
return nil, ctxerr.Wrap(ctx, err, "getting next step for host setup experience")
}
return payload, nil
}
func isDeviceReleasedManually(ctx context.Context, ds fleet.Datastore, host *fleet.Host) (bool, error) {
var manualRelease bool
if host.TeamID == nil {
ac, err := ds.AppConfig(ctx)
if err != nil {
return false, ctxerr.Wrap(ctx, err, "get AppConfig to read enable_release_device_manually")
}
manualRelease = ac.MDM.MacOSSetup.EnableReleaseDeviceManually.Value
} else {
tm, err := ds.Team(ctx, *host.TeamID)
if err != nil {
return false, ctxerr.Wrap(ctx, err, "get Team to read enable_release_device_manually")
}
manualRelease = tm.Config.MDM.MacOSSetup.EnableReleaseDeviceManually.Value
}
return manualRelease, nil
}
func isDeviceReadyForRelease(payload *fleet.SetupExperienceStatusPayload) bool {
// default to "do release" and return false as soon as we find a reason not
// to.
if payload.BootstrapPackage != nil {
if payload.BootstrapPackage.Status != fleet.MDMBootstrapPackageFailed &&
payload.BootstrapPackage.Status != fleet.MDMBootstrapPackageInstalled {
// bootstrap package is still pending, not ready for release
return false
}
}
if payload.AccountConfiguration != nil {
if payload.AccountConfiguration.Status != fleet.MDMAppleStatusAcknowledged &&
payload.AccountConfiguration.Status != fleet.MDMAppleStatusError &&
payload.AccountConfiguration.Status != fleet.MDMAppleStatusCommandFormatError {
// account configuration command is still pending, not ready for release
return false
}
}
for _, prof := range payload.ConfigurationProfiles {
if prof.Status != fleet.MDMDeliveryFailed &&
prof.Status != fleet.MDMDeliveryVerifying &&
prof.Status != fleet.MDMDeliveryVerified {
// profile is still pending, not ready for release
return false
}
}
for _, sw := range payload.Software {
if sw.Status != fleet.SetupExperienceStatusFailure &&
sw.Status != fleet.SetupExperienceStatusSuccess {
// software is still pending, not ready for release
return false
}
}
if payload.Script != nil {
if payload.Script.Status != fleet.SetupExperienceStatusFailure &&
payload.Script.Status != fleet.SetupExperienceStatusSuccess {
// script is still pending, not ready for release
return false
}
}
return true
}