agentendpoint/patch_windows.go (113 lines of code) (raw):

// Copyright 2018 Google Inc. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. //go:build windows // +build windows package agentendpoint import ( "context" "fmt" "time" "github.com/GoogleCloudPlatform/osconfig/clog" "github.com/GoogleCloudPlatform/osconfig/ospatch" "github.com/GoogleCloudPlatform/osconfig/packages" "github.com/GoogleCloudPlatform/osconfig/retryutil" "cloud.google.com/go/osconfig/agentendpoint/apiv1/agentendpointpb" ) func (r *patchTask) classFilter() ([]string, error) { var classifications = map[agentendpointpb.WindowsUpdateSettings_Classification]string{ agentendpointpb.WindowsUpdateSettings_CRITICAL: "e6cf1350-c01b-414d-a61f-263d14d133b4", agentendpointpb.WindowsUpdateSettings_SECURITY: "0fa1201d-4330-4fa8-8ae9-b877473b6441", agentendpointpb.WindowsUpdateSettings_DEFINITION: "e0789628-ce08-4437-be74-2495b842f43b", agentendpointpb.WindowsUpdateSettings_DRIVER: "ebfc1fc5-71a4-4f7b-9aca-3b9a503104a0", agentendpointpb.WindowsUpdateSettings_FEATURE_PACK: "b54e7d24-7add-428f-8b75-90a396fa584f", agentendpointpb.WindowsUpdateSettings_SERVICE_PACK: "68c5b0a3-d1a6-4553-ae49-01d3a7827828", agentendpointpb.WindowsUpdateSettings_TOOL: "b4832bd8-e735-4761-8daf-37f882276dab", agentendpointpb.WindowsUpdateSettings_UPDATE_ROLLUP: "28bc880e-0592-4cbf-8f95-c79b17911d5f", agentendpointpb.WindowsUpdateSettings_UPDATE: "cd5ffd1e-e932-4e3a-bf74-18bf0b1bbd83", } var cf []string for _, c := range r.Task.GetPatchConfig().GetWindowsUpdate().GetClassifications() { sc, ok := classifications[c] if !ok { return nil, fmt.Errorf("Unknown classification: %s", c) } cf = append(cf, sc) } return cf, nil } func (r *patchTask) installWUAUpdates(ctx context.Context, cf []string) (int32, error) { clog.Infof(ctx, "Searching for available Windows updates.") session, err := packages.NewUpdateSession() if err != nil { return 0, err } defer session.Close() updts, err := ospatch.GetWUAUpdates(ctx, session, cf, r.Task.GetPatchConfig().GetWindowsUpdate().GetExcludes(), r.Task.GetPatchConfig().GetWindowsUpdate().GetExclusivePatches()) if err != nil { return 0, err } defer updts.Release() count, err := updts.Count() if err != nil { return 0, err } if count == 0 { clog.Infof(ctx, "No Windows updates available to install") return 0, nil } clog.Infof(ctx, "%d Windows updates to install", count) if r.Task.GetDryRun() { clog.Infof(ctx, "Running in dryrun mode, not updating.") return 0, nil } for i := int32(0); i < count; i++ { if err := r.reportContinuingState(ctx, agentendpointpb.ApplyPatchesTaskProgress_APPLYING_PATCHES); err != nil { return i, err } updt, err := updts.Item(int(i)) if err != nil { return i, err } defer updt.Release() if err := session.InstallWUAUpdate(ctx, updt); err != nil { return i, fmt.Errorf(`installUpdate(updt): %v`, err) } } return count, nil } func (r *patchTask) wuaUpdates(ctx context.Context) error { cf, err := r.classFilter() if err != nil { return err } // We keep searching for and installing updates until the count == 0, // we get a stop signal, or retries exceed 10. retries := 10 for i := 1; i <= retries; i++ { if err := r.reportContinuingState(ctx, agentendpointpb.ApplyPatchesTaskProgress_APPLYING_PATCHES); err != nil { return err } count, err := r.installWUAUpdates(ctx, cf) if err != nil { clog.Errorf(ctx, "Error installing Windows updates (attempt %d): %v", i, err) time.Sleep(60 * time.Second) continue } if count == 0 { return nil } } return fmt.Errorf("failed to install all updates after trying %d times", retries) } func (r *patchTask) runUpdates(ctx context.Context) error { // Install GooGet updates first as this will allow us to update the agent prior to any potential WUA bugs/errors. if packages.GooGetExists { if err := r.reportContinuingState(ctx, agentendpointpb.ApplyPatchesTaskProgress_APPLYING_PATCHES); err != nil { return err } clog.Debugf(ctx, "Installing GooGet package updates.") opts := []ospatch.GooGetUpdateOption{ ospatch.GooGetDryRun(r.Task.GetDryRun()), } if err := retryutil.RetryFunc(ctx, 3*time.Minute, "installing GooGet package updates", func() error { return ospatch.RunGooGetUpdate(ctx, opts...) }); err != nil { return err } } // Don't use retry function as wuaUpdates handles it's own retries. if err := r.wuaUpdates(ctx); err != nil { return err } return nil }