diff --git a/api/v1alpha1/generated.pb.go b/api/v1alpha1/generated.pb.go index 475e7b0c1..3b9247777 100644 --- a/api/v1alpha1/generated.pb.go +++ b/api/v1alpha1/generated.pb.go @@ -1430,10 +1430,38 @@ func (m *StageStatus) XXX_DiscardUnknown() { var xxx_messageInfo_StageStatus proto.InternalMessageInfo +func (m *StepExecutionMetadata) Reset() { *m = StepExecutionMetadata{} } +func (*StepExecutionMetadata) ProtoMessage() {} +func (*StepExecutionMetadata) Descriptor() ([]byte, []int) { + return fileDescriptor_e26b7f7bbc391025, []int{50} +} +func (m *StepExecutionMetadata) XXX_Unmarshal(b []byte) error { + return m.Unmarshal(b) +} +func (m *StepExecutionMetadata) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + b = b[:cap(b)] + n, err := m.MarshalToSizedBuffer(b) + if err != nil { + return nil, err + } + return b[:n], nil +} +func (m *StepExecutionMetadata) XXX_Merge(src proto.Message) { + xxx_messageInfo_StepExecutionMetadata.Merge(m, src) +} +func (m *StepExecutionMetadata) XXX_Size() int { + return m.Size() +} +func (m *StepExecutionMetadata) XXX_DiscardUnknown() { + xxx_messageInfo_StepExecutionMetadata.DiscardUnknown(m) +} + +var xxx_messageInfo_StepExecutionMetadata proto.InternalMessageInfo + func (m *Verification) Reset() { *m = Verification{} } func (*Verification) ProtoMessage() {} func (*Verification) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{50} + return fileDescriptor_e26b7f7bbc391025, []int{51} } func (m *Verification) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1461,7 +1489,7 @@ var xxx_messageInfo_Verification proto.InternalMessageInfo func (m *VerificationInfo) Reset() { *m = VerificationInfo{} } func (*VerificationInfo) ProtoMessage() {} func (*VerificationInfo) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{51} + return fileDescriptor_e26b7f7bbc391025, []int{52} } func (m *VerificationInfo) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1489,7 +1517,7 @@ var xxx_messageInfo_VerificationInfo proto.InternalMessageInfo func (m *VerifiedStage) Reset() { *m = VerifiedStage{} } func (*VerifiedStage) ProtoMessage() {} func (*VerifiedStage) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{52} + return fileDescriptor_e26b7f7bbc391025, []int{53} } func (m *VerifiedStage) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1517,7 +1545,7 @@ var xxx_messageInfo_VerifiedStage proto.InternalMessageInfo func (m *Warehouse) Reset() { *m = Warehouse{} } func (*Warehouse) ProtoMessage() {} func (*Warehouse) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{53} + return fileDescriptor_e26b7f7bbc391025, []int{54} } func (m *Warehouse) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1545,7 +1573,7 @@ var xxx_messageInfo_Warehouse proto.InternalMessageInfo func (m *WarehouseList) Reset() { *m = WarehouseList{} } func (*WarehouseList) ProtoMessage() {} func (*WarehouseList) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{54} + return fileDescriptor_e26b7f7bbc391025, []int{55} } func (m *WarehouseList) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1573,7 +1601,7 @@ var xxx_messageInfo_WarehouseList proto.InternalMessageInfo func (m *WarehouseSpec) Reset() { *m = WarehouseSpec{} } func (*WarehouseSpec) ProtoMessage() {} func (*WarehouseSpec) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{55} + return fileDescriptor_e26b7f7bbc391025, []int{56} } func (m *WarehouseSpec) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1601,7 +1629,7 @@ var xxx_messageInfo_WarehouseSpec proto.InternalMessageInfo func (m *WarehouseStatus) Reset() { *m = WarehouseStatus{} } func (*WarehouseStatus) ProtoMessage() {} func (*WarehouseStatus) Descriptor() ([]byte, []int) { - return fileDescriptor_e26b7f7bbc391025, []int{56} + return fileDescriptor_e26b7f7bbc391025, []int{57} } func (m *WarehouseStatus) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1682,6 +1710,7 @@ func init() { proto.RegisterType((*StageList)(nil), "github.com.akuity.kargo.api.v1alpha1.StageList") proto.RegisterType((*StageSpec)(nil), "github.com.akuity.kargo.api.v1alpha1.StageSpec") proto.RegisterType((*StageStatus)(nil), "github.com.akuity.kargo.api.v1alpha1.StageStatus") + proto.RegisterType((*StepExecutionMetadata)(nil), "github.com.akuity.kargo.api.v1alpha1.StepExecutionMetadata") proto.RegisterType((*Verification)(nil), "github.com.akuity.kargo.api.v1alpha1.Verification") proto.RegisterType((*VerificationInfo)(nil), "github.com.akuity.kargo.api.v1alpha1.VerificationInfo") proto.RegisterType((*VerifiedStage)(nil), "github.com.akuity.kargo.api.v1alpha1.VerifiedStage") @@ -1696,238 +1725,245 @@ func init() { } var fileDescriptor_e26b7f7bbc391025 = []byte{ - // 3691 bytes of a gzipped FileDescriptorProto + // 3803 bytes of a gzipped FileDescriptorProto 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xdc, 0x5c, 0xcd, 0x6f, 0x1c, 0x47, - 0x76, 0x57, 0xcf, 0x90, 0x43, 0xce, 0x1b, 0x7e, 0x96, 0x28, 0x9b, 0xcb, 0x8d, 0x49, 0xa7, 0x6d, - 0x18, 0x76, 0x6c, 0x0f, 0x23, 0x59, 0xb2, 0x3e, 0x9c, 0x28, 0x98, 0x19, 0xea, 0x83, 0x0a, 0xd7, - 0x62, 0x6a, 0x64, 0x69, 0x57, 0x2b, 0xc1, 0x29, 0xce, 0x14, 0x67, 0x3a, 0x9c, 0xe9, 0x9e, 0xed, - 0xaa, 0xe1, 0x9a, 0x49, 0x90, 0x6c, 0xbe, 0x80, 0x05, 0xf2, 0x81, 0x1c, 0x0c, 0x78, 0x03, 0x24, - 0x40, 0x90, 0x1c, 0x83, 0xe4, 0x1f, 0xc8, 0xc1, 0x87, 0x5c, 0x8c, 0x60, 0x13, 0x18, 0x49, 0x0e, - 0x3e, 0x2c, 0x88, 0x98, 0x0b, 0xe4, 0xb8, 0xb7, 0x5c, 0x04, 0x04, 0x08, 0xea, 0xa3, 0xbb, 0xab, - 0x7b, 0x7a, 0xc4, 0xe9, 0x11, 0x29, 0x08, 0x7b, 0x1b, 0xd6, 0xab, 0xf7, 0x7b, 0xf5, 0xf1, 0xde, - 0xab, 0xf7, 0x5e, 0x55, 0x13, 0x2e, 0xb6, 0x1c, 0xde, 0xee, 0xef, 0x94, 0x1b, 0x5e, 0x77, 0x9d, - 0xec, 0xf5, 0x1d, 0x7e, 0xb0, 0xbe, 0x47, 0xfc, 0x96, 0xb7, 0x4e, 0x7a, 0xce, 0xfa, 0xfe, 0x79, - 0xd2, 0xe9, 0xb5, 0xc9, 0xf9, 0xf5, 0x16, 0x75, 0xa9, 0x4f, 0x38, 0x6d, 0x96, 0x7b, 0xbe, 0xc7, - 0x3d, 0xf4, 0x7a, 0xc4, 0x55, 0x56, 0x5c, 0x65, 0xc9, 0x55, 0x26, 0x3d, 0xa7, 0x1c, 0x70, 0xad, - 0xbc, 0x6b, 0x60, 0xb7, 0xbc, 0x96, 0xb7, 0x2e, 0x99, 0x77, 0xfa, 0xbb, 0xf2, 0x2f, 0xf9, 0x87, - 0xfc, 0xa5, 0x40, 0x57, 0x6e, 0xef, 0x5d, 0x61, 0x65, 0x47, 0x4a, 0xa6, 0x9f, 0x70, 0xea, 0x32, - 0xc7, 0x73, 0xd9, 0xbb, 0xa4, 0xe7, 0x30, 0xea, 0xef, 0x53, 0x7f, 0xbd, 0xb7, 0xd7, 0x12, 0x34, - 0x16, 0xef, 0xb0, 0xbe, 0x3f, 0x30, 0xbc, 0x95, 0x8b, 0x11, 0x52, 0x97, 0x34, 0xda, 0x8e, 0x4b, - 0xfd, 0x83, 0x88, 0xbd, 0x4b, 0x39, 0x49, 0xe3, 0x5a, 0x1f, 0xc6, 0xe5, 0xf7, 0x5d, 0xee, 0x74, - 0xe9, 0x00, 0xc3, 0xfb, 0xc7, 0x31, 0xb0, 0x46, 0x9b, 0x76, 0x49, 0x92, 0xcf, 0x7e, 0x04, 0x67, - 0x2b, 0x2e, 0xe9, 0x1c, 0x30, 0x87, 0xe1, 0xbe, 0x5b, 0xf1, 0x5b, 0xfd, 0x2e, 0x75, 0x39, 0x7a, - 0x15, 0x26, 0x5c, 0xd2, 0xa5, 0xcb, 0xd6, 0xab, 0xd6, 0x9b, 0xc5, 0xea, 0xcc, 0x17, 0x87, 0x6b, - 0x67, 0x8e, 0x0e, 0xd7, 0x26, 0x3e, 0x24, 0x5d, 0x8a, 0x25, 0x05, 0xbd, 0x06, 0x93, 0xfb, 0xa4, - 0xd3, 0xa7, 0xcb, 0x39, 0xd9, 0x65, 0x56, 0x77, 0x99, 0xbc, 0x2f, 0x1a, 0xb1, 0xa2, 0xd9, 0x7f, - 0x94, 0x8f, 0xc1, 0x7f, 0x8b, 0x72, 0xd2, 0x24, 0x9c, 0xa0, 0x2e, 0x14, 0x3a, 0x64, 0x87, 0x76, - 0xd8, 0xb2, 0xf5, 0x6a, 0xfe, 0xcd, 0xd2, 0x85, 0x1b, 0xe5, 0x51, 0x36, 0xb1, 0x9c, 0x02, 0x55, - 0xde, 0x92, 0x38, 0x37, 0x5c, 0xee, 0x1f, 0x54, 0xe7, 0xf4, 0x20, 0x0a, 0xaa, 0x11, 0x6b, 0x21, - 0xe8, 0x0f, 0x2c, 0x28, 0x11, 0xd7, 0xf5, 0x38, 0xe1, 0x62, 0x9b, 0x96, 0x73, 0x52, 0xe8, 0x9d, - 0xf1, 0x85, 0x56, 0x22, 0x30, 0x25, 0xf9, 0xac, 0x96, 0x5c, 0x32, 0x28, 0xd8, 0x94, 0xb9, 0x72, - 0x15, 0x4a, 0xc6, 0x50, 0xd1, 0x02, 0xe4, 0xf7, 0xe8, 0x81, 0x5a, 0x5f, 0x2c, 0x7e, 0xa2, 0xa5, - 0xd8, 0x82, 0xea, 0x15, 0xbc, 0x96, 0xbb, 0x62, 0xad, 0x5c, 0x87, 0x85, 0xa4, 0xc0, 0x2c, 0xfc, - 0xf6, 0x5f, 0x58, 0xb0, 0x64, 0xcc, 0x02, 0xd3, 0x5d, 0xea, 0x53, 0xb7, 0x41, 0xd1, 0x3a, 0x14, - 0xc5, 0x5e, 0xb2, 0x1e, 0x69, 0x04, 0x5b, 0xbd, 0xa8, 0x27, 0x52, 0xfc, 0x30, 0x20, 0xe0, 0xa8, - 0x4f, 0xa8, 0x16, 0xb9, 0xa7, 0xa9, 0x45, 0xaf, 0x4d, 0x18, 0x5d, 0xce, 0xc7, 0xd5, 0x62, 0x5b, - 0x34, 0x62, 0x45, 0xb3, 0x7f, 0x15, 0xbe, 0x11, 0x8c, 0xe7, 0x1e, 0xed, 0xf6, 0x3a, 0x84, 0xd3, - 0x68, 0x50, 0xc7, 0xaa, 0x9e, 0xbd, 0x07, 0xb3, 0x95, 0x5e, 0xcf, 0xf7, 0xf6, 0x69, 0xb3, 0xce, - 0x49, 0x8b, 0xa2, 0x87, 0x00, 0x44, 0x37, 0x54, 0xb8, 0x64, 0x2c, 0x5d, 0xf8, 0xa5, 0xb2, 0xb2, - 0x88, 0xb2, 0x69, 0x11, 0xe5, 0xde, 0x5e, 0x4b, 0x34, 0xb0, 0xb2, 0x30, 0xbc, 0xf2, 0xfe, 0xf9, - 0xf2, 0x3d, 0xa7, 0x4b, 0xab, 0x73, 0x47, 0x87, 0x6b, 0x50, 0x09, 0x11, 0xb0, 0x81, 0x66, 0xff, - 0xa1, 0x05, 0xe7, 0x2a, 0x7e, 0xcb, 0xab, 0x6d, 0x54, 0x7a, 0xbd, 0xdb, 0x94, 0x74, 0x78, 0xbb, - 0xce, 0x09, 0xef, 0x33, 0x74, 0x1d, 0x0a, 0x4c, 0xfe, 0xd2, 0x43, 0x7d, 0x23, 0xd0, 0x3e, 0x45, - 0x7f, 0x72, 0xb8, 0xb6, 0x94, 0xc2, 0x48, 0xb1, 0xe6, 0x42, 0x6f, 0xc1, 0x54, 0x97, 0x32, 0x46, - 0x5a, 0xc1, 0x7a, 0xce, 0x6b, 0x80, 0xa9, 0x6f, 0xa9, 0x66, 0x1c, 0xd0, 0xed, 0x7f, 0xcd, 0xc1, - 0x7c, 0x88, 0xa5, 0xc5, 0x9f, 0xc2, 0xe6, 0xf5, 0x61, 0xa6, 0x6d, 0xcc, 0x50, 0xee, 0x61, 0xe9, - 0xc2, 0x07, 0x23, 0xda, 0x49, 0xda, 0x22, 0x55, 0x97, 0xb4, 0x98, 0x19, 0xb3, 0x15, 0xc7, 0xc4, - 0xa0, 0x2e, 0x00, 0x3b, 0x70, 0x1b, 0x5a, 0xe8, 0x84, 0x14, 0x7a, 0x35, 0xa3, 0xd0, 0x7a, 0x08, - 0x50, 0x45, 0x5a, 0x24, 0x44, 0x6d, 0xd8, 0x10, 0x60, 0xff, 0x93, 0x05, 0x67, 0x53, 0xf8, 0xd0, - 0xaf, 0x24, 0xf6, 0xf3, 0xf5, 0x81, 0xfd, 0x44, 0x03, 0x6c, 0xd1, 0x6e, 0xbe, 0x03, 0xd3, 0x3e, - 0xdd, 0x77, 0xc4, 0x39, 0xa0, 0x57, 0x78, 0x41, 0xf3, 0x4f, 0x63, 0xdd, 0x8e, 0xc3, 0x1e, 0xe8, - 0x6d, 0x28, 0x06, 0xbf, 0xc5, 0x32, 0xe7, 0x85, 0xa9, 0x88, 0x8d, 0x0b, 0xba, 0x32, 0x1c, 0xd1, - 0xed, 0xdf, 0x87, 0xc9, 0x5a, 0x9b, 0xf8, 0x5c, 0x68, 0x8c, 0x4f, 0x7b, 0xde, 0x47, 0x78, 0x4b, - 0x0f, 0x31, 0xd4, 0x18, 0xac, 0x9a, 0x71, 0x40, 0x1f, 0x61, 0xb3, 0xdf, 0x82, 0xa9, 0x7d, 0xea, - 0xcb, 0xf1, 0xe6, 0xe3, 0x60, 0xf7, 0x55, 0x33, 0x0e, 0xe8, 0xf6, 0x7f, 0x5a, 0xb0, 0x24, 0x47, - 0xb0, 0xe1, 0xb0, 0x86, 0xb7, 0x4f, 0xfd, 0x03, 0x4c, 0x59, 0xbf, 0x73, 0xc2, 0x03, 0xda, 0x80, - 0x05, 0x46, 0xbb, 0xfb, 0xd4, 0xaf, 0x79, 0x2e, 0xe3, 0x3e, 0x71, 0x5c, 0xae, 0x47, 0xb6, 0xac, - 0x7b, 0x2f, 0xd4, 0x13, 0x74, 0x3c, 0xc0, 0x81, 0xde, 0x84, 0x69, 0x3d, 0x6c, 0xa1, 0x4a, 0x62, - 0x61, 0x67, 0xc4, 0x1e, 0xe8, 0x39, 0x31, 0x1c, 0x52, 0xed, 0xff, 0xb1, 0x60, 0x51, 0xce, 0xaa, - 0xde, 0xdf, 0x61, 0x0d, 0xdf, 0xe9, 0x09, 0xf7, 0xfa, 0x22, 0x4e, 0xe9, 0x3a, 0xcc, 0x35, 0x83, - 0x85, 0xdf, 0x72, 0xba, 0x0e, 0x97, 0x36, 0x32, 0x59, 0x7d, 0x49, 0x63, 0xcc, 0x6d, 0xc4, 0xa8, - 0x38, 0xd1, 0xdb, 0xfe, 0xeb, 0x3c, 0x9c, 0x0d, 0xba, 0xd0, 0x66, 0xc5, 0xe7, 0xce, 0x2e, 0x69, - 0x70, 0x86, 0x9a, 0x30, 0xd3, 0x8c, 0x9a, 0xb9, 0xb6, 0xbc, 0x2c, 0x8e, 0x33, 0xb4, 0x6e, 0x03, - 0x9e, 0xe3, 0x18, 0x2a, 0x7a, 0x00, 0xf9, 0x96, 0xc3, 0xf5, 0x41, 0x7f, 0x65, 0x34, 0xb3, 0xbe, - 0xe5, 0x24, 0x55, 0xad, 0x5a, 0xd2, 0xa2, 0xf2, 0xb7, 0x1c, 0x8e, 0x05, 0x22, 0xda, 0x81, 0x82, - 0xd3, 0x25, 0x2d, 0x1a, 0x9c, 0xe7, 0xd7, 0x46, 0xc3, 0xde, 0x14, 0x3c, 0x49, 0xf4, 0x30, 0x72, - 0x90, 0x54, 0x86, 0x35, 0xb2, 0x90, 0xd1, 0x10, 0x2a, 0xa2, 0x8c, 0x74, 0x64, 0x19, 0x69, 0xc6, - 0x12, 0xc9, 0x90, 0x54, 0x86, 0x35, 0xb2, 0xfd, 0x55, 0x0e, 0x16, 0xa2, 0xf5, 0xab, 0x79, 0xdd, - 0xae, 0xc3, 0xd1, 0x0a, 0xe4, 0x9c, 0xa6, 0xd6, 0x40, 0xd0, 0x8c, 0xb9, 0xcd, 0x0d, 0x9c, 0x73, - 0x9a, 0xe8, 0x0d, 0x28, 0xec, 0xf8, 0xc4, 0x6d, 0xb4, 0xb5, 0xe6, 0x85, 0xc0, 0x55, 0xd9, 0x8a, - 0x35, 0x15, 0xbd, 0x02, 0x79, 0x4e, 0x5a, 0x5a, 0xe1, 0xc2, 0xf5, 0xbb, 0x47, 0x5a, 0x58, 0xb4, - 0x0b, 0x4d, 0x67, 0xfd, 0x9d, 0xdf, 0xa2, 0x0d, 0xb5, 0xf3, 0x86, 0xa6, 0xd7, 0x55, 0x33, 0x0e, - 0xe8, 0x42, 0x22, 0xe9, 0xf3, 0xb6, 0xe7, 0x2f, 0x4f, 0xc6, 0x25, 0x56, 0x64, 0x2b, 0xd6, 0x54, - 0x71, 0x26, 0x35, 0xe4, 0xf8, 0x39, 0xf5, 0x97, 0x0b, 0xf1, 0x33, 0xa9, 0x16, 0x10, 0x70, 0xd4, - 0x07, 0x3d, 0x86, 0x52, 0xc3, 0xa7, 0x84, 0x7b, 0xfe, 0x06, 0xe1, 0x74, 0x79, 0x2a, 0xb3, 0x06, - 0xce, 0x8b, 0xa0, 0xab, 0x16, 0x41, 0x60, 0x13, 0xcf, 0xfe, 0x99, 0x05, 0xcb, 0xd1, 0xd2, 0xca, - 0xbd, 0x8d, 0x02, 0x0d, 0xbd, 0x3c, 0xd6, 0x90, 0xe5, 0x79, 0x03, 0x0a, 0x4d, 0xa7, 0x45, 0x19, - 0x4f, 0xae, 0xf2, 0x86, 0x6c, 0xc5, 0x9a, 0x8a, 0x2e, 0x00, 0xb4, 0x1c, 0xae, 0x9d, 0x83, 0x5e, - 0xec, 0xf0, 0x08, 0xba, 0x15, 0x52, 0xb0, 0xd1, 0x0b, 0x3d, 0x80, 0xa2, 0x1c, 0xe6, 0x98, 0x66, - 0x27, 0x8f, 0x8a, 0x5a, 0x00, 0x80, 0x23, 0x2c, 0xfb, 0xcb, 0x09, 0x98, 0xba, 0xe9, 0x53, 0xa7, - 0xd5, 0xe6, 0xe8, 0x37, 0x61, 0xba, 0xab, 0x03, 0x56, 0x1d, 0x13, 0xfd, 0xf2, 0x68, 0x32, 0xee, - 0xca, 0x4d, 0x17, 0xc1, 0x6e, 0x34, 0x91, 0xa8, 0x0d, 0x87, 0xa8, 0x22, 0xd8, 0x23, 0x1d, 0x87, - 0x30, 0xb9, 0x6f, 0x46, 0xb0, 0x57, 0x11, 0x8d, 0x58, 0xd1, 0xd0, 0x77, 0xa1, 0xe0, 0xf9, 0x4e, - 0xcb, 0x71, 0x97, 0x8b, 0x72, 0x10, 0xef, 0x8d, 0x66, 0x42, 0x7a, 0x16, 0x77, 0x25, 0x6b, 0xb4, - 0xf8, 0xea, 0x6f, 0xac, 0x21, 0xd1, 0x43, 0x98, 0x52, 0xca, 0x14, 0x18, 0xe8, 0xfa, 0xc8, 0x0e, - 0x46, 0xe9, 0x63, 0xa4, 0xf4, 0xea, 0x6f, 0x86, 0x03, 0x40, 0x54, 0x0f, 0xfd, 0xcb, 0x84, 0x84, - 0x7e, 0x3b, 0x83, 0x7f, 0x19, 0xea, 0x50, 0xea, 0xa1, 0x43, 0x99, 0xcc, 0x02, 0x2a, 0x5d, 0xc6, - 0x30, 0x0f, 0x22, 0x96, 0x58, 0x47, 0x2e, 0x85, 0x31, 0x96, 0x58, 0x87, 0x4d, 0x73, 0xf1, 0x70, - 0x27, 0x08, 0x6c, 0xec, 0x4f, 0xf3, 0xb0, 0xa8, 0x7b, 0xd6, 0xbc, 0x4e, 0x87, 0x36, 0xe4, 0x31, - 0xa9, 0xfc, 0x53, 0x3e, 0xd5, 0x3f, 0x39, 0x30, 0xe9, 0x70, 0xda, 0x0d, 0x92, 0xbb, 0x6a, 0xa6, - 0xd1, 0x44, 0x32, 0xca, 0x9b, 0x02, 0x44, 0xe5, 0x57, 0xe1, 0x2e, 0xe9, 0x5e, 0x58, 0x49, 0x40, - 0x7f, 0x62, 0xc1, 0xd9, 0x7d, 0xea, 0x3b, 0xbb, 0x4e, 0x43, 0x66, 0x47, 0xb7, 0x1d, 0xc6, 0x3d, - 0xff, 0x40, 0x9f, 0x08, 0xef, 0x8f, 0x26, 0xf9, 0xbe, 0x01, 0xb0, 0xe9, 0xee, 0x7a, 0xd5, 0x6f, - 0x6a, 0x69, 0x67, 0xef, 0x0f, 0x42, 0xe3, 0x34, 0x79, 0x2b, 0x3d, 0x80, 0x68, 0xb4, 0x29, 0xc9, - 0xd9, 0x96, 0x99, 0x9c, 0x8d, 0x3c, 0xb0, 0x60, 0xb2, 0x81, 0xcb, 0x32, 0x93, 0xba, 0xcf, 0x2d, - 0x28, 0x69, 0xfa, 0x96, 0xc3, 0x38, 0x7a, 0x34, 0x60, 0xed, 0xe5, 0xd1, 0xac, 0x5d, 0x70, 0x4b, - 0x5b, 0x0f, 0xe3, 0xd5, 0xa0, 0xc5, 0xb0, 0x74, 0x1c, 0x6c, 0xa9, 0x5a, 0xd8, 0x77, 0x33, 0x8d, - 0x3f, 0x72, 0x0c, 0x72, 0x8d, 0xf4, 0xde, 0xd9, 0x3e, 0xcc, 0xc6, 0x8c, 0x1c, 0x5d, 0x82, 0x89, - 0x3d, 0xc7, 0x0d, 0x4e, 0xbd, 0x5f, 0x0c, 0xe2, 0xa9, 0x5f, 0x77, 0xdc, 0xe6, 0x93, 0xc3, 0xb5, - 0xc5, 0x58, 0x67, 0xd1, 0x88, 0x65, 0xf7, 0xe3, 0xc3, 0xb0, 0x6b, 0xd3, 0x3f, 0xfa, 0xdb, 0xb5, - 0x33, 0x3f, 0xf8, 0xc9, 0xab, 0x67, 0xec, 0xcf, 0xf2, 0xb0, 0x90, 0x5c, 0xd5, 0x11, 0x8a, 0x1d, - 0x91, 0x0f, 0x9b, 0x3e, 0x55, 0x1f, 0x96, 0x3b, 0x3d, 0x1f, 0x96, 0x3f, 0x0d, 0x1f, 0x36, 0x71, - 0x62, 0x3e, 0xcc, 0xfe, 0x77, 0x0b, 0xe6, 0xc2, 0x9d, 0xf9, 0x5e, 0x5f, 0x9c, 0xac, 0xd1, 0xaa, - 0x5b, 0x27, 0xbf, 0xea, 0x1f, 0xc3, 0x14, 0xf3, 0xfa, 0x7e, 0x43, 0x86, 0x8f, 0x02, 0xfd, 0x62, - 0x36, 0xa7, 0xa9, 0x78, 0x8d, 0x98, 0x49, 0x35, 0xe0, 0x00, 0xd5, 0x7e, 0x14, 0xce, 0x47, 0x93, - 0x54, 0x44, 0xe1, 0x8b, 0x78, 0x4b, 0xcc, 0x67, 0xda, 0x8c, 0x28, 0x44, 0x2b, 0xd6, 0x54, 0x64, - 0x4b, 0x77, 0x1e, 0x04, 0xb6, 0xc5, 0x2a, 0x68, 0xaf, 0x2c, 0xf7, 0x40, 0x51, 0xec, 0x9f, 0xe5, - 0x43, 0xeb, 0xd1, 0xe9, 0xeb, 0xf7, 0x01, 0x94, 0x67, 0xa2, 0xcd, 0x4d, 0x57, 0xbb, 0xde, 0xda, - 0x18, 0x07, 0x81, 0x76, 0x87, 0x02, 0x45, 0xf9, 0xde, 0x30, 0x06, 0x88, 0x08, 0xd8, 0x10, 0x85, - 0x7e, 0x07, 0x4a, 0x41, 0xbd, 0xe4, 0xa6, 0xe7, 0x6b, 0x1d, 0xde, 0x18, 0x47, 0x72, 0x25, 0x82, - 0x49, 0x96, 0xd5, 0x22, 0x0a, 0x36, 0xa5, 0xad, 0xf8, 0x30, 0x9f, 0x18, 0x6f, 0x8a, 0xf7, 0xdd, - 0x8c, 0x7b, 0xdf, 0xf7, 0xb2, 0x1c, 0x0b, 0xba, 0xc6, 0x64, 0xd6, 0xe3, 0x18, 0x2c, 0x24, 0x47, - 0x7a, 0x62, 0x42, 0x63, 0x85, 0x2d, 0xd3, 0xdf, 0xff, 0x4d, 0x0e, 0x8a, 0xa1, 0xc5, 0x67, 0xc9, - 0x52, 0xd5, 0x49, 0x9d, 0x3b, 0x26, 0x93, 0xc8, 0x8f, 0x92, 0x49, 0x4c, 0x0c, 0xcf, 0x24, 0x82, - 0x4a, 0x56, 0xe1, 0xe9, 0x95, 0x2c, 0x23, 0x93, 0x98, 0x1a, 0x3d, 0x93, 0x98, 0x3e, 0x3e, 0x93, - 0xb0, 0xff, 0xce, 0x02, 0x34, 0x98, 0x36, 0x66, 0x59, 0x28, 0x92, 0xf4, 0xc3, 0x23, 0x9e, 0xd2, - 0xc9, 0xdc, 0x6d, 0xb8, 0x3b, 0xb6, 0x3f, 0x9f, 0x84, 0xf9, 0x5b, 0xce, 0xd8, 0x05, 0x07, 0x0e, - 0x2f, 0x2b, 0xa4, 0x3a, 0xd5, 0x31, 0x52, 0x9d, 0xfb, 0x84, 0xd3, 0xd6, 0x81, 0xde, 0xdf, 0x6b, - 0x9a, 0xf5, 0xe5, 0x5a, 0x7a, 0xb7, 0x27, 0xc3, 0x49, 0x78, 0x18, 0xf4, 0xc8, 0x4a, 0xf2, 0x01, - 0xcc, 0x32, 0xee, 0x3b, 0x0d, 0xae, 0x4a, 0x1a, 0x6c, 0xb9, 0x24, 0xbd, 0xdc, 0x39, 0xdd, 0x7d, - 0xb6, 0x6e, 0x12, 0x71, 0xbc, 0x6f, 0x6a, 0xa5, 0x64, 0x22, 0x73, 0xa5, 0x64, 0x1d, 0x8a, 0xa4, - 0xd3, 0xf1, 0xbe, 0x7f, 0x8f, 0xb4, 0x98, 0x4e, 0x55, 0x43, 0xad, 0xa9, 0x04, 0x04, 0x1c, 0xf5, - 0x41, 0x65, 0x00, 0xa7, 0xe5, 0x7a, 0x3e, 0x95, 0x1c, 0x05, 0xe9, 0x6e, 0x65, 0x35, 0x78, 0x33, - 0x6c, 0xc5, 0x46, 0x0f, 0x54, 0x87, 0x73, 0x8e, 0xcb, 0x68, 0xa3, 0xef, 0xd3, 0xfa, 0x9e, 0xd3, - 0xbb, 0xb7, 0x55, 0x97, 0x5e, 0xe2, 0x40, 0x6a, 0xf3, 0x74, 0xf5, 0x15, 0x2d, 0xec, 0xdc, 0x66, - 0x5a, 0x27, 0x9c, 0xce, 0x8b, 0x2e, 0xc2, 0x8c, 0xe3, 0x36, 0x3a, 0xfd, 0x26, 0xdd, 0x26, 0xbc, - 0xcd, 0x96, 0xa7, 0xe5, 0x30, 0x16, 0x8e, 0x0e, 0xd7, 0x66, 0x36, 0x8d, 0x76, 0x1c, 0xeb, 0x25, - 0xb8, 0xe8, 0x27, 0x06, 0x57, 0x31, 0xe2, 0xba, 0xf1, 0x89, 0xc9, 0x65, 0xf6, 0x4a, 0xa9, 0x25, - 0x41, 0xa6, 0x5a, 0xd2, 0x3f, 0xe4, 0xa0, 0xa0, 0x4a, 0xb9, 0xe8, 0x52, 0xa2, 0x5e, 0xfa, 0xca, - 0x40, 0xbd, 0xb4, 0x94, 0x56, 0xf6, 0xb6, 0xa1, 0xe0, 0x30, 0xd6, 0x8f, 0x9f, 0x6e, 0x9b, 0xb2, - 0x05, 0x6b, 0x8a, 0x2c, 0xbb, 0x78, 0xee, 0xae, 0xd3, 0xd2, 0xc9, 0xf1, 0x75, 0x23, 0x94, 0x8d, - 0xae, 0xdb, 0x3e, 0x0e, 0xef, 0xe3, 0xa2, 0xa8, 0x36, 0xd6, 0x41, 0x84, 0xb7, 0x77, 0xea, 0x77, - 0x3f, 0x54, 0x32, 0x6a, 0x12, 0x11, 0x6b, 0x64, 0x21, 0xc3, 0xeb, 0xf3, 0x5e, 0x9f, 0x4b, 0x45, - 0x39, 0x21, 0x19, 0x77, 0x25, 0x22, 0xd6, 0xc8, 0xf6, 0x67, 0x16, 0xcc, 0xab, 0x35, 0xa8, 0xb5, - 0x69, 0x63, 0xaf, 0xce, 0x69, 0x4f, 0x44, 0x9b, 0x7d, 0x46, 0x59, 0x32, 0xda, 0xfc, 0x88, 0x51, - 0x86, 0x25, 0xc5, 0x98, 0x7d, 0xee, 0xb4, 0x66, 0x6f, 0xff, 0xa3, 0x05, 0x93, 0x32, 0xac, 0xcb, - 0xe2, 0x7f, 0xe2, 0xa5, 0x8e, 0xdc, 0x48, 0xa5, 0x8e, 0x63, 0x8a, 0x50, 0x51, 0x95, 0x65, 0xe2, - 0x69, 0x55, 0x16, 0xfb, 0xa7, 0x16, 0x2c, 0xa5, 0x55, 0xee, 0xb2, 0x0c, 0xff, 0x1d, 0x98, 0xee, - 0x75, 0x08, 0xdf, 0xf5, 0xfc, 0x6e, 0xb2, 0x44, 0xbf, 0xad, 0xdb, 0x71, 0xd8, 0x03, 0xf9, 0x00, - 0x7e, 0x90, 0x22, 0x04, 0xe1, 0xf3, 0xf5, 0xac, 0x27, 0x42, 0xbc, 0xe4, 0x14, 0x2d, 0x56, 0xd8, - 0xc4, 0xb0, 0x21, 0xc5, 0xfe, 0xd3, 0x49, 0x58, 0x94, 0x2c, 0xe3, 0x9e, 0x10, 0xe3, 0xec, 0x50, - 0x0f, 0x5e, 0x92, 0x81, 0xfd, 0xe0, 0xa1, 0xa2, 0x36, 0xed, 0x8a, 0xe6, 0x7f, 0x69, 0x33, 0xb5, - 0xd7, 0x93, 0xa1, 0x14, 0x3c, 0x04, 0x77, 0xf0, 0xa4, 0x80, 0x9f, 0xbf, 0x93, 0xc2, 0x54, 0xb6, - 0xa9, 0x63, 0x95, 0x6d, 0xe8, 0xb9, 0x32, 0xfd, 0x0c, 0xe7, 0xca, 0xa0, 0xaf, 0x2f, 0x66, 0xf2, - 0xf5, 0x7f, 0x95, 0x83, 0xa9, 0x6d, 0xdf, 0x93, 0x15, 0xe0, 0xd3, 0x2f, 0x26, 0xde, 0x85, 0x09, - 0xd6, 0xa3, 0x0d, 0xed, 0xf3, 0xce, 0x8f, 0x66, 0x69, 0x7a, 0x78, 0xf5, 0x1e, 0x6d, 0x54, 0xa7, - 0x85, 0x1b, 0x15, 0xbf, 0xb0, 0x04, 0x32, 0xaa, 0x62, 0xf9, 0x2c, 0x11, 0x78, 0x00, 0xf9, 0xf4, - 0xaa, 0xd8, 0xe7, 0x16, 0x94, 0x74, 0xcf, 0x17, 0xb6, 0xfc, 0xa2, 0xc7, 0x37, 0xa4, 0xfc, 0xf2, - 0xe7, 0xd1, 0x0c, 0xc4, 0xa2, 0xa1, 0xdf, 0x83, 0xc5, 0x9e, 0xef, 0x75, 0x3d, 0x61, 0xa9, 0xdb, - 0x5e, 0xc7, 0x69, 0x38, 0x34, 0xa8, 0xe0, 0x5d, 0x1a, 0x59, 0x9e, 0xc1, 0x7e, 0x50, 0xfd, 0x86, - 0x96, 0xbb, 0xb8, 0x9d, 0xc4, 0xc5, 0x83, 0xa2, 0xec, 0xff, 0xb2, 0x60, 0x36, 0xb6, 0xf6, 0xa8, - 0x01, 0xd0, 0xf0, 0xdc, 0xa6, 0xc3, 0xc3, 0x5b, 0xd2, 0xd2, 0x85, 0xf5, 0xd1, 0x56, 0xb5, 0x16, - 0xf0, 0x45, 0x4a, 0x17, 0x36, 0x31, 0x6c, 0xc0, 0xa2, 0xf7, 0x82, 0x07, 0x0b, 0xf1, 0x20, 0x46, - 0x3d, 0x58, 0x78, 0x72, 0xb8, 0x36, 0xa3, 0xc7, 0x64, 0x3e, 0x60, 0xc8, 0x72, 0x75, 0xff, 0xf7, - 0x39, 0x28, 0x86, 0xf3, 0x7f, 0x0e, 0x66, 0xf4, 0x51, 0xcc, 0x8c, 0xde, 0xcb, 0xb8, 0x73, 0xd2, - 0x90, 0xc2, 0x98, 0xc4, 0x30, 0xa6, 0xc7, 0x09, 0x63, 0xca, 0xaa, 0x12, 0xc7, 0x98, 0xd3, 0xbf, - 0xa8, 0xcd, 0x57, 0x7d, 0x9f, 0x83, 0x41, 0xdd, 0x8b, 0x1b, 0xd4, 0x7a, 0xc6, 0xd9, 0x0c, 0x31, - 0xa9, 0x1f, 0x5a, 0x30, 0x9f, 0x30, 0x02, 0xf4, 0x1a, 0x4c, 0xca, 0x92, 0x8d, 0xd6, 0xaf, 0x90, - 0x51, 0x27, 0xf8, 0x92, 0x86, 0xb6, 0x61, 0x89, 0xf4, 0xb9, 0x17, 0xf2, 0xde, 0x70, 0xc9, 0x4e, - 0x87, 0xaa, 0xac, 0x7d, 0xba, 0xfa, 0x0b, 0x9a, 0x67, 0xa9, 0x92, 0xd2, 0x07, 0xa7, 0x72, 0xda, - 0x3f, 0xce, 0x01, 0x0a, 0x1b, 0xb3, 0x94, 0x3a, 0x1f, 0xc3, 0xd4, 0xae, 0x2a, 0xca, 0x3c, 0x5b, - 0xad, 0xba, 0x5a, 0x32, 0xcb, 0xf5, 0x01, 0x26, 0xfa, 0xce, 0xc9, 0xe8, 0x11, 0x0c, 0xea, 0x10, - 0x7a, 0x08, 0xb0, 0xeb, 0xb8, 0x0e, 0x6b, 0x8f, 0x79, 0xab, 0x26, 0x4f, 0xf3, 0x9b, 0x21, 0x02, - 0x36, 0xd0, 0xec, 0x4f, 0x73, 0x86, 0x7e, 0x4a, 0x77, 0x39, 0xd2, 0xbe, 0xbe, 0x15, 0x5f, 0xcc, - 0xe2, 0xe0, 0x3d, 0x86, 0xb1, 0x30, 0x13, 0xfb, 0xc4, 0x0f, 0x4a, 0xaa, 0x97, 0x33, 0x2e, 0xcb, - 0x7d, 0xe2, 0x3b, 0x62, 0xe3, 0xa3, 0x2d, 0xbd, 0x4f, 0x7c, 0x86, 0x25, 0x24, 0xfa, 0xb6, 0x18, - 0x2a, 0xed, 0x05, 0x2e, 0x34, 0xb3, 0x4f, 0xe0, 0xb4, 0x67, 0xce, 0x8f, 0xf6, 0x18, 0x56, 0x80, - 0xf6, 0xbf, 0x15, 0x0c, 0x85, 0xd7, 0x5e, 0xfb, 0x0e, 0xa0, 0x0e, 0x61, 0xfc, 0x36, 0x71, 0x9b, - 0x42, 0x3d, 0xe9, 0xae, 0x4f, 0x59, 0x5b, 0x47, 0x68, 0x2b, 0x1a, 0x05, 0x6d, 0x0d, 0xf4, 0xc0, - 0x29, 0x5c, 0xe8, 0x52, 0xdc, 0x39, 0xaf, 0x25, 0x9d, 0xf3, 0x5c, 0x64, 0x6d, 0xe3, 0xb9, 0x67, - 0x53, 0xdd, 0x27, 0x4f, 0x41, 0xdd, 0x7f, 0x17, 0x16, 0x77, 0x93, 0xf7, 0x5a, 0xfa, 0x96, 0xfb, - 0xf2, 0x98, 0xd7, 0x62, 0xd5, 0x73, 0x47, 0xd1, 0x65, 0x48, 0xd4, 0x8c, 0x07, 0x05, 0x21, 0x2f, - 0x78, 0xcf, 0x25, 0xb3, 0x4f, 0x55, 0x58, 0x18, 0xd9, 0xe4, 0x12, 0x79, 0x6b, 0xf2, 0x25, 0x97, - 0x82, 0xc4, 0x31, 0x01, 0x09, 0x13, 0x2c, 0x9c, 0xa4, 0x09, 0xa2, 0x4b, 0x50, 0x6a, 0xf4, 0x7d, - 0x9f, 0xba, 0x5c, 0x0c, 0x47, 0x86, 0xb2, 0xf9, 0xa8, 0x40, 0x5c, 0x8b, 0x48, 0xd8, 0xec, 0x27, - 0xd4, 0xd1, 0xf8, 0xb3, 0xc2, 0x39, 0xed, 0xf6, 0xb8, 0x2c, 0x4d, 0xe5, 0x23, 0x75, 0xac, 0x0d, - 0xf4, 0xc0, 0x29, 0x5c, 0xe8, 0x63, 0x69, 0xf3, 0x9c, 0xca, 0x7c, 0xe5, 0xd9, 0xf3, 0xf2, 0xa2, - 0xf6, 0x17, 0x5c, 0xf9, 0x0b, 0x4e, 0xed, 0x3f, 0x8b, 0xb9, 0x99, 0xd1, 0xaa, 0x05, 0x2b, 0x90, - 0x23, 0x2c, 0x59, 0xdf, 0xad, 0x30, 0x9c, 0x23, 0x0c, 0x7d, 0x07, 0x26, 0x7d, 0xca, 0xfd, 0x03, - 0xed, 0x0d, 0xaf, 0x8c, 0x61, 0xf9, 0x58, 0xf0, 0xab, 0xa1, 0xca, 0x9f, 0x58, 0x21, 0x1a, 0x45, - 0x8a, 0xfc, 0xa9, 0x15, 0x29, 0xaa, 0xc6, 0x19, 0x16, 0x8e, 0x45, 0x64, 0x56, 0x44, 0x6d, 0x88, - 0x5a, 0x96, 0x7c, 0x74, 0xd2, 0xeb, 0x8d, 0x62, 0x38, 0xec, 0x61, 0xfb, 0x10, 0x85, 0x9f, 0xc1, - 0x63, 0x53, 0xf4, 0x58, 0x07, 0x49, 0x56, 0x96, 0x07, 0x8e, 0x03, 0x30, 0xc3, 0x82, 0x25, 0xfb, - 0xc7, 0x16, 0x9c, 0x4b, 0xed, 0x1d, 0x7a, 0xf9, 0xdc, 0x29, 0x7a, 0x79, 0xeb, 0xa4, 0xbd, 0xfc, - 0x43, 0x63, 0x09, 0x83, 0x21, 0x9c, 0xd4, 0x0b, 0xf1, 0x1f, 0xe5, 0x60, 0x01, 0xd3, 0x9e, 0x17, - 0x2b, 0x78, 0x6c, 0x07, 0x4f, 0xc6, 0x32, 0x44, 0x08, 0x89, 0xb2, 0x7a, 0x75, 0x2a, 0xf6, 0x56, - 0xec, 0xdb, 0x30, 0x29, 0x6b, 0x11, 0x3a, 0xa6, 0xb9, 0x9c, 0xe1, 0x1a, 0x34, 0x86, 0x2a, 0xed, - 0x40, 0x15, 0x75, 0x14, 0xa0, 0x40, 0x96, 0x37, 0x98, 0xda, 0x0c, 0x2e, 0x67, 0xb8, 0x0b, 0x1d, - 0x44, 0x96, 0xcd, 0x58, 0x01, 0xda, 0x9f, 0xe5, 0x40, 0x45, 0x13, 0xcf, 0x21, 0x6b, 0xf8, 0x8d, - 0x58, 0xd6, 0x30, 0x62, 0x38, 0x2c, 0x07, 0x37, 0x34, 0x63, 0x48, 0x46, 0x7a, 0xe7, 0xb3, 0x80, - 0x3e, 0x3d, 0x5b, 0xf8, 0x67, 0x0b, 0x8a, 0xb2, 0xdf, 0x73, 0xc8, 0x14, 0xb6, 0xe3, 0x99, 0xc2, - 0xdb, 0x19, 0x66, 0x31, 0x24, 0x4b, 0xf8, 0x34, 0xaf, 0x47, 0x1f, 0xc6, 0x91, 0x6d, 0xe2, 0x37, - 0x75, 0x84, 0x14, 0x59, 0xa0, 0x68, 0xc4, 0x8a, 0x86, 0x7e, 0x1b, 0x16, 0x7c, 0x75, 0x29, 0x4e, - 0x9b, 0x37, 0xc3, 0x70, 0x25, 0x9f, 0xf9, 0xd6, 0x5a, 0xdf, 0xac, 0x47, 0x95, 0x32, 0x9c, 0x40, - 0xc5, 0x03, 0x72, 0x44, 0x08, 0xd3, 0x4b, 0xfa, 0x32, 0x7d, 0xb4, 0x5f, 0x1e, 0xd3, 0x71, 0xaa, - 0x10, 0x66, 0xa0, 0x19, 0x0f, 0x0a, 0x42, 0x6d, 0x98, 0x31, 0xdf, 0xdb, 0x68, 0x5d, 0xba, 0x90, - 0xfd, 0x61, 0x8f, 0xba, 0x19, 0x31, 0x5b, 0x70, 0x0c, 0xd9, 0x3e, 0x2c, 0x40, 0xc9, 0x50, 0xbe, - 0x44, 0xf5, 0x61, 0xf6, 0x74, 0xaa, 0x0f, 0xe9, 0xc1, 0x72, 0x69, 0xac, 0x60, 0xf9, 0x7c, 0x3c, - 0x58, 0xfe, 0x66, 0x32, 0x58, 0x06, 0x39, 0xbb, 0x58, 0xa0, 0xcc, 0x60, 0x4e, 0x47, 0x8d, 0xc1, - 0xc3, 0xa9, 0x4c, 0xe9, 0xc7, 0x60, 0x6c, 0x8a, 0x8e, 0x0e, 0xd7, 0x82, 0x67, 0x0f, 0xc1, 0x83, - 0xa9, 0x84, 0x08, 0x74, 0x3d, 0x14, 0x5a, 0xef, 0x77, 0xbb, 0xc4, 0x3f, 0x58, 0x9e, 0x91, 0x03, - 0x0e, 0xcb, 0x92, 0x37, 0x63, 0x54, 0x9c, 0xe8, 0x8d, 0xb6, 0xa1, 0xa0, 0x82, 0x4e, 0xfd, 0x18, - 0xe7, 0x9d, 0x2c, 0xf1, 0xac, 0x8a, 0x33, 0xd4, 0x6f, 0xac, 0x71, 0xcc, 0x7c, 0xa1, 0x78, 0x4c, - 0xbe, 0x70, 0x07, 0x90, 0xb7, 0x23, 0x23, 0x9a, 0xe6, 0x2d, 0xf5, 0x29, 0x95, 0xd0, 0xca, 0x42, - 0x3c, 0x9c, 0xbc, 0x3b, 0xd0, 0x03, 0xa7, 0x70, 0x09, 0xab, 0xd6, 0x41, 0x66, 0x68, 0x0a, 0x3a, - 0x37, 0xc8, 0x1a, 0xa8, 0x45, 0x69, 0xc8, 0x92, 0xb0, 0xea, 0x5a, 0x02, 0x15, 0x0f, 0xc8, 0x41, - 0xdf, 0x83, 0x59, 0xa1, 0x42, 0x91, 0x60, 0x78, 0x46, 0xc1, 0x8b, 0x47, 0x87, 0x6b, 0xb3, 0x5b, - 0x26, 0x24, 0x8e, 0x4b, 0xb0, 0xff, 0x2f, 0x07, 0x31, 0xfb, 0x43, 0x3f, 0xb4, 0x60, 0x91, 0x24, - 0xbe, 0x03, 0x0a, 0xc2, 0x97, 0x5f, 0xcb, 0xf6, 0x71, 0xd6, 0xc0, 0x67, 0x44, 0x51, 0xf1, 0x31, - 0xd9, 0x85, 0xe1, 0x41, 0xa1, 0xe8, 0x8f, 0x2d, 0x38, 0x4b, 0x06, 0x3f, 0xf4, 0xd2, 0xe7, 0xe1, - 0xd5, 0xb1, 0xbf, 0x14, 0xab, 0xbe, 0x7c, 0x74, 0xb8, 0x96, 0xf6, 0x09, 0x1c, 0x4e, 0x13, 0x87, - 0xbe, 0x0b, 0x13, 0xc4, 0x6f, 0x05, 0x89, 0x7a, 0x76, 0xb1, 0xc1, 0xf7, 0x7b, 0xd1, 0x81, 0x5c, - 0xf1, 0x5b, 0x0c, 0x4b, 0x50, 0xfb, 0x27, 0x79, 0x58, 0x48, 0x3e, 0x75, 0xd4, 0xaf, 0x43, 0x26, - 0x52, 0x5f, 0x87, 0xbc, 0x06, 0x93, 0xa4, 0xc1, 0xc3, 0xa7, 0x1a, 0xd1, 0xf3, 0x5e, 0xd1, 0x88, - 0x15, 0x0d, 0x3d, 0x80, 0x22, 0xe3, 0xc4, 0xe7, 0x22, 0x7f, 0xd3, 0x29, 0x74, 0xe6, 0xa7, 0xcc, - 0xf5, 0x00, 0x00, 0x47, 0x58, 0xe8, 0x4a, 0xdc, 0x9d, 0xd9, 0x49, 0x77, 0xb6, 0x68, 0xce, 0x65, - 0xdc, 0xf4, 0xbf, 0x0b, 0x25, 0x63, 0x1f, 0xf4, 0xe9, 0x72, 0x2d, 0xf3, 0xba, 0x47, 0x6a, 0x37, - 0xaf, 0x3e, 0x02, 0x8c, 0x28, 0x26, 0x7e, 0x94, 0x1f, 0xcb, 0xd5, 0x7a, 0xa6, 0xfc, 0x58, 0x2e, - 0x97, 0x81, 0x66, 0xef, 0xc1, 0x6c, 0xec, 0xc5, 0x92, 0x10, 0x16, 0xbc, 0xd2, 0x1a, 0xff, 0xab, - 0xb8, 0xfb, 0x21, 0x02, 0x36, 0xd0, 0x64, 0x55, 0xfb, 0x01, 0xf1, 0x69, 0xdb, 0xeb, 0x33, 0xfa, - 0xa2, 0x56, 0xb5, 0xc3, 0x01, 0x9e, 0x74, 0x55, 0x3b, 0x02, 0x3e, 0xbe, 0xaa, 0x1d, 0xf6, 0x7d, - 0x61, 0xab, 0xda, 0xe1, 0x08, 0x87, 0xc4, 0xab, 0xff, 0x9b, 0x33, 0x66, 0x11, 0x8f, 0x59, 0x73, - 0x4f, 0x89, 0x59, 0x1f, 0xc1, 0xb4, 0xe3, 0x72, 0xea, 0xef, 0x93, 0x8e, 0x2e, 0x3f, 0x8c, 0x38, - 0xd5, 0x8d, 0xbe, 0xaf, 0x23, 0xb6, 0x60, 0xaa, 0x9b, 0x1a, 0x07, 0x87, 0x88, 0xa8, 0x03, 0xe7, - 0x82, 0x7a, 0x97, 0x4f, 0x49, 0x54, 0x6f, 0xd7, 0x77, 0xd6, 0xef, 0x07, 0x17, 0xa6, 0x37, 0xd3, - 0x3a, 0x3d, 0x19, 0x46, 0xc0, 0xe9, 0xa0, 0x88, 0xc1, 0x2c, 0x33, 0x92, 0xb5, 0xe0, 0x90, 0x1a, - 0xb1, 0x56, 0x98, 0xcc, 0x6f, 0x8d, 0x8b, 0x6e, 0x13, 0x14, 0xc7, 0x65, 0xd8, 0xff, 0x91, 0x87, - 0xf9, 0x84, 0xa6, 0x25, 0x82, 0xd2, 0xe2, 0xf3, 0x0c, 0x4a, 0x0b, 0x63, 0x05, 0xa5, 0xe9, 0xf1, - 0xd2, 0xc4, 0x58, 0xf1, 0xd2, 0x07, 0x2a, 0x66, 0xd1, 0x3b, 0xb7, 0xb9, 0xa1, 0xdf, 0x05, 0x86, - 0xab, 0xb9, 0x65, 0x12, 0x71, 0xbc, 0xaf, 0x3c, 0xe1, 0x9b, 0x83, 0x1f, 0xc1, 0xe9, 0x80, 0xeb, - 0x6a, 0xd6, 0x87, 0x1d, 0x21, 0x80, 0x3a, 0xe1, 0x53, 0x08, 0x38, 0x4d, 0x5c, 0xf5, 0xce, 0x17, - 0x5f, 0xaf, 0x9e, 0xf9, 0xf2, 0xeb, 0xd5, 0x33, 0x5f, 0x7d, 0xbd, 0x7a, 0xe6, 0x07, 0x47, 0xab, - 0xd6, 0x17, 0x47, 0xab, 0xd6, 0x97, 0x47, 0xab, 0xd6, 0x57, 0x47, 0xab, 0xd6, 0x7f, 0x1f, 0xad, - 0x5a, 0x7f, 0xf9, 0xd3, 0xd5, 0x33, 0x0f, 0x5f, 0x1f, 0xe5, 0x3f, 0x21, 0xfc, 0x7f, 0x00, 0x00, - 0x00, 0xff, 0xff, 0x5f, 0x0e, 0x38, 0x51, 0x30, 0x41, 0x00, 0x00, + 0x76, 0x57, 0xcf, 0x90, 0x43, 0xce, 0x1b, 0x52, 0x24, 0x4b, 0x94, 0xcd, 0xe5, 0xc6, 0xa4, 0xd2, + 0x6b, 0x18, 0x76, 0x6c, 0x0f, 0x23, 0xd9, 0xb2, 0x64, 0x39, 0x51, 0xc0, 0x21, 0x25, 0x99, 0x0a, + 0xd7, 0x62, 0x6a, 0x64, 0x79, 0x57, 0x6b, 0xc3, 0x29, 0xce, 0x14, 0x67, 0x3a, 0x9c, 0xe9, 0x9e, + 0xad, 0xaa, 0xe1, 0x9a, 0x49, 0x90, 0x6c, 0xbe, 0x80, 0x45, 0xbe, 0xb0, 0x07, 0x03, 0xde, 0x00, + 0x09, 0x10, 0x24, 0xc7, 0x45, 0xf2, 0x0f, 0xe4, 0xe0, 0x43, 0x2e, 0x46, 0xb0, 0x08, 0x8c, 0x24, + 0x07, 0x1f, 0x16, 0x44, 0xcc, 0x05, 0x72, 0xdc, 0x5b, 0x2e, 0x02, 0x02, 0x04, 0xf5, 0xd1, 0xdd, + 0xd5, 0x3d, 0x3d, 0xe2, 0xf4, 0x88, 0x14, 0x8c, 0xbd, 0x0d, 0xeb, 0xd5, 0xfb, 0xbd, 0xfa, 0x78, + 0xef, 0xd5, 0x7b, 0xaf, 0xaa, 0x09, 0xaf, 0xb7, 0x3c, 0xd1, 0xee, 0xef, 0x56, 0x1b, 0x41, 0x77, + 0x8d, 0xec, 0xf7, 0x3d, 0x71, 0xb8, 0xb6, 0x4f, 0x58, 0x2b, 0x58, 0x23, 0x3d, 0x6f, 0xed, 0xe0, + 0x32, 0xe9, 0xf4, 0xda, 0xe4, 0xf2, 0x5a, 0x8b, 0xfa, 0x94, 0x11, 0x41, 0x9b, 0xd5, 0x1e, 0x0b, + 0x44, 0x80, 0x9e, 0x8f, 0xb9, 0xaa, 0x9a, 0xab, 0xaa, 0xb8, 0xaa, 0xa4, 0xe7, 0x55, 0x43, 0xae, + 0xe5, 0x57, 0x2d, 0xec, 0x56, 0xd0, 0x0a, 0xd6, 0x14, 0xf3, 0x6e, 0x7f, 0x4f, 0xfd, 0xa5, 0xfe, + 0x50, 0xbf, 0x34, 0xe8, 0xf2, 0xdb, 0xfb, 0xd7, 0x79, 0xd5, 0x53, 0x92, 0xe9, 0x47, 0x82, 0xfa, + 0xdc, 0x0b, 0x7c, 0xfe, 0x2a, 0xe9, 0x79, 0x9c, 0xb2, 0x03, 0xca, 0xd6, 0x7a, 0xfb, 0x2d, 0x49, + 0xe3, 0xc9, 0x0e, 0x6b, 0x07, 0x03, 0xc3, 0x5b, 0x7e, 0x3d, 0x46, 0xea, 0x92, 0x46, 0xdb, 0xf3, + 0x29, 0x3b, 0x8c, 0xd9, 0xbb, 0x54, 0x90, 0x2c, 0xae, 0xb5, 0x61, 0x5c, 0xac, 0xef, 0x0b, 0xaf, + 0x4b, 0x07, 0x18, 0xde, 0x38, 0x89, 0x81, 0x37, 0xda, 0xb4, 0x4b, 0xd2, 0x7c, 0xee, 0xfb, 0x70, + 0x61, 0xdd, 0x27, 0x9d, 0x43, 0xee, 0x71, 0xdc, 0xf7, 0xd7, 0x59, 0xab, 0xdf, 0xa5, 0xbe, 0x40, + 0x97, 0x60, 0xc2, 0x27, 0x5d, 0xba, 0xe4, 0x5c, 0x72, 0x5e, 0x2c, 0xd7, 0x66, 0x3e, 0x3b, 0x5a, + 0x3d, 0x77, 0x7c, 0xb4, 0x3a, 0xf1, 0x0e, 0xe9, 0x52, 0xac, 0x28, 0xe8, 0x1b, 0x30, 0x79, 0x40, + 0x3a, 0x7d, 0xba, 0x54, 0x50, 0x5d, 0x66, 0x4d, 0x97, 0xc9, 0x07, 0xb2, 0x11, 0x6b, 0x9a, 0xfb, + 0x27, 0xc5, 0x04, 0xfc, 0x37, 0xa9, 0x20, 0x4d, 0x22, 0x08, 0xea, 0x42, 0xa9, 0x43, 0x76, 0x69, + 0x87, 0x2f, 0x39, 0x97, 0x8a, 0x2f, 0x56, 0xae, 0xdc, 0xaa, 0x8e, 0xb2, 0x89, 0xd5, 0x0c, 0xa8, + 0xea, 0xb6, 0xc2, 0xb9, 0xe5, 0x0b, 0x76, 0x58, 0x3b, 0x6f, 0x06, 0x51, 0xd2, 0x8d, 0xd8, 0x08, + 0x41, 0x7f, 0xe4, 0x40, 0x85, 0xf8, 0x7e, 0x20, 0x88, 0x90, 0xdb, 0xb4, 0x54, 0x50, 0x42, 0xef, + 0x8e, 0x2f, 0x74, 0x3d, 0x06, 0xd3, 0x92, 0x2f, 0x18, 0xc9, 0x15, 0x8b, 0x82, 0x6d, 0x99, 0xcb, + 0x6f, 0x42, 0xc5, 0x1a, 0x2a, 0x9a, 0x87, 0xe2, 0x3e, 0x3d, 0xd4, 0xeb, 0x8b, 0xe5, 0x4f, 0xb4, + 0x98, 0x58, 0x50, 0xb3, 0x82, 0x37, 0x0a, 0xd7, 0x9d, 0xe5, 0x9b, 0x30, 0x9f, 0x16, 0x98, 0x87, + 0xdf, 0xfd, 0x6b, 0x07, 0x16, 0xad, 0x59, 0x60, 0xba, 0x47, 0x19, 0xf5, 0x1b, 0x14, 0xad, 0x41, + 0x59, 0xee, 0x25, 0xef, 0x91, 0x46, 0xb8, 0xd5, 0x0b, 0x66, 0x22, 0xe5, 0x77, 0x42, 0x02, 0x8e, + 0xfb, 0x44, 0x6a, 0x51, 0x78, 0x9c, 0x5a, 0xf4, 0xda, 0x84, 0xd3, 0xa5, 0x62, 0x52, 0x2d, 0x76, + 0x64, 0x23, 0xd6, 0x34, 0xf7, 0xd7, 0xe1, 0x6b, 0xe1, 0x78, 0xee, 0xd3, 0x6e, 0xaf, 0x43, 0x04, + 0x8d, 0x07, 0x75, 0xa2, 0xea, 0xb9, 0xfb, 0x30, 0xbb, 0xde, 0xeb, 0xb1, 0xe0, 0x80, 0x36, 0xeb, + 0x82, 0xb4, 0x28, 0x7a, 0x08, 0x40, 0x4c, 0xc3, 0xba, 0x50, 0x8c, 0x95, 0x2b, 0xbf, 0x52, 0xd5, + 0x16, 0x51, 0xb5, 0x2d, 0xa2, 0xda, 0xdb, 0x6f, 0xc9, 0x06, 0x5e, 0x95, 0x86, 0x57, 0x3d, 0xb8, + 0x5c, 0xbd, 0xef, 0x75, 0x69, 0xed, 0xfc, 0xf1, 0xd1, 0x2a, 0xac, 0x47, 0x08, 0xd8, 0x42, 0x73, + 0xff, 0xd8, 0x81, 0x8b, 0xeb, 0xac, 0x15, 0x6c, 0x6c, 0xae, 0xf7, 0x7a, 0x6f, 0x53, 0xd2, 0x11, + 0xed, 0xba, 0x20, 0xa2, 0xcf, 0xd1, 0x4d, 0x28, 0x71, 0xf5, 0xcb, 0x0c, 0xf5, 0x85, 0x50, 0xfb, + 0x34, 0xfd, 0xd1, 0xd1, 0xea, 0x62, 0x06, 0x23, 0xc5, 0x86, 0x0b, 0xbd, 0x04, 0x53, 0x5d, 0xca, + 0x39, 0x69, 0x85, 0xeb, 0x39, 0x67, 0x00, 0xa6, 0xbe, 0xa9, 0x9b, 0x71, 0x48, 0x77, 0xff, 0xad, + 0x00, 0x73, 0x11, 0x96, 0x11, 0x7f, 0x06, 0x9b, 0xd7, 0x87, 0x99, 0xb6, 0x35, 0x43, 0xb5, 0x87, + 0x95, 0x2b, 0x6f, 0x8d, 0x68, 0x27, 0x59, 0x8b, 0x54, 0x5b, 0x34, 0x62, 0x66, 0xec, 0x56, 0x9c, + 0x10, 0x83, 0xba, 0x00, 0xfc, 0xd0, 0x6f, 0x18, 0xa1, 0x13, 0x4a, 0xe8, 0x9b, 0x39, 0x85, 0xd6, + 0x23, 0x80, 0x1a, 0x32, 0x22, 0x21, 0x6e, 0xc3, 0x96, 0x00, 0xf7, 0x9f, 0x1d, 0xb8, 0x90, 0xc1, + 0x87, 0x7e, 0x2d, 0xb5, 0x9f, 0xcf, 0x0f, 0xec, 0x27, 0x1a, 0x60, 0x8b, 0x77, 0xf3, 0x15, 0x98, + 0x66, 0xf4, 0xc0, 0x93, 0xe7, 0x80, 0x59, 0xe1, 0x79, 0xc3, 0x3f, 0x8d, 0x4d, 0x3b, 0x8e, 0x7a, + 0xa0, 0x97, 0xa1, 0x1c, 0xfe, 0x96, 0xcb, 0x5c, 0x94, 0xa6, 0x22, 0x37, 0x2e, 0xec, 0xca, 0x71, + 0x4c, 0x77, 0xff, 0x10, 0x26, 0x37, 0xda, 0x84, 0x09, 0xa9, 0x31, 0x8c, 0xf6, 0x82, 0x77, 0xf1, + 0xb6, 0x19, 0x62, 0xa4, 0x31, 0x58, 0x37, 0xe3, 0x90, 0x3e, 0xc2, 0x66, 0xbf, 0x04, 0x53, 0x07, + 0x94, 0xa9, 0xf1, 0x16, 0x93, 0x60, 0x0f, 0x74, 0x33, 0x0e, 0xe9, 0xee, 0x7f, 0x3a, 0xb0, 0xa8, + 0x46, 0xb0, 0xe9, 0xf1, 0x46, 0x70, 0x40, 0xd9, 0x21, 0xa6, 0xbc, 0xdf, 0x39, 0xe5, 0x01, 0x6d, + 0xc2, 0x3c, 0xa7, 0xdd, 0x03, 0xca, 0x36, 0x02, 0x9f, 0x0b, 0x46, 0x3c, 0x5f, 0x98, 0x91, 0x2d, + 0x99, 0xde, 0xf3, 0xf5, 0x14, 0x1d, 0x0f, 0x70, 0xa0, 0x17, 0x61, 0xda, 0x0c, 0x5b, 0xaa, 0x92, + 0x5c, 0xd8, 0x19, 0xb9, 0x07, 0x66, 0x4e, 0x1c, 0x47, 0x54, 0xf7, 0x7f, 0x1c, 0x58, 0x50, 0xb3, + 0xaa, 0xf7, 0x77, 0x79, 0x83, 0x79, 0x3d, 0xe9, 0x5e, 0xbf, 0x8a, 0x53, 0xba, 0x09, 0xe7, 0x9b, + 0xe1, 0xc2, 0x6f, 0x7b, 0x5d, 0x4f, 0x28, 0x1b, 0x99, 0xac, 0x3d, 0x63, 0x30, 0xce, 0x6f, 0x26, + 0xa8, 0x38, 0xd5, 0xdb, 0xfd, 0xdb, 0x22, 0x5c, 0x08, 0xbb, 0xd0, 0xe6, 0x3a, 0x13, 0xde, 0x1e, + 0x69, 0x08, 0x8e, 0x9a, 0x30, 0xd3, 0x8c, 0x9b, 0x85, 0xb1, 0xbc, 0x3c, 0x8e, 0x33, 0xb2, 0x6e, + 0x0b, 0x5e, 0xe0, 0x04, 0x2a, 0x7a, 0x0f, 0x8a, 0x2d, 0x4f, 0x98, 0x83, 0xfe, 0xfa, 0x68, 0x66, + 0x7d, 0xc7, 0x4b, 0xab, 0x5a, 0xad, 0x62, 0x44, 0x15, 0xef, 0x78, 0x02, 0x4b, 0x44, 0xb4, 0x0b, + 0x25, 0xaf, 0x4b, 0x5a, 0x34, 0x3c, 0xcf, 0x6f, 0x8c, 0x86, 0xbd, 0x25, 0x79, 0xd2, 0xe8, 0x51, + 0xe4, 0xa0, 0xa8, 0x1c, 0x1b, 0x64, 0x29, 0xa3, 0x21, 0x55, 0x44, 0x1b, 0xe9, 0xc8, 0x32, 0xb2, + 0x8c, 0x25, 0x96, 0xa1, 0xa8, 0x1c, 0x1b, 0x64, 0xf7, 0x8b, 0x02, 0xcc, 0xc7, 0xeb, 0xb7, 0x11, + 0x74, 0xbb, 0x9e, 0x40, 0xcb, 0x50, 0xf0, 0x9a, 0x46, 0x03, 0xc1, 0x30, 0x16, 0xb6, 0x36, 0x71, + 0xc1, 0x6b, 0xa2, 0x17, 0xa0, 0xb4, 0xcb, 0x88, 0xdf, 0x68, 0x1b, 0xcd, 0x8b, 0x80, 0x6b, 0xaa, + 0x15, 0x1b, 0x2a, 0x7a, 0x0e, 0x8a, 0x82, 0xb4, 0x8c, 0xc2, 0x45, 0xeb, 0x77, 0x9f, 0xb4, 0xb0, + 0x6c, 0x97, 0x9a, 0xce, 0xfb, 0xbb, 0xbf, 0x43, 0x1b, 0x7a, 0xe7, 0x2d, 0x4d, 0xaf, 0xeb, 0x66, + 0x1c, 0xd2, 0xa5, 0x44, 0xd2, 0x17, 0xed, 0x80, 0x2d, 0x4d, 0x26, 0x25, 0xae, 0xab, 0x56, 0x6c, + 0xa8, 0xf2, 0x4c, 0x6a, 0xa8, 0xf1, 0x0b, 0xca, 0x96, 0x4a, 0xc9, 0x33, 0x69, 0x23, 0x24, 0xe0, + 0xb8, 0x0f, 0xfa, 0x00, 0x2a, 0x0d, 0x46, 0x89, 0x08, 0xd8, 0x26, 0x11, 0x74, 0x69, 0x2a, 0xb7, + 0x06, 0xce, 0xc9, 0xa0, 0x6b, 0x23, 0x86, 0xc0, 0x36, 0x9e, 0xfb, 0x73, 0x07, 0x96, 0xe2, 0xa5, + 0x55, 0x7b, 0x1b, 0x07, 0x1a, 0x66, 0x79, 0x9c, 0x21, 0xcb, 0xf3, 0x02, 0x94, 0x9a, 0x5e, 0x8b, + 0x72, 0x91, 0x5e, 0xe5, 0x4d, 0xd5, 0x8a, 0x0d, 0x15, 0x5d, 0x01, 0x68, 0x79, 0xc2, 0x38, 0x07, + 0xb3, 0xd8, 0xd1, 0x11, 0x74, 0x27, 0xa2, 0x60, 0xab, 0x17, 0x7a, 0x0f, 0xca, 0x6a, 0x98, 0x63, + 0x9a, 0x9d, 0x3a, 0x2a, 0x36, 0x42, 0x00, 0x1c, 0x63, 0xb9, 0x9f, 0x4f, 0xc0, 0xd4, 0x6d, 0x46, + 0xbd, 0x56, 0x5b, 0xa0, 0xdf, 0x86, 0xe9, 0xae, 0x09, 0x58, 0x4d, 0x4c, 0xf4, 0xab, 0xa3, 0xc9, + 0xb8, 0xa7, 0x36, 0x5d, 0x06, 0xbb, 0xf1, 0x44, 0xe2, 0x36, 0x1c, 0xa1, 0xca, 0x60, 0x8f, 0x74, + 0x3c, 0xc2, 0xd5, 0xbe, 0x59, 0xc1, 0xde, 0xba, 0x6c, 0xc4, 0x9a, 0x86, 0xbe, 0x03, 0xa5, 0x80, + 0x79, 0x2d, 0xcf, 0x5f, 0x2a, 0xab, 0x41, 0xbc, 0x36, 0x9a, 0x09, 0x99, 0x59, 0xdc, 0x53, 0xac, + 0xf1, 0xe2, 0xeb, 0xbf, 0xb1, 0x81, 0x44, 0x0f, 0x61, 0x4a, 0x2b, 0x53, 0x68, 0xa0, 0x6b, 0x23, + 0x3b, 0x18, 0xad, 0x8f, 0xb1, 0xd2, 0xeb, 0xbf, 0x39, 0x0e, 0x01, 0x51, 0x3d, 0xf2, 0x2f, 0x13, + 0x0a, 0xfa, 0xe5, 0x1c, 0xfe, 0x65, 0xa8, 0x43, 0xa9, 0x47, 0x0e, 0x65, 0x32, 0x0f, 0xa8, 0x72, + 0x19, 0xc3, 0x3c, 0x88, 0x5c, 0x62, 0x13, 0xb9, 0x94, 0xc6, 0x58, 0x62, 0x13, 0x36, 0x9d, 0x4f, + 0x86, 0x3b, 0x61, 0x60, 0xe3, 0x7e, 0x5c, 0x84, 0x05, 0xd3, 0x73, 0x23, 0xe8, 0x74, 0x68, 0x43, + 0x1d, 0x93, 0xda, 0x3f, 0x15, 0x33, 0xfd, 0x93, 0x07, 0x93, 0x9e, 0xa0, 0xdd, 0x30, 0xb9, 0xab, + 0xe5, 0x1a, 0x4d, 0x2c, 0xa3, 0xba, 0x25, 0x41, 0x74, 0x7e, 0x15, 0xed, 0x92, 0xe9, 0x85, 0xb5, + 0x04, 0xf4, 0x67, 0x0e, 0x5c, 0x38, 0xa0, 0xcc, 0xdb, 0xf3, 0x1a, 0x2a, 0x3b, 0x7a, 0xdb, 0xe3, + 0x22, 0x60, 0x87, 0xe6, 0x44, 0x78, 0x63, 0x34, 0xc9, 0x0f, 0x2c, 0x80, 0x2d, 0x7f, 0x2f, 0xa8, + 0x7d, 0xdd, 0x48, 0xbb, 0xf0, 0x60, 0x10, 0x1a, 0x67, 0xc9, 0x5b, 0xee, 0x01, 0xc4, 0xa3, 0xcd, + 0x48, 0xce, 0xb6, 0xed, 0xe4, 0x6c, 0xe4, 0x81, 0x85, 0x93, 0x0d, 0x5d, 0x96, 0x9d, 0xd4, 0x7d, + 0xea, 0x40, 0xc5, 0xd0, 0xb7, 0x3d, 0x2e, 0xd0, 0xfb, 0x03, 0xd6, 0x5e, 0x1d, 0xcd, 0xda, 0x25, + 0xb7, 0xb2, 0xf5, 0x28, 0x5e, 0x0d, 0x5b, 0x2c, 0x4b, 0xc7, 0xe1, 0x96, 0xea, 0x85, 0x7d, 0x35, + 0xd7, 0xf8, 0x63, 0xc7, 0xa0, 0xd6, 0xc8, 0xec, 0x9d, 0xcb, 0x60, 0x36, 0x61, 0xe4, 0xe8, 0x2a, + 0x4c, 0xec, 0x7b, 0x7e, 0x78, 0xea, 0xfd, 0x72, 0x18, 0x4f, 0xfd, 0xa6, 0xe7, 0x37, 0x1f, 0x1d, + 0xad, 0x2e, 0x24, 0x3a, 0xcb, 0x46, 0xac, 0xba, 0x9f, 0x1c, 0x86, 0xdd, 0x98, 0xfe, 0xd1, 0xdf, + 0xaf, 0x9e, 0xfb, 0xfe, 0x4f, 0x2f, 0x9d, 0x73, 0x3f, 0x29, 0xc2, 0x7c, 0x7a, 0x55, 0x47, 0x28, + 0x76, 0xc4, 0x3e, 0x6c, 0xfa, 0x4c, 0x7d, 0x58, 0xe1, 0xec, 0x7c, 0x58, 0xf1, 0x2c, 0x7c, 0xd8, + 0xc4, 0xa9, 0xf9, 0x30, 0xf7, 0xdf, 0x1d, 0x38, 0x1f, 0xed, 0xcc, 0x77, 0xfb, 0xf2, 0x64, 0x8d, + 0x57, 0xdd, 0x39, 0xfd, 0x55, 0xff, 0x10, 0xa6, 0x78, 0xd0, 0x67, 0x0d, 0x15, 0x3e, 0x4a, 0xf4, + 0xd7, 0xf3, 0x39, 0x4d, 0xcd, 0x6b, 0xc5, 0x4c, 0xba, 0x01, 0x87, 0xa8, 0xee, 0xfb, 0xd1, 0x7c, + 0x0c, 0x49, 0x47, 0x14, 0x4c, 0xc6, 0x5b, 0x72, 0x3e, 0xd3, 0x76, 0x44, 0x21, 0x5b, 0xb1, 0xa1, + 0x22, 0x57, 0xb9, 0xf3, 0x30, 0xb0, 0x2d, 0xd7, 0xc0, 0x78, 0x65, 0xb5, 0x07, 0x9a, 0xe2, 0xfe, + 0xbc, 0x18, 0x59, 0x8f, 0x49, 0x5f, 0xbf, 0x07, 0xa0, 0x3d, 0x13, 0x6d, 0x6e, 0xf9, 0xc6, 0xf5, + 0x6e, 0x8c, 0x71, 0x10, 0x18, 0x77, 0x28, 0x51, 0xb4, 0xef, 0x8d, 0x62, 0x80, 0x98, 0x80, 0x2d, + 0x51, 0xe8, 0xf7, 0xa0, 0x12, 0xd6, 0x4b, 0x6e, 0x07, 0xcc, 0xe8, 0xf0, 0xe6, 0x38, 0x92, 0xd7, + 0x63, 0x98, 0x74, 0x59, 0x2d, 0xa6, 0x60, 0x5b, 0xda, 0x32, 0x83, 0xb9, 0xd4, 0x78, 0x33, 0xbc, + 0xef, 0x56, 0xd2, 0xfb, 0xbe, 0x96, 0xe7, 0x58, 0x30, 0x35, 0x26, 0xbb, 0x1e, 0xc7, 0x61, 0x3e, + 0x3d, 0xd2, 0x53, 0x13, 0x9a, 0x28, 0x6c, 0xd9, 0xfe, 0xfe, 0xef, 0x0a, 0x50, 0x8e, 0x2c, 0x3e, + 0x4f, 0x96, 0xaa, 0x4f, 0xea, 0xc2, 0x09, 0x99, 0x44, 0x71, 0x94, 0x4c, 0x62, 0x62, 0x78, 0x26, + 0x11, 0x56, 0xb2, 0x4a, 0x8f, 0xaf, 0x64, 0x59, 0x99, 0xc4, 0xd4, 0xe8, 0x99, 0xc4, 0xf4, 0xc9, + 0x99, 0x84, 0xfb, 0x0f, 0x0e, 0xa0, 0xc1, 0xb4, 0x31, 0xcf, 0x42, 0x91, 0xb4, 0x1f, 0x1e, 0xf1, + 0x94, 0x4e, 0xe7, 0x6e, 0xc3, 0xdd, 0xb1, 0xfb, 0xe9, 0x24, 0xcc, 0xdd, 0xf1, 0xc6, 0x2e, 0x38, + 0x08, 0x78, 0x56, 0x23, 0xd5, 0xa9, 0x89, 0x91, 0xea, 0x82, 0x11, 0x41, 0x5b, 0x87, 0x66, 0x7f, + 0x6f, 0x18, 0xd6, 0x67, 0x37, 0xb2, 0xbb, 0x3d, 0x1a, 0x4e, 0xc2, 0xc3, 0xa0, 0x47, 0x56, 0x92, + 0xb7, 0x60, 0x96, 0x0b, 0xe6, 0x35, 0x84, 0x2e, 0x69, 0xf0, 0xa5, 0x8a, 0xf2, 0x72, 0x17, 0x4d, + 0xf7, 0xd9, 0xba, 0x4d, 0xc4, 0xc9, 0xbe, 0x99, 0x95, 0x92, 0x89, 0xdc, 0x95, 0x92, 0x35, 0x28, + 0x93, 0x4e, 0x27, 0xf8, 0xde, 0x7d, 0xd2, 0xe2, 0x26, 0x55, 0x8d, 0xb4, 0x66, 0x3d, 0x24, 0xe0, + 0xb8, 0x0f, 0xaa, 0x02, 0x78, 0x2d, 0x3f, 0x60, 0x54, 0x71, 0x94, 0x94, 0xbb, 0x55, 0xd5, 0xe0, + 0xad, 0xa8, 0x15, 0x5b, 0x3d, 0x50, 0x1d, 0x2e, 0x7a, 0x3e, 0xa7, 0x8d, 0x3e, 0xa3, 0xf5, 0x7d, + 0xaf, 0x77, 0x7f, 0xbb, 0xae, 0xbc, 0xc4, 0xa1, 0xd2, 0xe6, 0xe9, 0xda, 0x73, 0x46, 0xd8, 0xc5, + 0xad, 0xac, 0x4e, 0x38, 0x9b, 0x17, 0xbd, 0x0e, 0x33, 0x9e, 0xdf, 0xe8, 0xf4, 0x9b, 0x74, 0x87, + 0x88, 0x36, 0x5f, 0x9a, 0x56, 0xc3, 0x98, 0x3f, 0x3e, 0x5a, 0x9d, 0xd9, 0xb2, 0xda, 0x71, 0xa2, + 0x97, 0xe4, 0xa2, 0x1f, 0x59, 0x5c, 0xe5, 0x98, 0xeb, 0xd6, 0x47, 0x36, 0x97, 0xdd, 0x2b, 0xa3, + 0x96, 0x04, 0xb9, 0x6a, 0x49, 0x3f, 0x2e, 0x40, 0x49, 0x97, 0x72, 0xd1, 0xd5, 0x54, 0xbd, 0xf4, + 0xb9, 0x81, 0x7a, 0x69, 0x25, 0xab, 0xec, 0xed, 0x42, 0xc9, 0xe3, 0xbc, 0x9f, 0x3c, 0xdd, 0xb6, + 0x54, 0x0b, 0x36, 0x14, 0x55, 0x76, 0x09, 0xfc, 0x3d, 0xaf, 0x65, 0x92, 0xe3, 0x9b, 0x56, 0x28, + 0x1b, 0x5f, 0xb7, 0x7d, 0x18, 0xdd, 0xc7, 0xc5, 0x51, 0x6d, 0xa2, 0x83, 0x0c, 0x6f, 0xef, 0xd6, + 0xef, 0xbd, 0xa3, 0x65, 0x6c, 0x28, 0x44, 0x6c, 0x90, 0xa5, 0x8c, 0xa0, 0x2f, 0x7a, 0x7d, 0xa1, + 0x14, 0xe5, 0x94, 0x64, 0xdc, 0x53, 0x88, 0xd8, 0x20, 0xbb, 0x9f, 0x38, 0x30, 0xa7, 0xd7, 0x60, + 0xa3, 0x4d, 0x1b, 0xfb, 0x75, 0x41, 0x7b, 0x32, 0xda, 0xec, 0x73, 0xca, 0xd3, 0xd1, 0xe6, 0xbb, + 0x9c, 0x72, 0xac, 0x28, 0xd6, 0xec, 0x0b, 0x67, 0x35, 0x7b, 0xf7, 0x9f, 0x1c, 0x98, 0x54, 0x61, + 0x5d, 0x1e, 0xff, 0x93, 0x2c, 0x75, 0x14, 0x46, 0x2a, 0x75, 0x9c, 0x50, 0x84, 0x8a, 0xab, 0x2c, + 0x13, 0x8f, 0xab, 0xb2, 0xb8, 0x3f, 0x73, 0x60, 0x31, 0xab, 0x72, 0x97, 0x67, 0xf8, 0xaf, 0xc0, + 0x74, 0xaf, 0x43, 0xc4, 0x5e, 0xc0, 0xba, 0xe9, 0x12, 0xfd, 0x8e, 0x69, 0xc7, 0x51, 0x0f, 0xc4, + 0x00, 0x58, 0x98, 0x22, 0x84, 0xe1, 0xf3, 0xcd, 0xbc, 0x27, 0x42, 0xb2, 0xe4, 0x14, 0x2f, 0x56, + 0xd4, 0xc4, 0xb1, 0x25, 0xc5, 0xfd, 0x8b, 0x49, 0x58, 0x50, 0x2c, 0xe3, 0x9e, 0x10, 0xe3, 0xec, + 0x50, 0x0f, 0x9e, 0x51, 0x81, 0xfd, 0xe0, 0xa1, 0xa2, 0x37, 0xed, 0xba, 0xe1, 0x7f, 0x66, 0x2b, + 0xb3, 0xd7, 0xa3, 0xa1, 0x14, 0x3c, 0x04, 0x77, 0xf0, 0xa4, 0x80, 0x5f, 0xbc, 0x93, 0xc2, 0x56, + 0xb6, 0xa9, 0x13, 0x95, 0x6d, 0xe8, 0xb9, 0x32, 0xfd, 0x04, 0xe7, 0xca, 0xa0, 0xaf, 0x2f, 0xe7, + 0xf2, 0xf5, 0x7f, 0x53, 0x80, 0xa9, 0x1d, 0x16, 0xa8, 0x0a, 0xf0, 0xd9, 0x17, 0x13, 0xef, 0xc1, + 0x04, 0xef, 0xd1, 0x86, 0xf1, 0x79, 0x97, 0x47, 0xb3, 0x34, 0x33, 0xbc, 0x7a, 0x8f, 0x36, 0x6a, + 0xd3, 0xd2, 0x8d, 0xca, 0x5f, 0x58, 0x01, 0x59, 0x55, 0xb1, 0x62, 0x9e, 0x08, 0x3c, 0x84, 0x7c, + 0x7c, 0x55, 0xec, 0x53, 0x07, 0x2a, 0xa6, 0xe7, 0x57, 0xb6, 0xfc, 0x62, 0xc6, 0x37, 0xa4, 0xfc, + 0xf2, 0x57, 0xf1, 0x0c, 0xe4, 0xa2, 0xa1, 0x3f, 0x80, 0x85, 0x1e, 0x0b, 0xba, 0x81, 0xb4, 0xd4, + 0x9d, 0xa0, 0xe3, 0x35, 0x3c, 0x1a, 0x56, 0xf0, 0xae, 0x8e, 0x2c, 0xcf, 0x62, 0x3f, 0xac, 0x7d, + 0xcd, 0xc8, 0x5d, 0xd8, 0x49, 0xe3, 0xe2, 0x41, 0x51, 0xee, 0x7f, 0x39, 0x30, 0x9b, 0x58, 0x7b, + 0xd4, 0x00, 0x68, 0x04, 0x7e, 0xd3, 0x13, 0xd1, 0x2d, 0x69, 0xe5, 0xca, 0xda, 0x68, 0xab, 0xba, + 0x11, 0xf2, 0xc5, 0x4a, 0x17, 0x35, 0x71, 0x6c, 0xc1, 0xa2, 0xd7, 0xc2, 0x07, 0x0b, 0xc9, 0x20, + 0x46, 0x3f, 0x58, 0x78, 0x74, 0xb4, 0x3a, 0x63, 0xc6, 0x64, 0x3f, 0x60, 0xc8, 0x73, 0x75, 0xff, + 0x8f, 0x05, 0x28, 0x47, 0xf3, 0x7f, 0x0a, 0x66, 0xf4, 0x6e, 0xc2, 0x8c, 0x5e, 0xcb, 0xb9, 0x73, + 0xca, 0x90, 0xa2, 0x98, 0xc4, 0x32, 0xa6, 0x0f, 0x52, 0xc6, 0x94, 0x57, 0x25, 0x4e, 0x30, 0xa7, + 0x7f, 0xd5, 0x9b, 0xaf, 0xfb, 0x3e, 0x05, 0x83, 0xba, 0x9f, 0x34, 0xa8, 0xb5, 0x9c, 0xb3, 0x19, + 0x62, 0x52, 0x3f, 0x70, 0x60, 0x2e, 0x65, 0x04, 0xe8, 0x1b, 0x30, 0xa9, 0x4a, 0x36, 0x46, 0xbf, + 0x22, 0x46, 0x93, 0xe0, 0x2b, 0x1a, 0xda, 0x81, 0x45, 0xd2, 0x17, 0x41, 0xc4, 0x7b, 0xcb, 0x27, + 0xbb, 0x1d, 0xaa, 0xb3, 0xf6, 0xe9, 0xda, 0x2f, 0x19, 0x9e, 0xc5, 0xf5, 0x8c, 0x3e, 0x38, 0x93, + 0xd3, 0xfd, 0x49, 0x01, 0x50, 0xd4, 0x98, 0xa7, 0xd4, 0xf9, 0x01, 0x4c, 0xed, 0xe9, 0xa2, 0xcc, + 0x93, 0xd5, 0xaa, 0x6b, 0x15, 0xbb, 0x5c, 0x1f, 0x62, 0xa2, 0x6f, 0x9f, 0x8e, 0x1e, 0xc1, 0xa0, + 0x0e, 0xa1, 0x87, 0x00, 0x7b, 0x9e, 0xef, 0xf1, 0xf6, 0x98, 0xb7, 0x6a, 0xea, 0x34, 0xbf, 0x1d, + 0x21, 0x60, 0x0b, 0xcd, 0xfd, 0xb8, 0x60, 0xe9, 0xa7, 0x72, 0x97, 0x23, 0xed, 0xeb, 0x4b, 0xc9, + 0xc5, 0x2c, 0x0f, 0xde, 0x63, 0x58, 0x0b, 0x33, 0x71, 0x40, 0x58, 0x58, 0x52, 0xbd, 0x96, 0x73, + 0x59, 0x1e, 0x10, 0xe6, 0xc9, 0x8d, 0x8f, 0xb7, 0xf4, 0x01, 0x61, 0x1c, 0x2b, 0x48, 0xf4, 0x2d, + 0x39, 0x54, 0xda, 0x0b, 0x5d, 0x68, 0x6e, 0x9f, 0x20, 0x68, 0xcf, 0x9e, 0x1f, 0xed, 0x71, 0xac, + 0x01, 0xdd, 0x8f, 0xa7, 0x2c, 0x85, 0x37, 0x5e, 0xfb, 0x2e, 0xa0, 0x0e, 0xe1, 0xe2, 0x6d, 0xe2, + 0x37, 0xa5, 0x7a, 0xd2, 0x3d, 0x46, 0x79, 0xdb, 0x44, 0x68, 0xcb, 0x06, 0x05, 0x6d, 0x0f, 0xf4, + 0xc0, 0x19, 0x5c, 0xe8, 0x6a, 0xd2, 0x39, 0xaf, 0xa6, 0x9d, 0xf3, 0xf9, 0xd8, 0xda, 0xc6, 0x73, + 0xcf, 0xb6, 0xba, 0x4f, 0x9e, 0x81, 0xba, 0xff, 0x3e, 0x2c, 0xec, 0xa5, 0xef, 0xb5, 0xcc, 0x2d, + 0xf7, 0xb5, 0x31, 0xaf, 0xc5, 0x6a, 0x17, 0x8f, 0xe3, 0xcb, 0x90, 0xb8, 0x19, 0x0f, 0x0a, 0x42, + 0x41, 0xf8, 0x9e, 0x4b, 0x65, 0x9f, 0xba, 0xb0, 0x30, 0xb2, 0xc9, 0xa5, 0xf2, 0xd6, 0xf4, 0x4b, + 0x2e, 0x0d, 0x89, 0x13, 0x02, 0x52, 0x26, 0x58, 0x3a, 0x4d, 0x13, 0x44, 0x57, 0xa1, 0xd2, 0xe8, + 0x33, 0x46, 0x7d, 0x21, 0x87, 0xa3, 0x42, 0xd9, 0x62, 0x5c, 0x20, 0xde, 0x88, 0x49, 0xd8, 0xee, + 0x87, 0x7e, 0xe8, 0xc0, 0x45, 0xa9, 0xac, 0xb7, 0x3e, 0xa2, 0x8d, 0xbe, 0x5c, 0x95, 0xf0, 0x11, + 0xe7, 0x52, 0x45, 0xad, 0xc6, 0x88, 0xaf, 0xdb, 0xea, 0x59, 0x10, 0x71, 0x5c, 0x9e, 0x49, 0xc6, + 0xd9, 0x82, 0xd1, 0x87, 0xca, 0x75, 0x08, 0xaa, 0xd2, 0x9e, 0x27, 0x4f, 0xef, 0xcb, 0xc6, 0xed, + 0x08, 0xed, 0x76, 0x04, 0x75, 0xff, 0x32, 0xe1, 0xad, 0x46, 0x2b, 0x3a, 0x2c, 0x43, 0x81, 0xf0, + 0x74, 0x99, 0x78, 0x9d, 0xe3, 0x02, 0xe1, 0xe8, 0xdb, 0x30, 0xc9, 0xa8, 0x60, 0x87, 0xc6, 0xa9, + 0x5e, 0x1f, 0xc3, 0x81, 0x60, 0xc9, 0xaf, 0x87, 0xaa, 0x7e, 0x62, 0x8d, 0x68, 0xd5, 0x3a, 0x8a, + 0x67, 0x56, 0xeb, 0xf8, 0xb1, 0x63, 0x9d, 0x85, 0xd1, 0x60, 0xd0, 0xbb, 0x30, 0x25, 0xbc, 0x2e, + 0x0d, 0xfa, 0x22, 0x5f, 0x80, 0xb1, 0xd9, 0x67, 0xea, 0xee, 0x57, 0x9b, 0xfc, 0x7d, 0x0d, 0x81, + 0x43, 0x2c, 0x99, 0x75, 0x51, 0xc6, 0x02, 0x76, 0xbf, 0x2d, 0x5d, 0x58, 0xd0, 0xd1, 0xa7, 0xf8, + 0x6c, 0x9c, 0x75, 0xdd, 0x4a, 0x50, 0x71, 0xaa, 0xb7, 0xcb, 0x20, 0x8e, 0x97, 0xc3, 0xd7, 0xb1, + 0xe8, 0x03, 0x13, 0xd5, 0x39, 0x79, 0x5e, 0x64, 0x0e, 0xc0, 0x0c, 0x8b, 0xee, 0xdc, 0x9f, 0x38, + 0x70, 0x31, 0xb3, 0x77, 0x74, 0x2c, 0x15, 0xce, 0xf0, 0x58, 0x72, 0x4e, 0xfb, 0x58, 0x7a, 0x68, + 0x2d, 0x61, 0x38, 0x84, 0xd3, 0x7a, 0xd2, 0xfe, 0xa3, 0x02, 0xcc, 0x63, 0xda, 0x0b, 0x12, 0x15, + 0x9a, 0x9d, 0xf0, 0x8d, 0x5b, 0x8e, 0x90, 0x26, 0x75, 0x0f, 0x50, 0x9b, 0x4a, 0x3c, 0x6e, 0xfb, + 0x16, 0x4c, 0xaa, 0xe2, 0x89, 0x09, 0xc2, 0xae, 0xe5, 0xb8, 0xb7, 0x4d, 0xa0, 0x2a, 0x8b, 0xd3, + 0x55, 0x28, 0x0d, 0x28, 0x91, 0xd5, 0x95, 0xab, 0x31, 0xb8, 0x6b, 0x39, 0x2e, 0x6f, 0x07, 0x91, + 0x55, 0x33, 0xd6, 0x80, 0xee, 0x27, 0x05, 0xd0, 0xe1, 0xcf, 0x53, 0x48, 0x73, 0x7e, 0x2b, 0x91, + 0xe6, 0xac, 0x8d, 0xea, 0xc4, 0xe5, 0xf2, 0x0c, 0x4b, 0x71, 0xd2, 0xa1, 0xe9, 0xe5, 0x3c, 0xa0, + 0x8f, 0x4f, 0x6f, 0xfe, 0xc5, 0x81, 0xb2, 0xea, 0xf7, 0x14, 0x52, 0x9b, 0x9d, 0x64, 0x6a, 0xf3, + 0x72, 0x8e, 0x59, 0x0c, 0x49, 0x6b, 0x3e, 0x2e, 0x9a, 0xd1, 0x47, 0x81, 0x6f, 0x9b, 0xb0, 0xa6, + 0x09, 0xe9, 0x62, 0x0b, 0x94, 0x8d, 0x58, 0xd3, 0xd0, 0xef, 0xc2, 0x3c, 0xd3, 0xb7, 0xf8, 0xb4, + 0x79, 0x3b, 0x8a, 0xaf, 0x8a, 0xb9, 0xaf, 0xd9, 0xcd, 0x53, 0x80, 0xb8, 0xb4, 0x87, 0x53, 0xa8, + 0x78, 0x40, 0x8e, 0x8c, 0xb9, 0x7a, 0x69, 0x5f, 0x66, 0x62, 0x91, 0x6b, 0x63, 0x3a, 0x4e, 0x1d, + 0x73, 0x0d, 0x34, 0xe3, 0x41, 0x41, 0xa8, 0x0d, 0x33, 0xf6, 0x03, 0x21, 0xa3, 0x4b, 0x57, 0xf2, + 0xbf, 0x44, 0xd2, 0x57, 0x39, 0x76, 0x0b, 0x4e, 0x20, 0xbb, 0x47, 0x25, 0xa8, 0x58, 0xca, 0x97, + 0x2a, 0x97, 0xcc, 0x9e, 0x4d, 0xb9, 0x24, 0x3b, 0xba, 0xaf, 0x8c, 0x15, 0xdd, 0x5f, 0x4e, 0x46, + 0xf7, 0x5f, 0x4f, 0x47, 0xf7, 0xa0, 0x66, 0x97, 0x88, 0xec, 0x39, 0x9c, 0x37, 0x61, 0x6e, 0xf8, + 0xd2, 0x2b, 0x57, 0xbe, 0x34, 0x18, 0x4c, 0x23, 0x79, 0x22, 0xdf, 0x4e, 0x40, 0xe2, 0x94, 0x08, + 0x79, 0xa2, 0x9b, 0x96, 0x7a, 0xbf, 0xdb, 0x25, 0xec, 0x70, 0x69, 0x46, 0x0d, 0x38, 0x3a, 0xd1, + 0x6f, 0x27, 0xa8, 0x38, 0xd5, 0x1b, 0xed, 0x40, 0x49, 0x47, 0xc9, 0xe6, 0xf5, 0xd0, 0x2b, 0x79, + 0x02, 0x70, 0x1d, 0xd1, 0xe8, 0xdf, 0xd8, 0xe0, 0xd8, 0x09, 0x4e, 0xf9, 0x84, 0x04, 0xe7, 0x2e, + 0xa0, 0x60, 0x57, 0xc5, 0x4e, 0xcd, 0x3b, 0xfa, 0xdb, 0x2f, 0xa9, 0x95, 0x25, 0x15, 0x3d, 0x47, + 0x1b, 0x76, 0x6f, 0xa0, 0x07, 0xce, 0xe0, 0x92, 0x56, 0x6d, 0x42, 0xeb, 0xc8, 0x14, 0x4c, 0x32, + 0x93, 0x37, 0x24, 0x8c, 0xf3, 0xa6, 0x45, 0x69, 0xd5, 0x1b, 0x29, 0x54, 0x3c, 0x20, 0x07, 0x7d, + 0x17, 0x66, 0xa5, 0x0a, 0xc5, 0x82, 0xe1, 0x09, 0x05, 0x2f, 0x1c, 0x1f, 0xad, 0xce, 0x6e, 0xdb, + 0x90, 0x38, 0x29, 0xc1, 0xfd, 0xf3, 0x22, 0x64, 0x07, 0xf6, 0xf1, 0xc3, 0x57, 0xe7, 0x31, 0x0f, + 0x5f, 0xdf, 0x83, 0x32, 0x17, 0x84, 0xe9, 0x47, 0xbe, 0x85, 0xf1, 0x1e, 0xf9, 0xd6, 0x43, 0x00, + 0x1c, 0x63, 0xa5, 0xb2, 0xac, 0xe2, 0xa9, 0x66, 0x59, 0x57, 0x00, 0x54, 0x3c, 0xba, 0x11, 0xf4, + 0xcd, 0xbd, 0xca, 0x6c, 0xec, 0x13, 0x6e, 0x45, 0x14, 0x6c, 0xf5, 0x42, 0xd7, 0xa3, 0x83, 0x53, + 0x5f, 0xa4, 0x5c, 0x1a, 0xb8, 0x08, 0x4e, 0xe7, 0xe9, 0x19, 0x9f, 0x40, 0x9d, 0xf0, 0x70, 0xc4, + 0xfd, 0xbf, 0x02, 0x24, 0x9c, 0x21, 0xfa, 0x81, 0x03, 0x0b, 0x24, 0xf5, 0x15, 0x59, 0x18, 0x4b, + 0xfe, 0x46, 0xbe, 0x4f, 0xfb, 0x06, 0x3e, 0x42, 0x8b, 0x4b, 0xd7, 0xe9, 0x2e, 0x1c, 0x0f, 0x0a, + 0x45, 0x7f, 0xea, 0xc0, 0x05, 0x32, 0xf8, 0x99, 0xa0, 0xd9, 0xf4, 0x37, 0xc7, 0xfe, 0xce, 0xb0, + 0xf6, 0xec, 0xf1, 0xd1, 0x6a, 0xd6, 0x07, 0x94, 0x38, 0x4b, 0x1c, 0xfa, 0x0e, 0x4c, 0x10, 0xd6, + 0x0a, 0xcb, 0x3c, 0xf9, 0xc5, 0x86, 0x5f, 0x7f, 0xc6, 0xd1, 0xd1, 0x3a, 0x6b, 0x71, 0xac, 0x40, + 0xdd, 0x9f, 0x16, 0x61, 0x3e, 0xfd, 0x50, 0xd6, 0xbc, 0x2d, 0x9a, 0xc8, 0x7c, 0x5b, 0x24, 0x6d, + 0xa4, 0x21, 0xa2, 0x87, 0x3e, 0xb1, 0x8d, 0xc8, 0x46, 0xac, 0x69, 0x91, 0x8d, 0x48, 0xbd, 0x34, + 0x05, 0x98, 0xf1, 0x6c, 0x44, 0xfe, 0x89, 0x63, 0x2c, 0x74, 0x3d, 0x79, 0xb6, 0xb8, 0xe9, 0xb3, + 0x65, 0xc1, 0x9e, 0xcb, 0xb8, 0xc5, 0xa3, 0x2e, 0x54, 0xac, 0x7d, 0x30, 0x96, 0x78, 0x23, 0xf7, + 0xba, 0xc7, 0x6a, 0x37, 0xa7, 0x3f, 0x21, 0x8d, 0x29, 0x36, 0x7e, 0x6c, 0xf7, 0x6a, 0xb5, 0x9e, + 0xa8, 0xba, 0xa2, 0x96, 0xcb, 0x42, 0x73, 0xf7, 0x61, 0x36, 0xf1, 0xde, 0x4d, 0x0a, 0x0b, 0xdf, + 0xf8, 0x8d, 0xff, 0x4d, 0xe5, 0x83, 0x08, 0x01, 0x5b, 0x68, 0xea, 0x4e, 0xe4, 0x3d, 0xc2, 0x68, + 0x3b, 0xe8, 0x73, 0xfa, 0x55, 0xbd, 0x13, 0x89, 0x06, 0x78, 0xda, 0x77, 0x22, 0x31, 0xf0, 0xc9, + 0x77, 0x22, 0x51, 0xdf, 0xaf, 0xec, 0x9d, 0x48, 0x34, 0xc2, 0x21, 0xc9, 0xc3, 0xff, 0x16, 0xac, + 0x59, 0x24, 0x13, 0x88, 0xc2, 0x63, 0x12, 0x88, 0xf7, 0x61, 0xda, 0xf3, 0x05, 0x65, 0x07, 0xa4, + 0x63, 0xaa, 0x4e, 0x79, 0xab, 0x33, 0xd1, 0x54, 0xb7, 0x0c, 0x0e, 0x8e, 0x10, 0x51, 0x07, 0x2e, + 0x86, 0xd5, 0x52, 0x46, 0x49, 0x7c, 0x5b, 0x63, 0x5e, 0x3c, 0xbc, 0x11, 0x96, 0xf5, 0x6e, 0x67, + 0x75, 0x7a, 0x34, 0x8c, 0x80, 0xb3, 0x41, 0x11, 0x87, 0x59, 0x6e, 0x65, 0xce, 0xe1, 0x21, 0x35, + 0x62, 0xa5, 0x39, 0x5d, 0x6c, 0xb0, 0x9e, 0x49, 0xd8, 0xa0, 0x38, 0x29, 0xc3, 0xfd, 0x8f, 0x22, + 0xcc, 0xa5, 0x34, 0x2d, 0x95, 0x21, 0x94, 0x9f, 0x66, 0x86, 0x50, 0x1a, 0x2b, 0x43, 0xc8, 0x0e, + 0x5e, 0x27, 0xc6, 0x0a, 0x5e, 0xdf, 0xd2, 0x01, 0xa4, 0xd9, 0xb9, 0xad, 0x4d, 0xf3, 0xaa, 0x34, + 0x5a, 0xcd, 0x6d, 0x9b, 0x88, 0x93, 0x7d, 0xd5, 0x09, 0xdf, 0x1c, 0xfc, 0x84, 0xd2, 0x44, 0xbf, + 0x6f, 0xe6, 0x7d, 0x16, 0x14, 0x01, 0xe8, 0x13, 0x3e, 0x83, 0x80, 0xb3, 0xc4, 0xd5, 0xee, 0x7e, + 0xf6, 0xe5, 0xca, 0xb9, 0xcf, 0xbf, 0x5c, 0x39, 0xf7, 0xc5, 0x97, 0x2b, 0xe7, 0xbe, 0x7f, 0xbc, + 0xe2, 0x7c, 0x76, 0xbc, 0xe2, 0x7c, 0x7e, 0xbc, 0xe2, 0x7c, 0x71, 0xbc, 0xe2, 0xfc, 0xf7, 0xf1, + 0x8a, 0xf3, 0xc3, 0x9f, 0xad, 0x9c, 0x7b, 0xf8, 0xfc, 0x28, 0xff, 0x47, 0xe3, 0xff, 0x03, 0x00, + 0x00, 0xff, 0xff, 0x6a, 0xff, 0x8d, 0xf8, 0x6e, 0x43, 0x00, 0x00, } func (m *AnalysisRunArgument) Marshal() (dAtA []byte, err error) { @@ -4007,9 +4043,20 @@ func (m *PromotionStatus) MarshalToSizedBuffer(dAtA []byte) (int, error) { _ = i var l int _ = l - i = encodeVarintGenerated(dAtA, i, uint64(m.CurrentStepAttempt)) - i-- - dAtA[i] = 0x58 + if len(m.StepExecutionMetadata) > 0 { + for iNdEx := len(m.StepExecutionMetadata) - 1; iNdEx >= 0; iNdEx-- { + { + size, err := m.StepExecutionMetadata[iNdEx].MarshalToSizedBuffer(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = encodeVarintGenerated(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0x5a + } + } if m.State != nil { { size, err := m.State.MarshalToSizedBuffer(dAtA[:i]) @@ -4170,9 +4217,21 @@ func (m *PromotionStepRetry) MarshalToSizedBuffer(dAtA []byte) (int, error) { _ = i var l int _ = l - i = encodeVarintGenerated(dAtA, i, uint64(m.Attempts)) + i = encodeVarintGenerated(dAtA, i, uint64(m.ErrorThreshold)) i-- - dAtA[i] = 0x8 + dAtA[i] = 0x10 + if m.Timeout != nil { + { + size, err := m.Timeout.MarshalToSizedBuffer(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = encodeVarintGenerated(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0xa + } return len(dAtA) - i, nil } @@ -4628,6 +4687,71 @@ func (m *StageStatus) MarshalToSizedBuffer(dAtA []byte) (int, error) { return len(dAtA) - i, nil } +func (m *StepExecutionMetadata) Marshal() (dAtA []byte, err error) { + size := m.Size() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBuffer(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *StepExecutionMetadata) MarshalTo(dAtA []byte) (int, error) { + size := m.Size() + return m.MarshalToSizedBuffer(dAtA[:size]) +} + +func (m *StepExecutionMetadata) MarshalToSizedBuffer(dAtA []byte) (int, error) { + i := len(dAtA) + _ = i + var l int + _ = l + i -= len(m.Message) + copy(dAtA[i:], m.Message) + i = encodeVarintGenerated(dAtA, i, uint64(len(m.Message))) + i-- + dAtA[i] = 0x32 + i -= len(m.Status) + copy(dAtA[i:], m.Status) + i = encodeVarintGenerated(dAtA, i, uint64(len(m.Status))) + i-- + dAtA[i] = 0x2a + i = encodeVarintGenerated(dAtA, i, uint64(m.ErrorCount)) + i-- + dAtA[i] = 0x20 + if m.FinishedAt != nil { + { + size, err := m.FinishedAt.MarshalToSizedBuffer(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = encodeVarintGenerated(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0x1a + } + if m.StartedAt != nil { + { + size, err := m.StartedAt.MarshalToSizedBuffer(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = encodeVarintGenerated(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0x12 + } + i -= len(m.Alias) + copy(dAtA[i:], m.Alias) + i = encodeVarintGenerated(dAtA, i, uint64(len(m.Alias))) + i-- + dAtA[i] = 0xa + return len(dAtA) - i, nil +} + func (m *Verification) Marshal() (dAtA []byte, err error) { size := m.Size() dAtA = make([]byte, size) @@ -5845,7 +5969,12 @@ func (m *PromotionStatus) Size() (n int) { l = m.State.Size() n += 1 + l + sovGenerated(uint64(l)) } - n += 1 + sovGenerated(uint64(m.CurrentStepAttempt)) + if len(m.StepExecutionMetadata) > 0 { + for _, e := range m.StepExecutionMetadata { + l = e.Size() + n += 1 + l + sovGenerated(uint64(l)) + } + } return n } @@ -5876,7 +6005,11 @@ func (m *PromotionStepRetry) Size() (n int) { } var l int _ = l - n += 1 + sovGenerated(uint64(m.Attempts)) + if m.Timeout != nil { + l = m.Timeout.Size() + n += 1 + l + sovGenerated(uint64(l)) + } + n += 1 + sovGenerated(uint64(m.ErrorThreshold)) return n } @@ -6045,6 +6178,30 @@ func (m *StageStatus) Size() (n int) { return n } +func (m *StepExecutionMetadata) Size() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.Alias) + n += 1 + l + sovGenerated(uint64(l)) + if m.StartedAt != nil { + l = m.StartedAt.Size() + n += 1 + l + sovGenerated(uint64(l)) + } + if m.FinishedAt != nil { + l = m.FinishedAt.Size() + n += 1 + l + sovGenerated(uint64(l)) + } + n += 1 + sovGenerated(uint64(m.ErrorCount)) + l = len(m.Status) + n += 1 + l + sovGenerated(uint64(l)) + l = len(m.Message) + n += 1 + l + sovGenerated(uint64(l)) + return n +} + func (m *Verification) Size() (n int) { if m == nil { return 0 @@ -6825,6 +6982,11 @@ func (this *PromotionStatus) String() string { repeatedStringForHealthChecks += strings.Replace(strings.Replace(f.String(), "HealthCheckStep", "HealthCheckStep", 1), `&`, ``, 1) + "," } repeatedStringForHealthChecks += "}" + repeatedStringForStepExecutionMetadata := "[]StepExecutionMetadata{" + for _, f := range this.StepExecutionMetadata { + repeatedStringForStepExecutionMetadata += strings.Replace(strings.Replace(f.String(), "StepExecutionMetadata", "StepExecutionMetadata", 1), `&`, ``, 1) + "," + } + repeatedStringForStepExecutionMetadata += "}" s := strings.Join([]string{`&PromotionStatus{`, `Phase:` + fmt.Sprintf("%v", this.Phase) + `,`, `Message:` + fmt.Sprintf("%v", this.Message) + `,`, @@ -6835,7 +6997,7 @@ func (this *PromotionStatus) String() string { `HealthChecks:` + repeatedStringForHealthChecks + `,`, `CurrentStep:` + fmt.Sprintf("%v", this.CurrentStep) + `,`, `State:` + strings.Replace(fmt.Sprintf("%v", this.State), "JSON", "v11.JSON", 1) + `,`, - `CurrentStepAttempt:` + fmt.Sprintf("%v", this.CurrentStepAttempt) + `,`, + `StepExecutionMetadata:` + repeatedStringForStepExecutionMetadata + `,`, `}`, }, "") return s @@ -6858,7 +7020,8 @@ func (this *PromotionStepRetry) String() string { return "nil" } s := strings.Join([]string{`&PromotionStepRetry{`, - `Attempts:` + fmt.Sprintf("%v", this.Attempts) + `,`, + `Timeout:` + strings.Replace(fmt.Sprintf("%v", this.Timeout), "Duration", "v1.Duration", 1) + `,`, + `ErrorThreshold:` + fmt.Sprintf("%v", this.ErrorThreshold) + `,`, `}`, }, "") return s @@ -6992,6 +7155,21 @@ func (this *StageStatus) String() string { }, "") return s } +func (this *StepExecutionMetadata) String() string { + if this == nil { + return "nil" + } + s := strings.Join([]string{`&StepExecutionMetadata{`, + `Alias:` + fmt.Sprintf("%v", this.Alias) + `,`, + `StartedAt:` + strings.Replace(fmt.Sprintf("%v", this.StartedAt), "Time", "v1.Time", 1) + `,`, + `FinishedAt:` + strings.Replace(fmt.Sprintf("%v", this.FinishedAt), "Time", "v1.Time", 1) + `,`, + `ErrorCount:` + fmt.Sprintf("%v", this.ErrorCount) + `,`, + `Status:` + fmt.Sprintf("%v", this.Status) + `,`, + `Message:` + fmt.Sprintf("%v", this.Message) + `,`, + `}`, + }, "") + return s +} func (this *Verification) String() string { if this == nil { return "nil" @@ -14203,10 +14381,10 @@ func (m *PromotionStatus) Unmarshal(dAtA []byte) error { } iNdEx = postIndex case 11: - if wireType != 0 { - return fmt.Errorf("proto: wrong wireType = %d for field CurrentStepAttempt", wireType) + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field StepExecutionMetadata", wireType) } - m.CurrentStepAttempt = 0 + var msglen int for shift := uint(0); ; shift += 7 { if shift >= 64 { return ErrIntOverflowGenerated @@ -14216,11 +14394,26 @@ func (m *PromotionStatus) Unmarshal(dAtA []byte) error { } b := dAtA[iNdEx] iNdEx++ - m.CurrentStepAttempt |= int64(b&0x7F) << shift + msglen |= int(b&0x7F) << shift if b < 0x80 { break } } + if msglen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.StepExecutionMetadata = append(m.StepExecutionMetadata, StepExecutionMetadata{}) + if err := m.StepExecutionMetadata[len(m.StepExecutionMetadata)-1].Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex default: iNdEx = preIndex skippy, err := skipGenerated(dAtA[iNdEx:]) @@ -14458,10 +14651,46 @@ func (m *PromotionStepRetry) Unmarshal(dAtA []byte) error { } switch fieldNum { case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Timeout", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Timeout == nil { + m.Timeout = &v1.Duration{} + } + if err := m.Timeout.Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: if wireType != 0 { - return fmt.Errorf("proto: wrong wireType = %d for field Attempts", wireType) + return fmt.Errorf("proto: wrong wireType = %d for field ErrorThreshold", wireType) } - m.Attempts = 0 + m.ErrorThreshold = 0 for shift := uint(0); ; shift += 7 { if shift >= 64 { return ErrIntOverflowGenerated @@ -14471,7 +14700,7 @@ func (m *PromotionStepRetry) Unmarshal(dAtA []byte) error { } b := dAtA[iNdEx] iNdEx++ - m.Attempts |= int64(b&0x7F) << shift + m.ErrorThreshold |= uint32(b&0x7F) << shift if b < 0x80 { break } @@ -15797,6 +16026,243 @@ func (m *StageStatus) Unmarshal(dAtA []byte) error { } return nil } +func (m *StepExecutionMetadata) Unmarshal(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: StepExecutionMetadata: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: StepExecutionMetadata: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Alias", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Alias = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field StartedAt", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.StartedAt == nil { + m.StartedAt = &v1.Time{} + } + if err := m.StartedAt.Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field FinishedAt", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.FinishedAt == nil { + m.FinishedAt = &v1.Time{} + } + if err := m.FinishedAt.Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field ErrorCount", wireType) + } + m.ErrorCount = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.ErrorCount |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 5: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Status", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Status = PromotionPhase(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Message", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowGenerated + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return ErrInvalidLengthGenerated + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return ErrInvalidLengthGenerated + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Message = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := skipGenerated(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return ErrInvalidLengthGenerated + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} func (m *Verification) Unmarshal(dAtA []byte) error { l := len(dAtA) iNdEx := 0 diff --git a/api/v1alpha1/generated.proto b/api/v1alpha1/generated.proto index d0a97d9ff..7e79145a6 100644 --- a/api/v1alpha1/generated.proto +++ b/api/v1alpha1/generated.proto @@ -899,9 +899,9 @@ message PromotionStatus { // subsequent reconciliations attempts. optional int64 currentStep = 9; - // CurrentStepAttempt is the number of times the current step has been - // attempted. - optional int64 currentStepAttempt = 11; + // StepExecutionMetadata tracks metadata pertaining to the execution + // of individual promotion steps. + repeated StepExecutionMetadata stepExecutionMetadata = 11; // State stores the state of the promotion process between reconciliation // attempts. @@ -930,15 +930,42 @@ message PromotionStep { // PromotionStepRetry describes the retry policy for a PromotionStep. message PromotionStepRetry { - // Attempts is the number of times the step can be attempted before the - // PromotionStep is marked as failed. + // Timeout is the soft maximum interval in which a step that returns a Running + // status (which typically indicates it's waiting for something to happen) + // may be retried. // - // If this field is set to 1, the step will not be retried. If this - // field is set to -1, the step will be retried indefinitely. + // The maximum is a soft one because the check for whether the interval has + // elapsed occurs AFTER the step has run. This effectively means a step may + // run ONCE beyond the close of the interval. // - // The default of this field depends on the step being executed. Refer to - // the documentation for the specific step for more information. - optional int64 attempts = 1; + // If this field is set to nil, the effective default will be a step-specific + // one. If no step-specific default exists (i.e. is also nil), the effective + // default will be the system-wide default of 0. + // + // A value of 0 will cause the step to be retried indefinitely unless the + // ErrorThreshold is reached. + optional .k8s.io.apimachinery.pkg.apis.meta.v1.Duration timeout = 1; + + // ErrorThreshold is the number of consecutive times the step must fail (for + // any reason) before retries are abandoned and the entire Promotion is marked + // as failed. + // + // If this field is set to 0, the effective default will be a step-specific + // one. If no step-specific default exists (i.e. is also 0), the effective + // default will be the system-wide default of 1. + // + // A value of 1 will cause the Promotion to be marked as failed after just + // a single failure; i.e. no retries will be attempted. + // + // There is no option to specify an infinite number of retries using a value + // such as -1. + // + // In a future release, Kargo is likely to become capable of distinguishing + // between recoverable and non-recoverable step failures. At that time, it is + // planned that unrecoverable failures will not be subject to this threshold + // and will immediately cause the Promotion to be marked as failed without + // further condition. + optional uint32 errorThreshold = 2; } // PromotionTemplate defines a template for a Promotion that can be used to @@ -1099,6 +1126,30 @@ message StageStatus { optional PromotionReference lastPromotion = 10; } +// StepExecutionMetadata tracks metadata pertaining to the execution of +// a promotion step. +message StepExecutionMetadata { + // Alias is the alias of the step. + optional string alias = 1; + + // StartedAt is the time at which the first attempt to execute the step + // began. + optional .k8s.io.apimachinery.pkg.apis.meta.v1.Time startedAt = 2; + + // FinishedAt is the time at which the final attempt to execute the step + // completed. + optional .k8s.io.apimachinery.pkg.apis.meta.v1.Time finishedAt = 3; + + // ErrorCount tracks consecutive failed attempts to execute the step. + optional uint32 errorCount = 4; + + // Status is the high-level outcome of the step. + optional string status = 5; + + // Message is a display message about the step, including any errors. + optional string message = 6; +} + // Verification describes how to verify that a Promotion has been successful // using Argo Rollouts AnalysisTemplates. message Verification { diff --git a/api/v1alpha1/promotion_types.go b/api/v1alpha1/promotion_types.go index 7ae3a3651..da18f2dbe 100644 --- a/api/v1alpha1/promotion_types.go +++ b/api/v1alpha1/promotion_types.go @@ -1,6 +1,8 @@ package v1alpha1 import ( + "time" + apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "sigs.k8s.io/yaml" @@ -111,23 +113,58 @@ type PromotionVariable struct { // PromotionStepRetry describes the retry policy for a PromotionStep. type PromotionStepRetry struct { - // Attempts is the number of times the step can be attempted before the - // PromotionStep is marked as failed. + // Timeout is the soft maximum interval in which a step that returns a Running + // status (which typically indicates it's waiting for something to happen) + // may be retried. + // + // The maximum is a soft one because the check for whether the interval has + // elapsed occurs AFTER the step has run. This effectively means a step may + // run ONCE beyond the close of the interval. + // + // If this field is set to nil, the effective default will be a step-specific + // one. If no step-specific default exists (i.e. is also nil), the effective + // default will be the system-wide default of 0. + // + // A value of 0 will cause the step to be retried indefinitely unless the + // ErrorThreshold is reached. + Timeout *metav1.Duration `json:"timeout,omitempty" protobuf:"bytes,1,opt,name=timeout"` + // ErrorThreshold is the number of consecutive times the step must fail (for + // any reason) before retries are abandoned and the entire Promotion is marked + // as failed. // - // If this field is set to 1, the step will not be retried. If this - // field is set to -1, the step will be retried indefinitely. + // If this field is set to 0, the effective default will be a step-specific + // one. If no step-specific default exists (i.e. is also 0), the effective + // default will be the system-wide default of 1. // - // The default of this field depends on the step being executed. Refer to - // the documentation for the specific step for more information. - Attempts int64 `json:"attempts,omitempty" protobuf:"varint,1,opt,name=attempts"` + // A value of 1 will cause the Promotion to be marked as failed after just + // a single failure; i.e. no retries will be attempted. + // + // There is no option to specify an infinite number of retries using a value + // such as -1. + // + // In a future release, Kargo is likely to become capable of distinguishing + // between recoverable and non-recoverable step failures. At that time, it is + // planned that unrecoverable failures will not be subject to this threshold + // and will immediately cause the Promotion to be marked as failed without + // further condition. + ErrorThreshold uint32 `json:"errorThreshold,omitempty" protobuf:"varint,2,opt,name=errorThreshold"` } -// GetAttempts returns the Attempts field with the given fallback value. -func (r *PromotionStepRetry) GetAttempts(fallback int64) int64 { - if r == nil || r.Attempts == 0 { +// GetTimeout returns the Timeout field with the given fallback value. +func (r *PromotionStepRetry) GetTimeout(fallback *time.Duration) *time.Duration { + if r == nil || r.Timeout == nil { return fallback } - return r.Attempts + return &r.Timeout.Duration +} + +// GetErrorThreshold returns the ErrorThreshold field with the given fallback +// value. +func (r *PromotionStepRetry) GetErrorThreshold(fallback uint32) uint32 { + if r == nil || r.ErrorThreshold == 0 { + return fallback + } + return r.ErrorThreshold } // PromotionStep describes a directive to be executed as part of a Promotion. @@ -177,9 +214,9 @@ type PromotionStatus struct { // permits steps that have already run successfully to be skipped on // subsequent reconciliations attempts. CurrentStep int64 `json:"currentStep,omitempty" protobuf:"varint,9,opt,name=currentStep"` - // CurrentStepAttempt is the number of times the current step has been - // attempted. - CurrentStepAttempt int64 `json:"currentStepAttempt,omitempty" protobuf:"varint,11,opt,name=currentStepAttempt"` + // StepExecutionMetadata tracks metadata pertaining to the execution + // of individual promotion steps. + StepExecutionMetadata StepExecutionMetadataList `json:"stepExecutionMetadata,omitempty" protobuf:"bytes,11,rep,name=stepExecutionMetadata"` // State stores the state of the promotion process between reconciliation // attempts. State *apiextensionsv1.JSON `json:"state,omitempty" protobuf:"bytes,10,opt,name=state"` @@ -224,8 +261,8 @@ func (s *HealthCheckStep) GetConfig() map[string]any { } // WithPhase returns a copy of PromotionStatus with the given phase -func (p *PromotionStatus) WithPhase(phase PromotionPhase) *PromotionStatus { - status := p.DeepCopy() +func (s *PromotionStatus) WithPhase(phase PromotionPhase) *PromotionStatus { + status := s.DeepCopy() status.Phase = phase return status } @@ -238,3 +275,25 @@ type PromotionList struct { metav1.ListMeta `json:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"` Items []Promotion `json:"items" protobuf:"bytes,2,rep,name=items"` } + +// StepExecutionMetadataList is a list of StepExecutionMetadata. +type StepExecutionMetadataList []StepExecutionMetadata + +// StepExecutionMetadata tracks metadata pertaining to the execution of +// a promotion step. +type StepExecutionMetadata struct { + // Alias is the alias of the step. + Alias string `json:"alias,omitempty" protobuf:"bytes,1,opt,name=alias"` + // StartedAt is the time at which the first attempt to execute the step + // began. + StartedAt *metav1.Time `json:"startedAt,omitempty" protobuf:"bytes,2,opt,name=startedAt"` + // FinishedAt is the time at which the final attempt to execute the step + // completed. + FinishedAt *metav1.Time `json:"finishedAt,omitempty" protobuf:"bytes,3,opt,name=finishedAt"` + // ErrorCount tracks consecutive failed attempts to execute the step. + ErrorCount uint32 `json:"errorCount,omitempty" protobuf:"varint,4,opt,name=errorCount"` + // Status is the high-level outcome of the step. + Status PromotionPhase `json:"status,omitempty" protobuf:"bytes,5,opt,name=status"` + // Message is a display message about the step, including any errors. + Message string `json:"message,omitempty" protobuf:"bytes,6,opt,name=message"` +} diff --git a/api/v1alpha1/promotion_types_test.go b/api/v1alpha1/promotion_types_test.go index 980662786..c18759f29 100644 --- a/api/v1alpha1/promotion_types_test.go +++ b/api/v1alpha1/promotion_types_test.go @@ -2,16 +2,53 @@ package v1alpha1 import ( "testing" + "time" "github.com/stretchr/testify/require" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/utils/ptr" ) -func TestPromotionRetry_GetAttempts(t *testing.T) { +func TestPromotionRetry_GetTimeout(t *testing.T) { tests := []struct { name string retry *PromotionStepRetry - fallback int64 - want int64 + fallback *time.Duration + want *time.Duration + }{ + { + name: "retry is nil", + retry: nil, + fallback: ptr.To(time.Hour), + want: ptr.To(time.Hour), + }, + { + name: "timeout is not set", + retry: &PromotionStepRetry{}, + fallback: ptr.To(time.Hour), + want: ptr.To(time.Hour), + }, + { + name: "timeout is set", + retry: &PromotionStepRetry{ + Timeout: &metav1.Duration{ + Duration: 3 * time.Hour, + }, + }, + want: ptr.To(3 * time.Hour), + }, + } + for _, tt := range tests { + require.Equal(t, tt.want, tt.retry.GetTimeout(tt.fallback)) + } +} + +func TestPromotionRetry_GetErrorThreshold(t *testing.T) { + tests := []struct { + name string + retry *PromotionStepRetry + fallback uint32 + want uint32 }{ { name: "retry is nil", @@ -20,20 +57,20 @@ func TestPromotionRetry_GetAttempts(t *testing.T) { want: 1, }, { - name: "attempts is not set", + name: "threshold is not set", retry: &PromotionStepRetry{}, - fallback: -1, - want: -1, + fallback: 1, + want: 1, }, { - name: "attempts is set", + name: "threshold is set", retry: &PromotionStepRetry{ - Attempts: 3, + ErrorThreshold: 3, }, want: 3, }, } for _, tt := range tests { - require.Equal(t, tt.want, tt.retry.GetAttempts(tt.fallback)) + require.Equal(t, tt.want, tt.retry.GetErrorThreshold(tt.fallback)) } } diff --git a/api/v1alpha1/zz_generated.deepcopy.go b/api/v1alpha1/zz_generated.deepcopy.go index 2c686e0dc..517d1627b 100644 --- a/api/v1alpha1/zz_generated.deepcopy.go +++ b/api/v1alpha1/zz_generated.deepcopy.go @@ -953,6 +953,13 @@ func (in *PromotionStatus) DeepCopyInto(out *PromotionStatus) { in, out := &in.FinishedAt, &out.FinishedAt *out = (*in).DeepCopy() } + if in.StepExecutionMetadata != nil { + in, out := &in.StepExecutionMetadata, &out.StepExecutionMetadata + *out = make(StepExecutionMetadataList, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } if in.State != nil { in, out := &in.State, &out.State *out = new(apiextensionsv1.JSON) @@ -976,7 +983,7 @@ func (in *PromotionStep) DeepCopyInto(out *PromotionStep) { if in.Retry != nil { in, out := &in.Retry, &out.Retry *out = new(PromotionStepRetry) - **out = **in + (*in).DeepCopyInto(*out) } if in.Config != nil { in, out := &in.Config, &out.Config @@ -998,6 +1005,11 @@ func (in *PromotionStep) DeepCopy() *PromotionStep { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *PromotionStepRetry) DeepCopyInto(out *PromotionStepRetry) { *out = *in + if in.Timeout != nil { + in, out := &in.Timeout, &out.Timeout + *out = new(v1.Duration) + **out = **in + } } // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PromotionStepRetry. @@ -1237,6 +1249,50 @@ func (in *StageStatus) DeepCopy() *StageStatus { return out } +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *StepExecutionMetadata) DeepCopyInto(out *StepExecutionMetadata) { + *out = *in + if in.StartedAt != nil { + in, out := &in.StartedAt, &out.StartedAt + *out = (*in).DeepCopy() + } + if in.FinishedAt != nil { + in, out := &in.FinishedAt, &out.FinishedAt + *out = (*in).DeepCopy() + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new StepExecutionMetadata. +func (in *StepExecutionMetadata) DeepCopy() *StepExecutionMetadata { + if in == nil { + return nil + } + out := new(StepExecutionMetadata) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in StepExecutionMetadataList) DeepCopyInto(out *StepExecutionMetadataList) { + { + in := &in + *out = make(StepExecutionMetadataList, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new StepExecutionMetadataList. +func (in StepExecutionMetadataList) DeepCopy() StepExecutionMetadataList { + if in == nil { + return nil + } + out := new(StepExecutionMetadataList) + in.DeepCopyInto(out) + return *out +} + // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *Verification) DeepCopyInto(out *Verification) { *out = *in diff --git a/charts/kargo/resources/crds/kargo.akuity.io_promotions.yaml b/charts/kargo/resources/crds/kargo.akuity.io_promotions.yaml index 124e00677..8c8c53216 100644 --- a/charts/kargo/resources/crds/kargo.akuity.io_promotions.yaml +++ b/charts/kargo/resources/crds/kargo.akuity.io_promotions.yaml @@ -98,18 +98,46 @@ spec: retry: description: Retry is the retry policy for this step. properties: - attempts: + errorThreshold: description: |- - Attempts is the number of times the step can be attempted before the - PromotionStep is marked as failed. + ErrorThreshold is the number of consecutive times the step must fail (for + any reason) before retries are abandoned and the entire Promotion is marked + as failed. - If this field is set to 1, the step will not be retried. If this - field is set to -1, the step will be retried indefinitely. + If this field is set to 0, the effective default will be a step-specific + one. If no step-specific default exists (i.e. is also 0), the effective + default will be the system-wide default of 1. - The default of this field depends on the step being executed. Refer to - the documentation for the specific step for more information. - format: int64 + A value of 1 will cause the Promotion to be marked as failed after just + a single failure; i.e. no retries will be attempted. + + There is no option to specify an infinite number of retries using a value + such as -1. + + In a future release, Kargo is likely to become capable of distinguishing + between recoverable and non-recoverable step failures. At that time, it is + planned that unrecoverable failures will not be subject to this threshold + and will immediately cause the Promotion to be marked as failed without + further condition. + format: int32 type: integer + timeout: + description: |- + Timeout is the soft maximum interval in which a step that returns a Running + status (which typically indicates it's waiting for something to happen) + may be retried. + + The maximum is a soft one because the check for whether the interval has + elapsed occurs AFTER the step has run. This effectively means a step may + run ONCE beyond the close of the interval. + + If this field is set to nil, the effective default will be a step-specific + one. If no step-specific default exists (i.e. is also nil), the effective + default will be the system-wide default of 0. + + A value of 0 will cause the step to be retried indefinitely unless the + ErrorThreshold is reached. + type: string type: object uses: description: Uses identifies a runner that can execute this @@ -161,12 +189,6 @@ spec: subsequent reconciliations attempts. format: int64 type: integer - currentStepAttempt: - description: |- - CurrentStepAttempt is the number of times the current step has been - attempted. - format: int64 - type: integer finishedAt: description: FinishedAt is the time when the promotion was completed. format: date-time @@ -537,6 +559,44 @@ spec: State stores the state of the promotion process between reconciliation attempts. x-kubernetes-preserve-unknown-fields: true + stepExecutionMetadata: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution + of individual promotion steps. + items: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution of + a promotion step. + properties: + alias: + description: Alias is the alias of the step. + type: string + errorCount: + description: ErrorCount tracks consecutive failed attempts to + execute the step. + format: int32 + type: integer + finishedAt: + description: |- + FinishedAt is the time at which the final attempt to execute the step + completed. + format: date-time + type: string + message: + description: Message is a display message about the step, including + any errors. + type: string + startedAt: + description: |- + StartedAt is the time at which the first attempt to execute the step + began. + format: date-time + type: string + status: + description: Status is the high-level outcome of the step. + type: string + type: object + type: array type: object required: - spec diff --git a/charts/kargo/resources/crds/kargo.akuity.io_stages.yaml b/charts/kargo/resources/crds/kargo.akuity.io_stages.yaml index 5e471fa9d..fabd57e02 100644 --- a/charts/kargo/resources/crds/kargo.akuity.io_stages.yaml +++ b/charts/kargo/resources/crds/kargo.akuity.io_stages.yaml @@ -91,18 +91,46 @@ spec: retry: description: Retry is the retry policy for this step. properties: - attempts: + errorThreshold: description: |- - Attempts is the number of times the step can be attempted before the - PromotionStep is marked as failed. + ErrorThreshold is the number of consecutive times the step must fail (for + any reason) before retries are abandoned and the entire Promotion is marked + as failed. - If this field is set to 1, the step will not be retried. If this - field is set to -1, the step will be retried indefinitely. + If this field is set to 0, the effective default will be a step-specific + one. If no step-specific default exists (i.e. is also 0), the effective + default will be the system-wide default of 1. - The default of this field depends on the step being executed. Refer to - the documentation for the specific step for more information. - format: int64 + A value of 1 will cause the Promotion to be marked as failed after just + a single failure; i.e. no retries will be attempted. + + There is no option to specify an infinite number of retries using a value + such as -1. + + In a future release, Kargo is likely to become capable of distinguishing + between recoverable and non-recoverable step failures. At that time, it is + planned that unrecoverable failures will not be subject to this threshold + and will immediately cause the Promotion to be marked as failed without + further condition. + format: int32 type: integer + timeout: + description: |- + Timeout is the soft maximum interval in which a step that returns a Running + status (which typically indicates it's waiting for something to happen) + may be retried. + + The maximum is a soft one because the check for whether the interval has + elapsed occurs AFTER the step has run. This effectively means a step may + run ONCE beyond the close of the interval. + + If this field is set to nil, the effective default will be a step-specific + one. If no step-specific default exists (i.e. is also nil), the effective + default will be the system-wide default of 0. + + A value of 0 will cause the step to be retried indefinitely unless the + ErrorThreshold is reached. + type: string type: object uses: description: Uses identifies a runner that can execute @@ -485,12 +513,6 @@ spec: subsequent reconciliations attempts. format: int64 type: integer - currentStepAttempt: - description: |- - CurrentStepAttempt is the number of times the current step has been - attempted. - format: int64 - type: integer finishedAt: description: FinishedAt is the time when the promotion was completed. @@ -869,6 +891,45 @@ spec: State stores the state of the promotion process between reconciliation attempts. x-kubernetes-preserve-unknown-fields: true + stepExecutionMetadata: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution + of individual promotion steps. + items: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution of + a promotion step. + properties: + alias: + description: Alias is the alias of the step. + type: string + errorCount: + description: ErrorCount tracks consecutive failed attempts + to execute the step. + format: int32 + type: integer + finishedAt: + description: |- + FinishedAt is the time at which the final attempt to execute the step + completed. + format: date-time + type: string + message: + description: Message is a display message about the + step, including any errors. + type: string + startedAt: + description: |- + StartedAt is the time at which the first attempt to execute the step + began. + format: date-time + type: string + status: + description: Status is the high-level outcome of the + step. + type: string + type: object + type: array type: object required: - name @@ -1269,12 +1330,6 @@ spec: subsequent reconciliations attempts. format: int64 type: integer - currentStepAttempt: - description: |- - CurrentStepAttempt is the number of times the current step has been - attempted. - format: int64 - type: integer finishedAt: description: FinishedAt is the time when the promotion was completed. @@ -1653,6 +1708,45 @@ spec: State stores the state of the promotion process between reconciliation attempts. x-kubernetes-preserve-unknown-fields: true + stepExecutionMetadata: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution + of individual promotion steps. + items: + description: |- + StepExecutionMetadata tracks metadata pertaining to the execution of + a promotion step. + properties: + alias: + description: Alias is the alias of the step. + type: string + errorCount: + description: ErrorCount tracks consecutive failed attempts + to execute the step. + format: int32 + type: integer + finishedAt: + description: |- + FinishedAt is the time at which the final attempt to execute the step + completed. + format: date-time + type: string + message: + description: Message is a display message about the + step, including any errors. + type: string + startedAt: + description: |- + StartedAt is the time at which the first attempt to execute the step + began. + format: date-time + type: string + status: + description: Status is the high-level outcome of the + step. + type: string + type: object + type: array type: object required: - name diff --git a/internal/controller/promotions/promotions.go b/internal/controller/promotions/promotions.go index 138cc5beb..57014318d 100644 --- a/internal/controller/promotions/promotions.go +++ b/internal/controller/promotions/promotions.go @@ -476,17 +476,17 @@ func (r *reconciler) promote( } promoCtx := directives.PromotionContext{ - UIBaseURL: r.cfg.APIServerBaseURL, - WorkDir: filepath.Join(os.TempDir(), "promotion-"+string(workingPromo.UID)), - Project: stageNamespace, - Stage: stageName, - Promotion: workingPromo.Name, - FreightRequests: stage.Spec.RequestedFreight, - Freight: *workingPromo.Status.FreightCollection.DeepCopy(), - StartFromStep: promo.Status.CurrentStep, - Attempts: promo.Status.CurrentStepAttempt, - State: directives.State(workingPromo.Status.GetState()), - Vars: workingPromo.Spec.Vars, + UIBaseURL: r.cfg.APIServerBaseURL, + WorkDir: filepath.Join(os.TempDir(), "promotion-"+string(workingPromo.UID)), + Project: stageNamespace, + Stage: stageName, + Promotion: workingPromo.Name, + FreightRequests: stage.Spec.RequestedFreight, + Freight: *workingPromo.Status.FreightCollection.DeepCopy(), + StartFromStep: promo.Status.CurrentStep, + StepExecutionMetadata: promo.Status.StepExecutionMetadata, + State: directives.State(workingPromo.Status.GetState()), + Vars: workingPromo.Spec.Vars, } if err := os.Mkdir(promoCtx.WorkDir, 0o700); err == nil { // If we're working with a fresh directory, we should start the promotion @@ -494,7 +494,8 @@ func (r *reconciler) promote( // allows individual steps to self-discover that they've run before and // examine the results of their own previous execution. promoCtx.StartFromStep = 0 - promoCtx.Attempts = 0 + promoCtx.StepExecutionMetadata = nil + workingPromo.Status.HealthChecks = nil } else if !os.IsExist(err) { return nil, fmt.Errorf("error creating working directory: %w", err) } @@ -510,17 +511,16 @@ func (r *reconciler) promote( workingPromo.Status.Phase = res.Status workingPromo.Status.Message = res.Message workingPromo.Status.CurrentStep = res.CurrentStep - workingPromo.Status.CurrentStepAttempt = res.Attempt + workingPromo.Status.StepExecutionMetadata = res.StepExecutionMetadata workingPromo.Status.State = &apiextensionsv1.JSON{Raw: res.State.ToJSON()} - if res.Status == kargoapi.PromotionPhaseSucceeded { - var healthChecks []kargoapi.HealthCheckStep - for _, step := range res.HealthCheckSteps { - healthChecks = append(healthChecks, kargoapi.HealthCheckStep{ + for _, step := range res.HealthCheckSteps { + workingPromo.Status.HealthChecks = append( + workingPromo.Status.HealthChecks, + kargoapi.HealthCheckStep{ Uses: step.Kind, Config: &apiextensionsv1.JSON{Raw: step.Config.ToJSON()}, - }) - } - workingPromo.Status.HealthChecks = healthChecks + }, + ) } if err != nil { workingPromo.Status.Phase = kargoapi.PromotionPhaseErrored diff --git a/internal/directives/argocd_updater.go b/internal/directives/argocd_updater.go index 770e86e51..afff0328d 100644 --- a/internal/directives/argocd_updater.go +++ b/internal/directives/argocd_updater.go @@ -11,6 +11,7 @@ import ( corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" + "k8s.io/utils/ptr" "sigs.k8s.io/controller-runtime/pkg/client" kargoapi "github.com/akuity/kargo/api/v1alpha1" @@ -125,9 +126,14 @@ func (a *argocdUpdater) Name() string { return "argocd-update" } -// DefaultAttempts implements the RetryableStepRunner interface. -func (a *argocdUpdater) DefaultAttempts() int64 { - return -1 +// DefaultTimeout implements the RetryableStepRunner interface. +func (a *argocdUpdater) DefaultTimeout() *time.Duration { + return ptr.To(5 * time.Minute) +} + +// DefaultErrorThreshold implements the RetryableStepRunner interface. +func (a *argocdUpdater) DefaultErrorThreshold() uint32 { + return 0 // Will fall back to the system default. } // RunPromotionStep implements the PromotionStepRunner interface. diff --git a/internal/directives/git_pr_waiter.go b/internal/directives/git_pr_waiter.go index 531a13b4a..748bdc581 100644 --- a/internal/directives/git_pr_waiter.go +++ b/internal/directives/git_pr_waiter.go @@ -38,11 +38,6 @@ func (g *gitPRWaiter) Name() string { return "git-wait-for-pr" } -// DefaultAttempts implements the RetryableStepRunner interface. -func (g *gitPRWaiter) DefaultAttempts() int64 { - return -1 -} - // RunPromotionStep implements the PromotionStepRunner interface. func (g *gitPRWaiter) RunPromotionStep( ctx context.Context, diff --git a/internal/directives/promotions.go b/internal/directives/promotions.go index 76a68ed84..a4af12e14 100644 --- a/internal/directives/promotions.go +++ b/internal/directives/promotions.go @@ -3,8 +3,10 @@ package directives import ( "context" "fmt" + "time" "github.com/expr-lang/expr" + "k8s.io/utils/ptr" "sigs.k8s.io/controller-runtime/pkg/client" yaml "sigs.k8s.io/yaml/goyaml.v3" @@ -30,9 +32,12 @@ type PromotionStepRunner interface { // RetryableStepRunner is an interface for PromotionStepRunners that can be // retried in the event of a failure. type RetryableStepRunner interface { - // DefaultAttempts returns the default number of attempts the step is - // allowed to make before failing. - DefaultAttempts() int64 + // DefaultTimeout returns the default timeout for the step. + DefaultTimeout() *time.Duration + // DefaultErrorThreshold returns the number of consecutive times the step must + // fail (for any reason) before retries are abandoned and the entire Promotion + // is marked as failed. + DefaultErrorThreshold() uint32 } // PromotionContext is the context of a user-defined promotion process that is @@ -65,9 +70,9 @@ type PromotionContext struct { // StartFromStep is the index of the step from which the promotion should // begin execution. StartFromStep int64 - // Attempts is the number of attempts that have been made to execute - // the current step. - Attempts int64 + // StepExecutionMetadata tracks metadata pertaining to the execution + // of individual promotion steps. + StepExecutionMetadata kargoapi.StepExecutionMetadataList // State is the current state of the promotion process. State State // Vars is a list of variables definitions that can be used by the @@ -96,16 +101,29 @@ type PromotionStep struct { Config []byte } -// GetMaxAttempts returns the maximum number of attempts that can be made to -// execute the step using the provided runner. If the runner is a -// RetryableStepRunner, the value of its retry configuration is used as the -// maximum default. Otherwise, the default is 1. -func (s *PromotionStep) GetMaxAttempts(runner any) int64 { - fallback := int64(1) +// GetTimeout returns the maximum interval the provided runner may spend +// attempting to execute the step before retries are abandoned and the entire +// Promotion is marked as failed. If the runner is a RetryableStepRunner, its +// timeout is used as the default. Otherwise, the default is 0 (no limit). +func (s *PromotionStep) GetTimeout(runner any) *time.Duration { + fallback := ptr.To(time.Duration(0)) if retryCfg, isRetryable := runner.(RetryableStepRunner); isRetryable { - fallback = retryCfg.DefaultAttempts() + fallback = retryCfg.DefaultTimeout() } - return s.Retry.GetAttempts(fallback) + return s.Retry.GetTimeout(fallback) +} + +// GetErrorThreshold returns the number of consecutive times the provided runner +// must fail to execute the step (for any reason) before retries are abandoned +// and the entire Promotion is marked as failed. If the runner is a +// RetryableStepRunner, its threshold is used as the default. Otherwise, the +// default is 1. +func (s *PromotionStep) GetErrorThreshold(runner any) uint32 { + fallback := uint32(1) + if retryCfg, isRetryable := runner.(RetryableStepRunner); isRetryable { + fallback = retryCfg.DefaultErrorThreshold() + } + return s.Retry.GetErrorThreshold(fallback) } // GetConfig returns the Config unmarshalled into a map. Any expr-lang @@ -214,8 +232,9 @@ type PromotionResult struct { // in some external state, the value of this field will indicate where to // resume the process in the next reconciliation. CurrentStep int64 - // Attempt tracks the current execution attempt of the current step. - Attempt int64 + // StepExecutionMetadata tracks metadata pertaining to the execution + // of individual promotion steps. + StepExecutionMetadata kargoapi.StepExecutionMetadataList // State is the current state of the promotion process. State State } diff --git a/internal/directives/promotions_test.go b/internal/directives/promotions_test.go index 00cea7298..35871f19a 100644 --- a/internal/directives/promotions_test.go +++ b/internal/directives/promotions_test.go @@ -3,81 +3,153 @@ package directives import ( "context" "testing" + "time" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" v1 "k8s.io/apimachinery/pkg/apis/meta/v1" k8sruntime "k8s.io/apimachinery/pkg/runtime" + "k8s.io/utils/ptr" "sigs.k8s.io/controller-runtime/pkg/client/fake" kargoapi "github.com/akuity/kargo/api/v1alpha1" ) type mockRetryableRunner struct { - defaultAttempts int64 + defaultTimeout *time.Duration + defaultErrorThreshold uint32 } -func (m mockRetryableRunner) DefaultAttempts() int64 { - return m.defaultAttempts +func (m mockRetryableRunner) DefaultTimeout() *time.Duration { + return m.defaultTimeout } -func TestPromotionStep_GetMaxAttempts(t *testing.T) { +func (m mockRetryableRunner) DefaultErrorThreshold() uint32 { + return m.defaultErrorThreshold +} + +func TestPromotionStep_GetTimeout(t *testing.T) { + tests := []struct { + name string + step *PromotionStep + runner any + assertions func(t *testing.T, result *time.Duration) + }{ + { + name: "returns 0 with no retry config", + step: &PromotionStep{ + Retry: nil, + }, + assertions: func(t *testing.T, result *time.Duration) { + assert.Equal(t, ptr.To(time.Duration(0)), result) + }, + }, + { + name: "returns configured timeout for non-retryable runner", + step: &PromotionStep{ + Retry: &kargoapi.PromotionStepRetry{ + Timeout: &metav1.Duration{ + Duration: time.Duration(5), + }, + }, + }, + runner: nil, + assertions: func(t *testing.T, result *time.Duration) { + assert.Equal(t, ptr.To(time.Duration(5)), result) + }, + }, + { + name: "returns configured timeout for retryable runner", + step: &PromotionStep{ + Retry: &kargoapi.PromotionStepRetry{ + Timeout: &metav1.Duration{ + Duration: time.Duration(5), + }, + }, + }, + runner: mockRetryableRunner{defaultTimeout: ptr.To(time.Duration(3))}, + assertions: func(t *testing.T, result *time.Duration) { + assert.Equal(t, ptr.To(time.Duration(5)), result) + }, + }, + { + name: "returns default timeout when retry config returns nil", + step: &PromotionStep{ + Retry: &kargoapi.PromotionStepRetry{}, + }, + runner: mockRetryableRunner{defaultTimeout: ptr.To(time.Duration(3))}, + assertions: func(t *testing.T, result *time.Duration) { + assert.Equal(t, ptr.To(time.Duration(3)), result) + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + result := tt.step.GetTimeout(tt.runner) + tt.assertions(t, result) + }) + } +} + +func TestPromotionStep_GetErrorThreshold(t *testing.T) { tests := []struct { name string step *PromotionStep runner any - assertions func(t *testing.T, result int64) + assertions func(t *testing.T, result uint32) }{ { name: "returns 1 with no retry config", step: &PromotionStep{ Retry: nil, }, - assertions: func(t *testing.T, result int64) { - assert.Equal(t, int64(1), result) + assertions: func(t *testing.T, result uint32) { + assert.Equal(t, uint32(1), result) }, }, { - name: "returns configured attempts for non-retryable runner", + name: "returns configured threshold for non-retryable runner", step: &PromotionStep{ Retry: &kargoapi.PromotionStepRetry{ - Attempts: 5, + ErrorThreshold: 5, }, }, runner: nil, - assertions: func(t *testing.T, result int64) { - assert.Equal(t, int64(5), result) + assertions: func(t *testing.T, result uint32) { + assert.Equal(t, uint32(5), result) }, }, { - name: "returns configured attempts for retryable runner", + name: "returns configured threshold for retryable runner", step: &PromotionStep{ Retry: &kargoapi.PromotionStepRetry{ - Attempts: 5, + ErrorThreshold: 5, }, }, - runner: mockRetryableRunner{defaultAttempts: 3}, - assertions: func(t *testing.T, result int64) { - assert.Equal(t, int64(5), result) + runner: mockRetryableRunner{defaultErrorThreshold: 3}, + assertions: func(t *testing.T, result uint32) { + assert.Equal(t, uint32(5), result) }, }, { - name: "returns default attempts when retry config returns 0", + name: "returns default threshold when retry config returns 0", step: &PromotionStep{ Retry: &kargoapi.PromotionStepRetry{ - Attempts: 0, + ErrorThreshold: 0, }, }, - runner: mockRetryableRunner{defaultAttempts: 3}, - assertions: func(t *testing.T, result int64) { - assert.Equal(t, int64(3), result) + runner: mockRetryableRunner{defaultErrorThreshold: 3}, + assertions: func(t *testing.T, result uint32) { + assert.Equal(t, uint32(3), result) }, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { - result := tt.step.GetMaxAttempts(tt.runner) + result := tt.step.GetErrorThreshold(tt.runner) tt.assertions(t, result) }) } diff --git a/internal/directives/simple_engine_promote.go b/internal/directives/simple_engine_promote.go index cb221a04a..d262e83b8 100644 --- a/internal/directives/simple_engine_promote.go +++ b/internal/directives/simple_engine_promote.go @@ -7,6 +7,8 @@ import ( "strings" corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/utils/ptr" "sigs.k8s.io/controller-runtime/pkg/client" kargoapi "github.com/akuity/kargo/api/v1alpha1" @@ -55,9 +57,9 @@ func (e *SimpleEngine) executeSteps( } var ( - healthChecks []HealthCheckStep - err error - attempt = promoCtx.Attempts + healthChecks []HealthCheckStep + err error + stepExecMetas = promoCtx.StepExecutionMetadata.DeepCopy() ) // Execute each step in sequence, starting from the step index @@ -66,9 +68,11 @@ func (e *SimpleEngine) executeSteps( select { case <-ctx.Done(): return PromotionResult{ - Status: kargoapi.PromotionPhaseErrored, - CurrentStep: i, - State: state, + Status: kargoapi.PromotionPhaseErrored, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, }, ctx.Err() default: } @@ -77,94 +81,130 @@ func (e *SimpleEngine) executeSteps( step := steps[i] if step.Alias, err = e.stepAlias(step.Alias, i); err != nil { return PromotionResult{ - Status: kargoapi.PromotionPhaseErrored, - CurrentStep: i, - State: state, - }, err + Status: kargoapi.PromotionPhaseErrored, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, fmt.Errorf("error getting step alias for step %d: %w", i, err) } // Get the PromotionStepRunner for the step. reg, err := e.registry.GetPromotionStepRunnerRegistration(step.Kind) if err != nil { return PromotionResult{ - Status: kargoapi.PromotionPhaseErrored, - CurrentStep: i, - State: state, - }, err + Status: kargoapi.PromotionPhaseErrored, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, fmt.Errorf("error getting runner for step %d: %w", i, err) } - // Check if the step has exceeded the maximum number of attempts. - maxAttempts := step.GetMaxAttempts(reg.Runner) - if maxAttempts > 0 && attempt >= maxAttempts { - return PromotionResult{ - Status: kargoapi.PromotionPhaseErrored, - CurrentStep: i, - State: state, - Attempt: attempt, - }, fmt.Errorf("step %q exceeded max attempts", step.Alias) + // If we don't have metadata for this step yet, create it. + if int64(len(stepExecMetas)) == i { + stepExecMetas = append(stepExecMetas, kargoapi.StepExecutionMetadata{ + Alias: step.Alias, + StartedAt: ptr.To(metav1.Now()), + }) } + stepExecMeta := &stepExecMetas[i] - // Count the attempt we are about to make. - attempt++ - - // Execute the step. + // Execute the step result, err := e.executeStep(ctx, promoCtx, step, reg, workDir, state) + if err != nil { + // Let a hard error take precedence over the result status and message. + stepExecMeta.Status = kargoapi.PromotionPhaseErrored + stepExecMeta.Message = err.Error() + } else { + stepExecMeta.Status = result.Status + stepExecMeta.Message = result.Message + } + state[step.Alias] = result.Output - // If the step failed, and the maximum number of attempts has not been - // reached, we are still "Running" the step and will retry it. - if err != nil || result.Status == kargoapi.PromotionPhaseErrored || result.Status == kargoapi.PromotionPhaseFailed { - if maxAttempts < 0 || attempt < maxAttempts { - var message strings.Builder - _, _ = message.WriteString(fmt.Sprintf("step %q failed (attempt %d)", step.Alias, attempt)) - if result.Message != "" { - _, _ = message.WriteString(": ") - _, _ = message.WriteString(result.Message) - } - if err != nil { - _, _ = message.WriteString(": ") - _, _ = message.WriteString(err.Error()) - } - - // Update the result to indicate that the step is still running. - result.Status = kargoapi.PromotionPhaseRunning - result.Message = message.String() - - // Swallow the error if the step failed, as we are still - // retrying it. - err = nil + if stepExecMeta.Status == kargoapi.PromotionPhaseSucceeded { + stepExecMeta.FinishedAt = ptr.To(metav1.Now()) + if healthCheck := result.HealthCheckStep; healthCheck != nil { + healthChecks = append(healthChecks, *healthCheck) } + continue // Move on to the next step } - // Update the state with the step output, regardless of the result. - state[step.Alias] = result.Output + // Treat errors and logical failures the same for now. + // TODO(krancour): In the future, we should fail without retry for logical + // failures and unrecoverable errors and retry only those errors with a + // chance of recovery. + if stepExecMeta.Status != kargoapi.PromotionPhaseRunning { + stepExecMeta.ErrorCount++ + // Check if the error threshold has been met. + errorThreshold := step.GetErrorThreshold(reg.Runner) + if stepExecMeta.ErrorCount >= errorThreshold { + // The error threshold has been met. + stepExecMeta.FinishedAt = ptr.To(metav1.Now()) + return PromotionResult{ + Status: kargoapi.PromotionPhaseErrored, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, fmt.Errorf( + "step %d met error threshold of %d: %s", i, + errorThreshold, stepExecMeta.Message, + ) + } + } - // If the step was not successful, return the result to wait for - // a next attempt or to fail the promotion. - if result.Status != kargoapi.PromotionPhaseSucceeded { + // If we get to here, the step is either running (waiting for some external + // condition to be met) or it errored/failed but did not meet the error + // threshold. Now we need to check if the timeout has elapsed. A nil timeout + // or any non-positive timeout interval are treated as NO timeout, although + // a nil timeout really shouldn't happen. + timeout := step.GetTimeout(reg.Runner) + if timeout != nil && *timeout > 0 && metav1.Now().Sub(stepExecMeta.StartedAt.Time) > *timeout { + // Timeout has elapsed. + stepExecMeta.FinishedAt = ptr.To(metav1.Now()) return PromotionResult{ - Status: result.Status, - Message: result.Message, - CurrentStep: i, - Attempt: attempt, - State: state, - }, err + Status: kargoapi.PromotionPhaseErrored, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, fmt.Errorf("step %d timeout of %s has elapsed", i, timeout.String()) } - // If the step was successful, reset the attempts counter and add its - // health check to the list. - attempt = 0 - if healthCheck := result.HealthCheckStep; healthCheck != nil { - healthChecks = append(healthChecks, *healthCheck) + if stepExecMeta.Status != kargoapi.PromotionPhaseRunning { + // Treat the error/failure as if the step is still running so that the + // Promotion will be requeued. The step will be retried on the next + // reconciliation. + stepExecMeta.Message += "; step will be retried" + return PromotionResult{ + Status: kargoapi.PromotionPhaseRunning, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, nil } + + // If we get to here, the step is still running (waiting for some external + // condition to be met). + stepExecMeta.ErrorCount = 0 // Reset the error count + return PromotionResult{ + Status: kargoapi.PromotionPhaseRunning, + CurrentStep: i, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, + }, nil } // All steps have succeeded, return the final state. return PromotionResult{ - Status: kargoapi.PromotionPhaseSucceeded, - HealthCheckSteps: healthChecks, - CurrentStep: int64(len(steps)) - 1, - Attempt: 0, - State: state, + Status: kargoapi.PromotionPhaseSucceeded, + CurrentStep: int64(len(steps)) - 1, + StepExecutionMetadata: stepExecMetas, + State: state, + HealthCheckSteps: healthChecks, }, nil } diff --git a/internal/directives/simple_engine_promote_test.go b/internal/directives/simple_engine_promote_test.go index ed1a5cc4c..28625a6ac 100644 --- a/internal/directives/simple_engine_promote_test.go +++ b/internal/directives/simple_engine_promote_test.go @@ -5,10 +5,12 @@ import ( "errors" "os" "testing" + "time" "github.com/stretchr/testify/assert" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/utils/ptr" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client/fake" "sigs.k8s.io/controller-runtime/pkg/client/interceptor" @@ -65,9 +67,11 @@ func TestSimpleEngine_Promote(t *testing.T) { {Kind: "context-waiter"}, }, assertions: func(t *testing.T, result PromotionResult, err error) { - assert.Error(t, err) + assert.ErrorContains(t, err, "met error threshold") assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) - assert.ErrorIs(t, err, context.Canceled) + assert.Len(t, result.StepExecutionMetadata, 1) + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.StepExecutionMetadata[0].Status) + assert.Contains(t, result.StepExecutionMetadata[0].Message, context.Canceled.Error()) }, }, { @@ -144,6 +148,26 @@ func TestSimpleEngine_executeSteps(t *testing.T) { steps []PromotionStep assertions func(*testing.T, PromotionResult, error) }{ + { + name: "fail on invalid step alias", + steps: []PromotionStep{ + {Kind: "success-step", Alias: "step-1"}, + }, + assertions: func(t *testing.T, result PromotionResult, err error) { + assert.ErrorContains(t, err, "is forbidden") + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) + assert.Equal(t, int64(0), result.CurrentStep) + }, + }, + { + name: "runner not found", + steps: []PromotionStep{{Kind: "unknown-step"}}, + assertions: func(t *testing.T, result PromotionResult, err error) { + assert.ErrorContains(t, err, "error getting runner for step") + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) + assert.Equal(t, int64(0), result.CurrentStep) + }, + }, { name: "execute all steps successfully", steps: []PromotionStep{ @@ -154,7 +178,14 @@ func TestSimpleEngine_executeSteps(t *testing.T) { assert.NoError(t, err) assert.Equal(t, kargoapi.PromotionPhaseSucceeded, result.Status) assert.Equal(t, int64(1), result.CurrentStep) - assert.Equal(t, int64(0), result.Attempt) + + // Verify the result contains metadata from both steps + assert.Len(t, result.StepExecutionMetadata, 2) + for _, metadata := range result.StepExecutionMetadata { + assert.Equal(t, kargoapi.PromotionPhaseSucceeded, metadata.Status) + assert.NotNil(t, metadata.StartedAt) + assert.NotNil(t, metadata.FinishedAt) + } // Verify state contains outputs from both steps assert.Equal(t, State{ @@ -171,16 +202,29 @@ func TestSimpleEngine_executeSteps(t *testing.T) { name: "start from middle step", promoCtx: PromotionContext{ StartFromStep: 1, + // Dummy metadata for the 0 step, which must have succeeded already if + // we're starting from step 1 + StepExecutionMetadata: kargoapi.StepExecutionMetadataList{{}}, }, steps: []PromotionStep{ - {Kind: "error-step", Alias: "step1"}, // This step should be skipped + // This step must have already succeeded and should not be run again + // this time. + {Kind: "error-step", Alias: "step1"}, + // This step should be run {Kind: "success-step", Alias: "step2"}, }, assertions: func(t *testing.T, result PromotionResult, err error) { assert.NoError(t, err) assert.Equal(t, kargoapi.PromotionPhaseSucceeded, result.Status) assert.Equal(t, int64(1), result.CurrentStep) - assert.Equal(t, int64(0), result.Attempt) + + // Verify the result contains metadata from both steps + assert.Len(t, result.StepExecutionMetadata, 2) + // We're not bothering with assertions on the dummy metadata for the 0 + // step. + assert.Equal(t, kargoapi.PromotionPhaseSucceeded, result.StepExecutionMetadata[1].Status) + assert.NotNil(t, result.StepExecutionMetadata[1].StartedAt) + assert.NotNil(t, result.StepExecutionMetadata[1].FinishedAt) // Verify only second step output is in state assert.Equal(t, State{ @@ -191,16 +235,23 @@ func TestSimpleEngine_executeSteps(t *testing.T) { }, }, { - name: "fail on step execution", + name: "error on step execution; error threshold met", steps: []PromotionStep{ {Kind: "success-step", Alias: "step1"}, {Kind: "error-step", Alias: "step2"}, }, assertions: func(t *testing.T, result PromotionResult, err error) { - assert.ErrorContains(t, err, "something went wrong") + assert.ErrorContains(t, err, "met error threshold") assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) assert.Equal(t, int64(1), result.CurrentStep) - assert.Equal(t, int64(1), result.Attempt) + assert.Len(t, result.StepExecutionMetadata, 2) + assert.Equal(t, kargoapi.PromotionPhaseSucceeded, result.StepExecutionMetadata[0].Status) + assert.NotNil(t, result.StepExecutionMetadata[0].StartedAt) + assert.NotNil(t, result.StepExecutionMetadata[0].FinishedAt) + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.StepExecutionMetadata[1].Status) + assert.NotNil(t, result.StepExecutionMetadata[1].StartedAt) + assert.NotNil(t, result.StepExecutionMetadata[1].FinishedAt) + assert.Contains(t, result.StepExecutionMetadata[1].Message, "something went wrong") // Verify first step output is preserved in state assert.Equal(t, State{ @@ -212,65 +263,75 @@ func TestSimpleEngine_executeSteps(t *testing.T) { }, }, { - name: "fail on invalid step alias", + name: "error on step execution; error threshold not met", steps: []PromotionStep{ - {Kind: "success-step", Alias: "step-1"}, + { + Kind: "error-step", + Alias: "step1", + Retry: &kargoapi.PromotionStepRetry{ErrorThreshold: 3}, + }, }, assertions: func(t *testing.T, result PromotionResult, err error) { - assert.ErrorContains(t, err, "is forbidden") - assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) + assert.NoError(t, err) + assert.Equal(t, kargoapi.PromotionPhaseRunning, result.Status) assert.Equal(t, int64(0), result.CurrentStep) + assert.Len(t, result.StepExecutionMetadata, 1) + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.StepExecutionMetadata[0].Status) + assert.NotNil(t, result.StepExecutionMetadata[0].StartedAt) + assert.Nil(t, result.StepExecutionMetadata[0].FinishedAt) + assert.Equal(t, uint32(1), result.StepExecutionMetadata[0].ErrorCount) + assert.Contains(t, result.StepExecutionMetadata[0].Message, "will be retried") }, }, { - name: "context cancellation", + name: "step is still running; timeout elapsed", + promoCtx: PromotionContext{ + StepExecutionMetadata: kargoapi.StepExecutionMetadataList{{ + // Start time is set to an hour ago + StartedAt: ptr.To(metav1.NewTime(time.Now().Add(-time.Hour))), + }}, + }, steps: []PromotionStep{ - {Kind: "context-waiter", Alias: "step1"}, + { + Kind: "running-step", + Retry: &kargoapi.PromotionStepRetry{ + Timeout: &metav1.Duration{ + Duration: time.Hour, + }, + }, + }, }, assertions: func(t *testing.T, result PromotionResult, err error) { - assert.ErrorIs(t, err, context.Canceled) + assert.ErrorContains(t, err, "timeout") + assert.ErrorContains(t, err, "has elapsed") assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) assert.Equal(t, int64(0), result.CurrentStep) }, }, { - name: "retry failed step within max attempts", - promoCtx: PromotionContext{ - Attempts: 1, - }, - steps: []PromotionStep{ - { - Kind: "error-step", - Alias: "step1", - Retry: &kargoapi.PromotionStepRetry{Attempts: 3}, - }, - }, + name: "step is still running; timeout not elapsed", + steps: []PromotionStep{{Kind: "running-step"}}, assertions: func(t *testing.T, result PromotionResult, err error) { assert.NoError(t, err) assert.Equal(t, kargoapi.PromotionPhaseRunning, result.Status) assert.Equal(t, int64(0), result.CurrentStep) - assert.Equal(t, int64(2), result.Attempt) - assert.Contains(t, result.Message, "attempt 2") + assert.Len(t, result.StepExecutionMetadata, 1) + assert.Equal(t, kargoapi.PromotionPhaseRunning, result.StepExecutionMetadata[0].Status) + assert.NotNil(t, result.StepExecutionMetadata[0].StartedAt) }, }, { - name: "max attempts exceeded", - promoCtx: PromotionContext{ - Attempts: 3, - }, + name: "context cancellation", steps: []PromotionStep{ - { - Kind: "error-step", - Alias: "step1", - Retry: &kargoapi.PromotionStepRetry{Attempts: 3}, - }, + {Kind: "context-waiter", Alias: "step1"}, }, assertions: func(t *testing.T, result PromotionResult, err error) { - assert.Error(t, err) + assert.ErrorContains(t, err, "met error threshold") assert.Equal(t, kargoapi.PromotionPhaseErrored, result.Status) assert.Equal(t, int64(0), result.CurrentStep) - assert.Equal(t, int64(3), result.Attempt) - assert.Contains(t, err.Error(), "exceeded max attempts") + assert.Len(t, result.StepExecutionMetadata, 1) + assert.Equal(t, kargoapi.PromotionPhaseErrored, result.StepExecutionMetadata[0].Status) + assert.Contains(t, result.StepExecutionMetadata[0].Message, context.Canceled.Error()) }, }, } @@ -291,6 +352,15 @@ func TestSimpleEngine_executeSteps(t *testing.T) { }, &StepRunnerPermissions{}, ) + testRegistry.RegisterPromotionStepRunner( + &mockPromotionStepRunner{ + name: "running-step", + runResult: PromotionStepResult{ + Status: kargoapi.PromotionPhaseRunning, + }, + }, + &StepRunnerPermissions{}, + ) testRegistry.RegisterPromotionStepRunner( &mockPromotionStepRunner{ name: "error-step", diff --git a/ui/src/gen/schema/promotions.kargo.akuity.io_v1alpha1.json b/ui/src/gen/schema/promotions.kargo.akuity.io_v1alpha1.json index afd56da14..c28cf84a2 100644 --- a/ui/src/gen/schema/promotions.kargo.akuity.io_v1alpha1.json +++ b/ui/src/gen/schema/promotions.kargo.akuity.io_v1alpha1.json @@ -43,12 +43,16 @@ "retry": { "description": "Retry is the retry policy for this step.", "properties": { - "attempts": { - "description": "Attempts is the number of times the step can be attempted before the\nPromotionStep is marked as failed.\n\nIf this field is set to 1, the step will not be retried. If this\nfield is set to -1, the step will be retried indefinitely.\n\nThe default of this field depends on the step being executed. Refer to\nthe documentation for the specific step for more information.", - "format": "int64", - "maximum": 9223372036854776000, - "minimum": -9223372036854776000, + "errorThreshold": { + "description": "ErrorThreshold is the number of consecutive times the step must fail (for\nany reason) before retries are abandoned and the entire Promotion is marked\nas failed.\n\nIf this field is set to 0, the effective default will be a step-specific\none. If no step-specific default exists (i.e. is also 0), the effective\ndefault will be the system-wide default of 1.\n\nA value of 1 will cause the Promotion to be marked as failed after just\na single failure; i.e. no retries will be attempted.\n\nThere is no option to specify an infinite number of retries using a value\nsuch as -1.\n\nIn a future release, Kargo is likely to become capable of distinguishing\nbetween recoverable and non-recoverable step failures. At that time, it is\nplanned that unrecoverable failures will not be subject to this threshold\nand will immediately cause the Promotion to be marked as failed without\nfurther condition.", + "format": "int32", + "maximum": 2147483647, + "minimum": -2147483648, "type": "integer" + }, + "timeout": { + "description": "Timeout is the soft maximum interval in which a step that returns a Running\nstatus (which typically indicates it's waiting for something to happen)\nmay be retried.\n\nThe maximum is a soft one because the check for whether the interval has\nelapsed occurs AFTER the step has run. This effectively means a step may\nrun ONCE beyond the close of the interval.\n\nIf this field is set to nil, the effective default will be a step-specific\none. If no step-specific default exists (i.e. is also nil), the effective\ndefault will be the system-wide default of 0.\n\nA value of 0 will cause the step to be retried indefinitely unless the\nErrorThreshold is reached.", + "type": "string" } }, "type": "object" @@ -107,13 +111,6 @@ "minimum": -9223372036854776000, "type": "integer" }, - "currentStepAttempt": { - "description": "CurrentStepAttempt is the number of times the current step has been\nattempted.", - "format": "int64", - "maximum": 9223372036854776000, - "minimum": -9223372036854776000, - "type": "integer" - }, "finishedAt": { "description": "FinishedAt is the time when the promotion was completed.", "format": "date-time", @@ -466,6 +463,45 @@ "state": { "description": "State stores the state of the promotion process between reconciliation\nattempts.", "x-kubernetes-preserve-unknown-fields": true + }, + "stepExecutionMetadata": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution\nof individual promotion steps.", + "items": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution of\na promotion step.", + "properties": { + "alias": { + "description": "Alias is the alias of the step.", + "type": "string" + }, + "errorCount": { + "description": "ErrorCount tracks consecutive failed attempts to execute the step.", + "format": "int32", + "maximum": 2147483647, + "minimum": -2147483648, + "type": "integer" + }, + "finishedAt": { + "description": "FinishedAt is the time at which the final attempt to execute the step\ncompleted.", + "format": "date-time", + "type": "string" + }, + "message": { + "description": "Message is a display message about the step, including any errors.", + "type": "string" + }, + "startedAt": { + "description": "StartedAt is the time at which the first attempt to execute the step\nbegan.", + "format": "date-time", + "type": "string" + }, + "status": { + "description": "Status is the high-level outcome of the step.", + "type": "string" + } + }, + "type": "object" + }, + "type": "array" } }, "type": "object" diff --git a/ui/src/gen/schema/stages.kargo.akuity.io_v1alpha1.json b/ui/src/gen/schema/stages.kargo.akuity.io_v1alpha1.json index c82f3a025..c912502af 100644 --- a/ui/src/gen/schema/stages.kargo.akuity.io_v1alpha1.json +++ b/ui/src/gen/schema/stages.kargo.akuity.io_v1alpha1.json @@ -38,12 +38,16 @@ "retry": { "description": "Retry is the retry policy for this step.", "properties": { - "attempts": { - "description": "Attempts is the number of times the step can be attempted before the\nPromotionStep is marked as failed.\n\nIf this field is set to 1, the step will not be retried. If this\nfield is set to -1, the step will be retried indefinitely.\n\nThe default of this field depends on the step being executed. Refer to\nthe documentation for the specific step for more information.", - "format": "int64", - "maximum": 9223372036854776000, - "minimum": -9223372036854776000, + "errorThreshold": { + "description": "ErrorThreshold is the number of consecutive times the step must fail (for\nany reason) before retries are abandoned and the entire Promotion is marked\nas failed.\n\nIf this field is set to 0, the effective default will be a step-specific\none. If no step-specific default exists (i.e. is also 0), the effective\ndefault will be the system-wide default of 1.\n\nA value of 1 will cause the Promotion to be marked as failed after just\na single failure; i.e. no retries will be attempted.\n\nThere is no option to specify an infinite number of retries using a value\nsuch as -1.\n\nIn a future release, Kargo is likely to become capable of distinguishing\nbetween recoverable and non-recoverable step failures. At that time, it is\nplanned that unrecoverable failures will not be subject to this threshold\nand will immediately cause the Promotion to be marked as failed without\nfurther condition.", + "format": "int32", + "maximum": 2147483647, + "minimum": -2147483648, "type": "integer" + }, + "timeout": { + "description": "Timeout is the soft maximum interval in which a step that returns a Running\nstatus (which typically indicates it's waiting for something to happen)\nmay be retried.\n\nThe maximum is a soft one because the check for whether the interval has\nelapsed occurs AFTER the step has run. This effectively means a step may\nrun ONCE beyond the close of the interval.\n\nIf this field is set to nil, the effective default will be a step-specific\none. If no step-specific default exists (i.e. is also nil), the effective\ndefault will be the system-wide default of 0.\n\nA value of 0 will cause the step to be retried indefinitely unless the\nErrorThreshold is reached.", + "type": "string" } }, "type": "object" @@ -425,13 +429,6 @@ "minimum": -9223372036854776000, "type": "integer" }, - "currentStepAttempt": { - "description": "CurrentStepAttempt is the number of times the current step has been\nattempted.", - "format": "int64", - "maximum": 9223372036854776000, - "minimum": -9223372036854776000, - "type": "integer" - }, "finishedAt": { "description": "FinishedAt is the time when the promotion was completed.", "format": "date-time", @@ -784,6 +781,45 @@ "state": { "description": "State stores the state of the promotion process between reconciliation\nattempts.", "x-kubernetes-preserve-unknown-fields": true + }, + "stepExecutionMetadata": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution\nof individual promotion steps.", + "items": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution of\na promotion step.", + "properties": { + "alias": { + "description": "Alias is the alias of the step.", + "type": "string" + }, + "errorCount": { + "description": "ErrorCount tracks consecutive failed attempts to execute the step.", + "format": "int32", + "maximum": 2147483647, + "minimum": -2147483648, + "type": "integer" + }, + "finishedAt": { + "description": "FinishedAt is the time at which the final attempt to execute the step\ncompleted.", + "format": "date-time", + "type": "string" + }, + "message": { + "description": "Message is a display message about the step, including any errors.", + "type": "string" + }, + "startedAt": { + "description": "StartedAt is the time at which the first attempt to execute the step\nbegan.", + "format": "date-time", + "type": "string" + }, + "status": { + "description": "Status is the high-level outcome of the step.", + "type": "string" + } + }, + "type": "object" + }, + "type": "array" } }, "type": "object" @@ -1163,13 +1199,6 @@ "minimum": -9223372036854776000, "type": "integer" }, - "currentStepAttempt": { - "description": "CurrentStepAttempt is the number of times the current step has been\nattempted.", - "format": "int64", - "maximum": 9223372036854776000, - "minimum": -9223372036854776000, - "type": "integer" - }, "finishedAt": { "description": "FinishedAt is the time when the promotion was completed.", "format": "date-time", @@ -1522,6 +1551,45 @@ "state": { "description": "State stores the state of the promotion process between reconciliation\nattempts.", "x-kubernetes-preserve-unknown-fields": true + }, + "stepExecutionMetadata": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution\nof individual promotion steps.", + "items": { + "description": "StepExecutionMetadata tracks metadata pertaining to the execution of\na promotion step.", + "properties": { + "alias": { + "description": "Alias is the alias of the step.", + "type": "string" + }, + "errorCount": { + "description": "ErrorCount tracks consecutive failed attempts to execute the step.", + "format": "int32", + "maximum": 2147483647, + "minimum": -2147483648, + "type": "integer" + }, + "finishedAt": { + "description": "FinishedAt is the time at which the final attempt to execute the step\ncompleted.", + "format": "date-time", + "type": "string" + }, + "message": { + "description": "Message is a display message about the step, including any errors.", + "type": "string" + }, + "startedAt": { + "description": "StartedAt is the time at which the first attempt to execute the step\nbegan.", + "format": "date-time", + "type": "string" + }, + "status": { + "description": "Status is the high-level outcome of the step.", + "type": "string" + } + }, + "type": "object" + }, + "type": "array" } }, "type": "object" diff --git a/ui/src/gen/v1alpha1/generated_pb.ts b/ui/src/gen/v1alpha1/generated_pb.ts index 6be8c0619..343747c23 100644 --- a/ui/src/gen/v1alpha1/generated_pb.ts +++ b/ui/src/gen/v1alpha1/generated_pb.ts @@ -2768,12 +2768,12 @@ export class PromotionStatus extends Message<PromotionStatus> { currentStep?: bigint; /** - * CurrentStepAttempt is the number of times the current step has been - * attempted. + * StepExecutionMetadata tracks metadata pertaining to the execution + * of individual promotion steps. * - * @generated from field: optional int64 currentStepAttempt = 11; + * @generated from field: repeated github.com.akuity.kargo.api.v1alpha1.StepExecutionMetadata stepExecutionMetadata = 11; */ - currentStepAttempt?: bigint; + stepExecutionMetadata: StepExecutionMetadata[] = []; /** * State stores the state of the promotion process between reconciliation @@ -2799,7 +2799,7 @@ export class PromotionStatus extends Message<PromotionStatus> { { no: 8, name: "healthChecks", kind: "message", T: HealthCheckStep, repeated: true }, { no: 6, name: "finishedAt", kind: "message", T: Time, opt: true }, { no: 9, name: "currentStep", kind: "scalar", T: 3 /* ScalarType.INT64 */, opt: true }, - { no: 11, name: "currentStepAttempt", kind: "scalar", T: 3 /* ScalarType.INT64 */, opt: true }, + { no: 11, name: "stepExecutionMetadata", kind: "message", T: StepExecutionMetadata, repeated: true }, { no: 10, name: "state", kind: "message", T: JSON, opt: true }, ]); @@ -2897,18 +2897,49 @@ export class PromotionStep extends Message<PromotionStep> { */ export class PromotionStepRetry extends Message<PromotionStepRetry> { /** - * Attempts is the number of times the step can be attempted before the - * PromotionStep is marked as failed. + * Timeout is the soft maximum interval in which a step that returns a Running + * status (which typically indicates it's waiting for something to happen) + * may be retried. * - * If this field is set to 1, the step will not be retried. If this - * field is set to -1, the step will be retried indefinitely. + * The maximum is a soft one because the check for whether the interval has + * elapsed occurs AFTER the step has run. This effectively means a step may + * run ONCE beyond the close of the interval. * - * The default of this field depends on the step being executed. Refer to - * the documentation for the specific step for more information. + * If this field is set to nil, the effective default will be a step-specific + * one. If no step-specific default exists (i.e. is also nil), the effective + * default will be the system-wide default of 0. * - * @generated from field: optional int64 attempts = 1; + * A value of 0 will cause the step to be retried indefinitely unless the + * ErrorThreshold is reached. + * + * @generated from field: optional k8s.io.apimachinery.pkg.apis.meta.v1.Duration timeout = 1; */ - attempts?: bigint; + timeout?: Duration; + + /** + * ErrorThreshold is the number of consecutive times the step must fail (for + * any reason) before retries are abandoned and the entire Promotion is marked + * as failed. + * + * If this field is set to 0, the effective default will be a step-specific + * one. If no step-specific default exists (i.e. is also 0), the effective + * default will be the system-wide default of 1. + * + * A value of 1 will cause the Promotion to be marked as failed after just + * a single failure; i.e. no retries will be attempted. + * + * There is no option to specify an infinite number of retries using a value + * such as -1. + * + * In a future release, Kargo is likely to become capable of distinguishing + * between recoverable and non-recoverable step failures. At that time, it is + * planned that unrecoverable failures will not be subject to this threshold + * and will immediately cause the Promotion to be marked as failed without + * further condition. + * + * @generated from field: optional uint32 errorThreshold = 2; + */ + errorThreshold?: number; constructor(data?: PartialMessage<PromotionStepRetry>) { super(); @@ -2918,7 +2949,8 @@ export class PromotionStepRetry extends Message<PromotionStepRetry> { static readonly runtime: typeof proto2 = proto2; static readonly typeName = "github.com.akuity.kargo.api.v1alpha1.PromotionStepRetry"; static readonly fields: FieldList = proto2.util.newFieldList(() => [ - { no: 1, name: "attempts", kind: "scalar", T: 3 /* ScalarType.INT64 */, opt: true }, + { no: 1, name: "timeout", kind: "message", T: Duration, opt: true }, + { no: 2, name: "errorThreshold", kind: "scalar", T: 13 /* ScalarType.UINT32 */, opt: true }, ]); static fromBinary(bytes: Uint8Array, options?: Partial<BinaryReadOptions>): PromotionStepRetry { @@ -3467,6 +3499,90 @@ export class StageStatus extends Message<StageStatus> { } } +/** + * StepExecutionMetadata tracks metadata pertaining to the execution of + * a promotion step. + * + * @generated from message github.com.akuity.kargo.api.v1alpha1.StepExecutionMetadata + */ +export class StepExecutionMetadata extends Message<StepExecutionMetadata> { + /** + * Alias is the alias of the step. + * + * @generated from field: optional string alias = 1; + */ + alias?: string; + + /** + * StartedAt is the time at which the first attempt to execute the step + * began. + * + * @generated from field: optional k8s.io.apimachinery.pkg.apis.meta.v1.Time startedAt = 2; + */ + startedAt?: Time; + + /** + * FinishedAt is the time at which the final attempt to execute the step + * completed. + * + * @generated from field: optional k8s.io.apimachinery.pkg.apis.meta.v1.Time finishedAt = 3; + */ + finishedAt?: Time; + + /** + * ErrorCount tracks consecutive failed attempts to execute the step. + * + * @generated from field: optional uint32 errorCount = 4; + */ + errorCount?: number; + + /** + * Status is the high-level outcome of the step. + * + * @generated from field: optional string status = 5; + */ + status?: string; + + /** + * Message is a display message about the step, including any errors. + * + * @generated from field: optional string message = 6; + */ + message?: string; + + constructor(data?: PartialMessage<StepExecutionMetadata>) { + super(); + proto2.util.initPartial(data, this); + } + + static readonly runtime: typeof proto2 = proto2; + static readonly typeName = "github.com.akuity.kargo.api.v1alpha1.StepExecutionMetadata"; + static readonly fields: FieldList = proto2.util.newFieldList(() => [ + { no: 1, name: "alias", kind: "scalar", T: 9 /* ScalarType.STRING */, opt: true }, + { no: 2, name: "startedAt", kind: "message", T: Time, opt: true }, + { no: 3, name: "finishedAt", kind: "message", T: Time, opt: true }, + { no: 4, name: "errorCount", kind: "scalar", T: 13 /* ScalarType.UINT32 */, opt: true }, + { no: 5, name: "status", kind: "scalar", T: 9 /* ScalarType.STRING */, opt: true }, + { no: 6, name: "message", kind: "scalar", T: 9 /* ScalarType.STRING */, opt: true }, + ]); + + static fromBinary(bytes: Uint8Array, options?: Partial<BinaryReadOptions>): StepExecutionMetadata { + return new StepExecutionMetadata().fromBinary(bytes, options); + } + + static fromJson(jsonValue: JsonValue, options?: Partial<JsonReadOptions>): StepExecutionMetadata { + return new StepExecutionMetadata().fromJson(jsonValue, options); + } + + static fromJsonString(jsonString: string, options?: Partial<JsonReadOptions>): StepExecutionMetadata { + return new StepExecutionMetadata().fromJsonString(jsonString, options); + } + + static equals(a: StepExecutionMetadata | PlainMessage<StepExecutionMetadata> | undefined, b: StepExecutionMetadata | PlainMessage<StepExecutionMetadata> | undefined): boolean { + return proto2.util.equals(StepExecutionMetadata, a, b); + } +} + /** * Verification describes how to verify that a Promotion has been successful * using Argo Rollouts AnalysisTemplates.