v2.0.0
This commit is contained in:
commit
e455af7f1f
33 changed files with 8471 additions and 0 deletions
478
agent/wua_windows.go
Normal file
478
agent/wua_windows.go
Normal file
|
|
@ -0,0 +1,478 @@
|
|||
/*
|
||||
Copyright 2022 AmidaWare LLC.
|
||||
|
||||
Licensed under the Tactical RMM License Version 1.0 (the “License”).
|
||||
You may only use the Licensed Software in accordance with the License.
|
||||
A copy of the License is available at:
|
||||
|
||||
https://license.tacticalrmm.com
|
||||
|
||||
*/
|
||||
|
||||
// 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.
|
||||
|
||||
// code taken from https://github.com/GoogleCloudPlatform/osconfig/tree/master/ospatch
|
||||
// and modified by https://github.com/wh1te909
|
||||
package agent
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"sync"
|
||||
|
||||
rmm "github.com/amidaware/rmmagent/shared"
|
||||
ole "github.com/go-ole/go-ole"
|
||||
"github.com/go-ole/go-ole/oleutil"
|
||||
)
|
||||
|
||||
const (
|
||||
S_OK = 0
|
||||
S_FALSE = 1
|
||||
)
|
||||
|
||||
var wuaSession sync.Mutex
|
||||
|
||||
// IUpdateSession is a an IUpdateSession.
|
||||
type IUpdateSession struct {
|
||||
*ole.IDispatch
|
||||
}
|
||||
|
||||
func (s *IUpdateSession) Close() {
|
||||
if s.IDispatch != nil {
|
||||
s.IDispatch.Release()
|
||||
}
|
||||
ole.CoUninitialize()
|
||||
wuaSession.Unlock()
|
||||
}
|
||||
|
||||
func NewUpdateSession() (*IUpdateSession, error) {
|
||||
wuaSession.Lock()
|
||||
if err := ole.CoInitializeEx(0, ole.COINIT_MULTITHREADED); err != nil {
|
||||
e, ok := err.(*ole.OleError)
|
||||
// S_OK and S_FALSE are both are Success codes.
|
||||
// https://docs.microsoft.com/en-us/windows/win32/learnwin32/error-handling-in-com
|
||||
if !ok || (e.Code() != S_OK && e.Code() != S_FALSE) {
|
||||
wuaSession.Unlock()
|
||||
return nil, fmt.Errorf(`ole.CoInitializeEx(0, ole.COINIT_MULTITHREADED): %v`, err)
|
||||
}
|
||||
}
|
||||
|
||||
s := &IUpdateSession{}
|
||||
|
||||
unknown, err := oleutil.CreateObject("Microsoft.Update.Session")
|
||||
if err != nil {
|
||||
s.Close()
|
||||
return nil, fmt.Errorf(`oleutil.CreateObject("Microsoft.Update.Session"): %v`, err)
|
||||
}
|
||||
disp, err := unknown.QueryInterface(ole.IID_IDispatch)
|
||||
if err != nil {
|
||||
unknown.Release()
|
||||
s.Close()
|
||||
return nil, fmt.Errorf(`error creating Dispatch object from Microsoft.Update.Session connection: %v`, err)
|
||||
}
|
||||
s.IDispatch = disp
|
||||
|
||||
return s, nil
|
||||
}
|
||||
|
||||
// InstallWUAUpdate install a WIndows update.
|
||||
func (s *IUpdateSession) InstallWUAUpdate(updt *IUpdate) error {
|
||||
_, err := updt.GetProperty("Title")
|
||||
if err != nil {
|
||||
return fmt.Errorf(`updt.GetProperty("Title"): %v`, err)
|
||||
}
|
||||
|
||||
updts, err := NewUpdateCollection()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer updts.Release()
|
||||
|
||||
eula, err := updt.GetProperty("EulaAccepted")
|
||||
if err != nil {
|
||||
return fmt.Errorf(`updt.GetProperty("EulaAccepted"): %v`, err)
|
||||
}
|
||||
// https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-oaut/7b39eb24-9d39-498a-bcd8-75c38e5823d0
|
||||
if eula.Val == 0 {
|
||||
if _, err := updt.CallMethod("AcceptEula"); err != nil {
|
||||
return fmt.Errorf(`updt.CallMethod("AcceptEula"): %v`, err)
|
||||
}
|
||||
}
|
||||
|
||||
if err := updts.Add(updt); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err := s.DownloadWUAUpdateCollection(updts); err != nil {
|
||||
return fmt.Errorf("DownloadWUAUpdateCollection error: %v", err)
|
||||
}
|
||||
|
||||
if err := s.InstallWUAUpdateCollection(updts); err != nil {
|
||||
return fmt.Errorf("InstallWUAUpdateCollection error: %v", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func NewUpdateCollection() (*IUpdateCollection, error) {
|
||||
updateCollObj, err := oleutil.CreateObject("Microsoft.Update.UpdateColl")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`oleutil.CreateObject("Microsoft.Update.UpdateColl"): %v`, err)
|
||||
}
|
||||
defer updateCollObj.Release()
|
||||
|
||||
updateColl, err := updateCollObj.IDispatch(ole.IID_IDispatch)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return &IUpdateCollection{IDispatch: updateColl}, nil
|
||||
}
|
||||
|
||||
type IUpdateCollection struct {
|
||||
*ole.IDispatch
|
||||
}
|
||||
|
||||
type IUpdate struct {
|
||||
*ole.IDispatch
|
||||
}
|
||||
|
||||
func (c *IUpdateCollection) Add(updt *IUpdate) error {
|
||||
if _, err := c.CallMethod("Add", updt.IDispatch); err != nil {
|
||||
return fmt.Errorf(`IUpdateCollection.CallMethod("Add", updt): %v`, err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *IUpdateCollection) RemoveAt(i int) error {
|
||||
if _, err := c.CallMethod("RemoveAt", i); err != nil {
|
||||
return fmt.Errorf(`IUpdateCollection.CallMethod("RemoveAt", %d): %v`, i, err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *IUpdateCollection) Count() (int32, error) {
|
||||
return GetCount(c.IDispatch)
|
||||
}
|
||||
|
||||
func (c *IUpdateCollection) Item(i int) (*IUpdate, error) {
|
||||
updtRaw, err := c.GetProperty("Item", i)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`IUpdateCollection.GetProperty("Item", %d): %v`, i, err)
|
||||
}
|
||||
return &IUpdate{IDispatch: updtRaw.ToIDispatch()}, nil
|
||||
}
|
||||
|
||||
// GetCount returns the Count property.
|
||||
func GetCount(dis *ole.IDispatch) (int32, error) {
|
||||
countRaw, err := dis.GetProperty("Count")
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf(`IDispatch.GetProperty("Count"): %v`, err)
|
||||
}
|
||||
count, _ := countRaw.Value().(int32)
|
||||
|
||||
return count, nil
|
||||
}
|
||||
|
||||
func (u *IUpdate) kbaIDs() ([]string, error) {
|
||||
kbArticleIDsRaw, err := u.GetProperty("KBArticleIDs")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`IUpdate.GetProperty("KBArticleIDs"): %v`, err)
|
||||
}
|
||||
kbArticleIDs := kbArticleIDsRaw.ToIDispatch()
|
||||
defer kbArticleIDs.Release()
|
||||
|
||||
count, err := GetCount(kbArticleIDs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if count == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
var ss []string
|
||||
for i := 0; i < int(count); i++ {
|
||||
item, err := kbArticleIDs.GetProperty("Item", i)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`kbArticleIDs.GetProperty("Item", %d): %v`, i, err)
|
||||
}
|
||||
|
||||
ss = append(ss, item.ToString())
|
||||
}
|
||||
return ss, nil
|
||||
}
|
||||
|
||||
func (u *IUpdate) categories() ([]string, []string, error) {
|
||||
catRaw, err := u.GetProperty("Categories")
|
||||
if err != nil {
|
||||
return nil, nil, fmt.Errorf(`IUpdate.GetProperty("Categories"): %v`, err)
|
||||
}
|
||||
cat := catRaw.ToIDispatch()
|
||||
defer cat.Release()
|
||||
|
||||
count, err := GetCount(cat)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if count == 0 {
|
||||
return nil, nil, nil
|
||||
}
|
||||
|
||||
var cns, cids []string
|
||||
for i := 0; i < int(count); i++ {
|
||||
itemRaw, err := cat.GetProperty("Item", i)
|
||||
if err != nil {
|
||||
return nil, nil, fmt.Errorf(`cat.GetProperty("Item", %d): %v`, i, err)
|
||||
}
|
||||
item := itemRaw.ToIDispatch()
|
||||
defer item.Release()
|
||||
|
||||
name, err := item.GetProperty("Name")
|
||||
if err != nil {
|
||||
return nil, nil, fmt.Errorf(`item.GetProperty("Name"): %v`, err)
|
||||
}
|
||||
|
||||
categoryID, err := item.GetProperty("CategoryID")
|
||||
if err != nil {
|
||||
return nil, nil, fmt.Errorf(`item.GetProperty("CategoryID"): %v`, err)
|
||||
}
|
||||
|
||||
cns = append(cns, name.ToString())
|
||||
cids = append(cids, categoryID.ToString())
|
||||
}
|
||||
return cns, cids, nil
|
||||
}
|
||||
|
||||
func (u *IUpdate) moreInfoURLs() ([]string, error) {
|
||||
moreInfoURLsRaw, err := u.GetProperty("MoreInfoURLs")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`IUpdate.GetProperty("MoreInfoURLs"): %v`, err)
|
||||
}
|
||||
moreInfoURLs := moreInfoURLsRaw.ToIDispatch()
|
||||
defer moreInfoURLs.Release()
|
||||
|
||||
count, err := GetCount(moreInfoURLs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if count == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
var ss []string
|
||||
for i := 0; i < int(count); i++ {
|
||||
item, err := moreInfoURLs.GetProperty("Item", i)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`moreInfoURLs.GetProperty("Item", %d): %v`, i, err)
|
||||
}
|
||||
|
||||
ss = append(ss, item.ToString())
|
||||
}
|
||||
return ss, nil
|
||||
}
|
||||
|
||||
func (c *IUpdateCollection) extractPkg(item int) (*rmm.WUAPackage, error) {
|
||||
updt, err := c.Item(item)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer updt.Release()
|
||||
|
||||
title, err := updt.GetProperty("Title")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("Title"): %v`, err)
|
||||
}
|
||||
|
||||
description, err := updt.GetProperty("Description")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("Description"): %v`, err)
|
||||
}
|
||||
|
||||
kbArticleIDs, err := updt.kbaIDs()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
categories, categoryIDs, err := updt.categories()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
moreInfoURLs, err := updt.moreInfoURLs()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
supportURL, err := updt.GetProperty("SupportURL")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("SupportURL"): %v`, err)
|
||||
}
|
||||
|
||||
identityRaw, err := updt.GetProperty("Identity")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("Identity"): %v`, err)
|
||||
}
|
||||
identity := identityRaw.ToIDispatch()
|
||||
defer identity.Release()
|
||||
|
||||
revisionNumber, err := identity.GetProperty("RevisionNumber")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`identity.GetProperty("RevisionNumber"): %v`, err)
|
||||
}
|
||||
|
||||
updateID, err := identity.GetProperty("UpdateID")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`identity.GetProperty("UpdateID"): %v`, err)
|
||||
}
|
||||
|
||||
severity, err := updt.GetProperty("MsrcSeverity")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("MsrcSeverity"): %v`, err)
|
||||
}
|
||||
|
||||
isInstalled, err := updt.GetProperty("IsInstalled")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("IsInstalled"): %v`, err)
|
||||
}
|
||||
|
||||
isDownloaded, err := updt.GetProperty("IsDownloaded")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf(`updt.GetProperty("IsDownloaded"): %v`, err)
|
||||
}
|
||||
|
||||
return &rmm.WUAPackage{
|
||||
Title: title.ToString(),
|
||||
Description: description.ToString(),
|
||||
SupportURL: supportURL.ToString(),
|
||||
KBArticleIDs: kbArticleIDs,
|
||||
UpdateID: updateID.ToString(),
|
||||
Categories: categories,
|
||||
CategoryIDs: categoryIDs,
|
||||
MoreInfoURLs: moreInfoURLs,
|
||||
Severity: severity.ToString(),
|
||||
RevisionNumber: int32(revisionNumber.Val),
|
||||
Downloaded: isDownloaded.Value().(bool),
|
||||
Installed: isInstalled.Value().(bool),
|
||||
}, nil
|
||||
}
|
||||
|
||||
// WUAUpdates queries the Windows Update Agent API searcher with the provided query.
|
||||
func WUAUpdates(query string) ([]rmm.WUAPackage, error) {
|
||||
session, err := NewUpdateSession()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("error creating NewUpdateSession: %v", err)
|
||||
}
|
||||
defer session.Close()
|
||||
|
||||
updts, err := session.GetWUAUpdateCollection(query)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("error calling GetWUAUpdateCollection with query %q: %v", query, err)
|
||||
}
|
||||
defer updts.Release()
|
||||
|
||||
updtCnt, err := updts.Count()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if updtCnt == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
var packages []rmm.WUAPackage
|
||||
for i := 0; i < int(updtCnt); i++ {
|
||||
pkg, err := updts.extractPkg(i)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
packages = append(packages, *pkg)
|
||||
}
|
||||
return packages, nil
|
||||
}
|
||||
|
||||
// DownloadWUAUpdateCollection downloads all updates in a IUpdateCollection
|
||||
func (s *IUpdateSession) DownloadWUAUpdateCollection(updates *IUpdateCollection) error {
|
||||
// returns IUpdateDownloader
|
||||
// https://docs.microsoft.com/en-us/windows/desktop/api/wuapi/nn-wuapi-iupdatedownloader
|
||||
downloaderRaw, err := s.CallMethod("CreateUpdateDownloader")
|
||||
if err != nil {
|
||||
return fmt.Errorf("error calling method CreateUpdateDownloader on IUpdateSession: %v", err)
|
||||
}
|
||||
downloader := downloaderRaw.ToIDispatch()
|
||||
defer downloader.Release()
|
||||
|
||||
if _, err := downloader.PutProperty("Updates", updates.IDispatch); err != nil {
|
||||
return fmt.Errorf("error calling PutProperty Updates on IUpdateDownloader: %v", err)
|
||||
}
|
||||
|
||||
if _, err := downloader.CallMethod("Download"); err != nil {
|
||||
return fmt.Errorf("error calling method Download on IUpdateDownloader: %v", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// InstallWUAUpdateCollection installs all updates in a IUpdateCollection
|
||||
func (s *IUpdateSession) InstallWUAUpdateCollection(updates *IUpdateCollection) error {
|
||||
// returns IUpdateInstallersession *ole.IDispatch,
|
||||
// https://docs.microsoft.com/en-us/windows/desktop/api/wuapi/nf-wuapi-iupdatesession-createupdateinstaller
|
||||
installerRaw, err := s.CallMethod("CreateUpdateInstaller")
|
||||
if err != nil {
|
||||
return fmt.Errorf("error calling method CreateUpdateInstaller on IUpdateSession: %v", err)
|
||||
}
|
||||
installer := installerRaw.ToIDispatch()
|
||||
defer installer.Release()
|
||||
|
||||
if _, err := installer.PutProperty("Updates", updates.IDispatch); err != nil {
|
||||
return fmt.Errorf("error calling PutProperty Updates on IUpdateInstaller: %v", err)
|
||||
}
|
||||
|
||||
// TODO: Look into using the async methods and attempt to track/log progress.
|
||||
if _, err := installer.CallMethod("Install"); err != nil {
|
||||
return fmt.Errorf("error calling method Install on IUpdateInstaller: %v", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// GetWUAUpdateCollection queries the Windows Update Agent API searcher with the provided query
|
||||
// and returns a IUpdateCollection.
|
||||
func (s *IUpdateSession) GetWUAUpdateCollection(query string) (*IUpdateCollection, error) {
|
||||
// returns IUpdateSearcher
|
||||
// https://msdn.microsoft.com/en-us/library/windows/desktop/aa386515(v=vs.85).aspx
|
||||
searcherRaw, err := s.CallMethod("CreateUpdateSearcher")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("error calling CreateUpdateSearcher: %v", err)
|
||||
}
|
||||
searcher := searcherRaw.ToIDispatch()
|
||||
defer searcher.Release()
|
||||
|
||||
// returns ISearchResult
|
||||
// https://msdn.microsoft.com/en-us/library/windows/desktop/aa386077(v=vs.85).aspx
|
||||
resultRaw, err := searcher.CallMethod("Search", query)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("error calling method Search on IUpdateSearcher: %v", err)
|
||||
}
|
||||
result := resultRaw.ToIDispatch()
|
||||
defer result.Release()
|
||||
|
||||
// returns IUpdateCollection
|
||||
// https://msdn.microsoft.com/en-us/library/windows/desktop/aa386107(v=vs.85).aspx
|
||||
updtsRaw, err := result.GetProperty("Updates")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("error calling GetProperty Updates on ISearchResult: %v", err)
|
||||
}
|
||||
|
||||
return &IUpdateCollection{IDispatch: updtsRaw.ToIDispatch()}, nil
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue