@@ -44,6 +44,9 @@ func NewBlobStore(containerClient *container.Client, options *BlobStoreOptions)
44
44
45
45
// ClaimOwnership attempts to claim ownership of the partitions in partitionOwnership and returns
46
46
// the actual partitions that were claimed.
47
+ //
48
+ // If we fail to claim ownership because of another update then it will be omitted from the
49
+ // returned slice of [Ownership]'s. It is not considered an error.
47
50
func (b * BlobStore ) ClaimOwnership (ctx context.Context , partitionOwnership []azeventhubs.Ownership , options * azeventhubs.ClaimOwnershipOptions ) ([]azeventhubs.Ownership , error ) {
48
51
var ownerships []azeventhubs.Ownership
49
52
@@ -54,13 +57,12 @@ func (b *BlobStore) ClaimOwnership(ctx context.Context, partitionOwnership []aze
54
57
if err != nil {
55
58
return nil , err
56
59
}
57
- lastModified , etag , err := b .setMetadata (ctx , blobName , newOwnershipBlobMetadata ( po ), po . ETag )
60
+ lastModified , etag , err := b .setOwnershipMetadata (ctx , blobName , po )
58
61
59
62
if err != nil {
60
- if bloberror .HasCode (err , bloberror .ConditionNotMet ) {
61
- // we can fail to claim ownership and that's okay - it's expected that clients will
62
- // attempt to claim with whatever state they hold locally. If they fail it just means
63
- // someone else claimed ownership before them.
63
+ if bloberror .HasCode (err ,
64
+ bloberror .ConditionNotMet , // updated before we could update it
65
+ bloberror .BlobAlreadyExists ) { // created before we could create it
64
66
continue
65
67
}
66
68
@@ -179,25 +181,28 @@ func (b *BlobStore) ListOwnership(ctx context.Context, fullyQualifiedNamespace s
179
181
}
180
182
181
183
// UpdateCheckpoint updates a specific checkpoint with a sequence and offset.
184
+ //
185
+ // NOTE: This function doesn't attempt to prevent simultaneous checkpoint updates - ownership is assumed.
182
186
func (b * BlobStore ) UpdateCheckpoint (ctx context.Context , checkpoint azeventhubs.Checkpoint , options * azeventhubs.UpdateCheckpointOptions ) error {
183
187
blobName , err := nameForCheckpointBlob (checkpoint )
184
188
185
189
if err != nil {
186
190
return err
187
191
}
188
192
189
- _ , _ , err = b .setMetadata (ctx , blobName , newCheckpointBlobMetadata ( checkpoint ), nil )
193
+ _ , _ , err = b .setCheckpointMetadata (ctx , blobName , checkpoint )
190
194
return err
191
195
}
192
196
193
- func (b * BlobStore ) setMetadata (ctx context.Context , blobName string , blobMetadata map [string ]* string , etag * azcore.ETag ) (* time.Time , azcore.ETag , error ) {
197
+ func (b * BlobStore ) setOwnershipMetadata (ctx context.Context , blobName string , ownership azeventhubs.Ownership ) (* time.Time , azcore.ETag , error ) {
198
+ blobMetadata := newOwnershipBlobMetadata (ownership )
194
199
blobClient := b .cc .NewBlockBlobClient (blobName )
195
200
196
- if etag != nil {
201
+ if ownership . ETag != nil {
197
202
setMetadataResp , err := blobClient .SetMetadata (ctx , blobMetadata , & blob.SetMetadataOptions {
198
203
AccessConditions : & blob.AccessConditions {
199
204
ModifiedAccessConditions : & blob.ModifiedAccessConditions {
200
- IfMatch : etag ,
205
+ IfMatch : ownership . ETag ,
201
206
},
202
207
},
203
208
})
@@ -207,29 +212,52 @@ func (b *BlobStore) setMetadata(ctx context.Context, blobName string, blobMetada
207
212
}
208
213
209
214
return setMetadataResp .LastModified , * setMetadataResp .ETag , nil
210
- } else {
211
- setMetadataResp , err := blobClient .SetMetadata (ctx , blobMetadata , nil )
215
+ }
212
216
213
- if err == nil {
214
- return setMetadataResp .LastModified , * setMetadataResp .ETag , nil
215
- }
217
+ uploadResp , err := blobClient .Upload (ctx , streaming .NopCloser (bytes .NewReader ([]byte {})), & blockblob.UploadOptions {
218
+ Metadata : blobMetadata ,
219
+ AccessConditions : & blob.AccessConditions {
220
+ ModifiedAccessConditions : & blob.ModifiedAccessConditions {
221
+ IfNoneMatch : to .Ptr (azcore .ETag ("*" )),
222
+ },
223
+ },
224
+ })
216
225
217
- if ! bloberror . HasCode ( err , bloberror . BlobNotFound ) {
218
- return nil , "" , err
219
- }
226
+ if err != nil {
227
+ return nil , "" , err
228
+ }
220
229
221
- // in JS they check to see if the error is BlobNotFound. If it is, then they
222
- // do a full upload of a blob instead.
223
- uploadResp , err := blobClient .Upload (ctx , streaming .NopCloser (bytes .NewReader ([]byte {})), & blockblob.UploadOptions {
224
- Metadata : blobMetadata ,
225
- })
230
+ return uploadResp .LastModified , * uploadResp .ETag , nil
231
+ }
226
232
227
- if err != nil {
228
- return nil , "" , err
229
- }
233
+ // setCheckpointMetadata sets the metadata for a checkpoint, falling back to creating
234
+ // the blob if it doesn't already exist.
235
+ //
236
+ // NOTE: unlike [setOwnershipMetadata] this function doesn't attempt to prevent simultaneous
237
+ // checkpoint updates - ownership is assumed.
238
+ func (b * BlobStore ) setCheckpointMetadata (ctx context.Context , blobName string , checkpoint azeventhubs.Checkpoint ) (* time.Time , azcore.ETag , error ) {
239
+ blobMetadata := newCheckpointBlobMetadata (checkpoint )
240
+ blobClient := b .cc .NewBlockBlobClient (blobName )
241
+
242
+ setMetadataResp , err := blobClient .SetMetadata (ctx , blobMetadata , nil )
243
+
244
+ if err == nil {
245
+ return setMetadataResp .LastModified , * setMetadataResp .ETag , nil
246
+ }
230
247
231
- return uploadResp .LastModified , * uploadResp .ETag , nil
248
+ if ! bloberror .HasCode (err , bloberror .BlobNotFound ) {
249
+ return nil , "" , err
232
250
}
251
+
252
+ uploadResp , err := blobClient .Upload (ctx , streaming .NopCloser (bytes .NewReader ([]byte {})), & blockblob.UploadOptions {
253
+ Metadata : blobMetadata ,
254
+ })
255
+
256
+ if err != nil {
257
+ return nil , "" , err
258
+ }
259
+
260
+ return uploadResp .LastModified , * uploadResp .ETag , nil
233
261
}
234
262
235
263
func nameForCheckpointBlob (a azeventhubs.Checkpoint ) (string , error ) {
0 commit comments