-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathobject_array.go
234 lines (195 loc) · 7.36 KB
/
object_array.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
package wmi
import (
"fmt"
"github.com/oiweiwei/go-msrpc/msrpc/dcom"
"github.com/oiweiwei/go-msrpc/msrpc/dcom/wmio"
"github.com/oiweiwei/go-msrpc/msrpc/dtyp"
)
func UnmarshalObjectArrayWithClass(b []byte, cls wmio.Class) (*ObjectArray, error) {
o := &ObjectArray{}
return o, o.DecodeWithClass(wmio.NewCodec(b), cls)
}
// UnmarshalObjectArrayWithClasses decodes the ObjectArray structure
// using the classes lookup map. Note that classes lookup map is updated
// during the call and must be reused for the next object decoding.
func UnmarshalObjectArrayWithClasses(b []byte, classes map[string]*wmio.Class) (*ObjectArray, error) {
o := &ObjectArray{}
return o, o.DecodeWithClasses(wmio.NewCodec(b), classes)
}
// The ObjectArray structure MUST be used to encode multiple CIM
// objects that are returned in response to the IWbemWCOSmartEnum::Next
// method. This structure is also used to encode parameters of the
// optimized IWbemObjectSink::Indicate method.<6> To minimize network
// bandwidth, a server SHOULD support the ObjectArray structure when
// an array of CIM objects is sent.
type ObjectArray struct {
// dwByteOrdering (4 bytes): The byte ordering. It MUST be value 0.
ByteOrdering uint32
// abSignature (8 bytes): MUST be set to {0x57, 0x42, 0x45, 0x4D,
// 0x44, 0x41, 0x54, 0x41} (a byte array containing the unquoted,
// unterminated ASCII string "WBEMDATA").
Signature []byte
// dwSizeOfHeader1 (4 bytes): This stores the total size of these
// fields: dwByteOrdering, abSignature, dwSizeofHeader1, dwDataSize1,
// dwFlags, bVersion, and bPacketType.
// The size of the header MUST be 0x0000001A. Data immediately follows
// the header.
SizeOfHeader1 uint32
// dwDataSize1 (4 bytes): MUST indicate the length, in bytes, of the
// data that follows this header, starting at the dwSizeOfHeader2 field.
DataSize1 uint32
// dwFlags (4 bytes): The flag value MUST be 0x00000000.
Flags uint32
// bVersion (1 byte): The version number of the header. The version MUST be 1.
Version uint8
// bPacketType (1 byte): The value of this field is dependent on the call
// context.
PacketType uint8
// dwSizeOfHeader2 (4 bytes): This stores the size of these fields:
// dwSizeofHeader2 and dwDataSize2.
// This value MUST be 8. Data immediately follows after the field dwDataSize2.
SizeOfHeader2 uint32
// dwDataSize2 (4 bytes): MUST be the size, in bytes, of the data that follows
// this field.
DataSize2 uint32
// dwSizeOfHeader3 (4 bytes): This stores the size of these fields:
// dwSizeofHeader3, dwDataSize3, and dwNumObjects. This value MUST be 12.
// Data immediately follows after the field dwNumObjects.
SizeOfHeader3 uint32
// dwDataSize3 (4 bytes): MUST indicate the length of the remaining data,
// starting at the wbemObjects field.
DataSize3 uint32
// dwNumObjects (4 bytes): MUST be the number of CIM objects in the
// ObjectArray.
NumObjects uint32
// wbemObjects (variable): The objects array that contains the CIM class
// definition and CIM instances. These CIM objects MUST be encoded
// in the WBEM_DATAPACKET_OBJECT structure.
Objects []*DataPacketObject
}
// The WBEM_DATAPACKET_OBJECT MUST contain the CIM class definition and CIM instances.
type DataPacketObject struct {
// dwSizeOfHeader (4 bytes): The size, in bytes, of the
// WBEM_DATAPACKET_OBJECT header, which MUST be 0x00000009.
SizeOfHeader uint32
// dwSizeOfData (4 bytes): The size, in bytes, of the data
// following the WBEM_DATAPACKET_OBJECT header.
SizeOfData uint32
// bObjectType (1 byte): The type of data in the data packet.
// The type MUST take one of the following specified values.
//
// * 1:
// Object is type WBEMOBJECT_CLASS:
// Structure contains the complete CIM Class definition.
// * 2:
// Object is type WBEMOBJECT_INSTANCE:
// Structure contains the complete CIM Instance definition.
// * 3:
// Object is type WBEMOBJECT_INSTANCE_NOCLASS.
// Structure contains CIM Instance without the CIM Class definition.
ObjectType uint8
// dwSizeOfHeader (4 bytes): The size, in bytes, of the header, which
// MUST be 0x00000008 for WBEMOBJECT_CLASS and 0x00000018 for WBEMOBJECT_INSTANCE,
// and WBEMOBJECT_INSTANCE_NOCLASS.
SizeOfHeader2 uint32
// dwSizeOfData (4 bytes): The size, in bytes, of the data that follows the
/// header.
SizeOfData2 uint32
// classID (16 bytes): The unique identifier of the CIM class type.
ClassID *dcom.ClassID
// Object (variable): The CIM object carried into the
// WBEM_DATAPACKET_OBJECT, having dwSizeOfData bytes. The embedded CIM
// object MUST match the selector field bObjectType.
Object *wmio.Object
}
var (
// Object is type WBEMOBJECT_CLASS:
// Structure contains the complete CIM Class definition.
ObjectTypeClass uint8 = 1
// Object is type WBEMOBJECT_INSTANCE:
// Structure contains the complete CIM Instance definition.
ObjectTypeInstance uint8 = 2
// Object is type WBEMOBJECT_INSTANCE_NOCLASS.
// Structure contains CIM Instance without the CIM Class definition.
ObjectTypeInstanceNoClass uint8 = 3
)
var DefaultClassID = ""
func (o *ObjectArray) DecodeWithClass(r *wmio.Codec, cls wmio.Class) error {
return o.DecodeWithClasses(r, map[string]*wmio.Class{DefaultClassID: &cls})
}
func (o *ObjectArray) DecodeWithClasses(r *wmio.Codec, classes map[string]*wmio.Class) error {
r.ReadData(&o.ByteOrdering)
o.Signature = make([]byte, 8)
r.Read(o.Signature)
r.ReadData(&o.SizeOfHeader1)
r.ReadData(&o.DataSize1)
r.ReadData(&o.Flags)
r.ReadData(&o.Version)
r.ReadData(&o.PacketType)
r.ReadData(&o.SizeOfHeader2)
r.ReadData(&o.DataSize2)
r.ReadData(&o.SizeOfHeader3)
r.ReadData(&o.DataSize3)
r.ReadData(&o.NumObjects)
for i := 0; i < int(o.NumObjects); i++ {
var cls wmio.Class
if _, ok := classes[DefaultClassID]; ok {
cls = *classes[DefaultClassID]
}
var po DataPacketObject
r.ReadData(&po.SizeOfHeader)
r.ReadData(&po.SizeOfData)
r.ReadData(&po.ObjectType)
r.ReadData(&po.SizeOfHeader2)
r.ReadData(&po.SizeOfData2)
if r.Err() != nil {
return r.Err()
}
var err error
switch po.ObjectType {
case ObjectTypeClass:
if po.Object, err = wmio.UnmarshalWithClass(r.Bytes()[:po.SizeOfData2], wmio.Class{}); err != nil {
return err
}
cls = po.Object.Class.CurrentClass
case ObjectTypeInstance:
if po.ClassID, err = ReadClassID(r); err != nil {
return err
}
if po.Object, err = wmio.UnmarshalWithClass(r.Bytes()[:po.SizeOfData2], wmio.Class{}); err != nil {
return err
}
cls = po.Object.Instance.CurrentClass
// store classes into map.
classes[po.ClassID.String()] = &cls
case ObjectTypeInstanceNoClass:
if po.ClassID, err = ReadClassID(r); err != nil {
return err
}
if _, ok := classes[po.ClassID.String()]; ok {
cls = *(classes[po.ClassID.String()])
}
if cls.Name == "" {
return fmt.Errorf("class name is empty")
}
if po.Object, err = wmio.UnmarshalWithClass(r.Bytes()[:po.SizeOfData2], cls); err != nil {
return err
}
default:
return fmt.Errorf("unknown class type %d", po.ObjectType)
}
o.Objects, r = append(o.Objects, &po), wmio.NewCodec(r.Bytes()[po.SizeOfData2:])
}
return r.Err()
}
func ReadClassID(r *wmio.Codec) (*dcom.ClassID, error) {
clsID := make([]byte, 16)
if err := r.Read(clsID); err != nil {
return nil, fmt.Errorf("class_id: read: %w", err)
}
guid, err := dtyp.GUIDFromBytes(clsID)
if err != nil {
return nil, fmt.Errorf("class_id: decode: %w", err)
}
return (*dcom.ClassID)(guid), nil
}