From 0f09fc5c4cb1c3ee6ee61310aea8ba4054ff3cf1 Mon Sep 17 00:00:00 2001 From: Albin Severinson Date: Thu, 7 Mar 2024 08:39:40 +0000 Subject: [PATCH] Home-away scheduling fixes (#3447) * Correctly pass through priority to schedule pods at in nodeDb * Pass through tolerations after eviction * Restore temporary workaround * mage proto * mage proto * Remove unused proto import * mage proto --- internal/scheduler/context/context.go | 3 +- internal/scheduler/gang_scheduler_test.go | 183 ++++--- internal/scheduler/nodedb/nodedb.go | 45 +- .../scheduler/preempting_queue_scheduler.go | 13 +- .../preempting_queue_scheduler_test.go | 10 + internal/scheduler/queue_scheduler_test.go | 10 + .../schedulerobjects/schedulerobjects.pb.go | 5 +- .../schedulerobjects/schedulerobjects.proto | 5 +- pkg/armadaevents/events.pb.go | 452 +++++++++--------- pkg/armadaevents/events.proto | 2 +- 10 files changed, 407 insertions(+), 321 deletions(-) diff --git a/internal/scheduler/context/context.go b/internal/scheduler/context/context.go index 4822b59b041..0941ccf3a02 100644 --- a/internal/scheduler/context/context.go +++ b/internal/scheduler/context/context.go @@ -799,7 +799,8 @@ type PodSchedulingContext struct { NodeId string // If set, indicates that the pod was scheduled on a specific node type. WellKnownNodeTypeName string - // Priority at which this pod was scheduled. + // Priority this pod was most recently attempted to be scheduled at. + // If scheduling was successful, resources were marked as allocated to the job at this priority. ScheduledAtPriority int32 // Maximum priority that this pod preempted other pods at. PreemptedAtPriority int32 diff --git a/internal/scheduler/gang_scheduler_test.go b/internal/scheduler/gang_scheduler_test.go index 02a53f80068..7e13e675f10 100644 --- a/internal/scheduler/gang_scheduler_test.go +++ b/internal/scheduler/gang_scheduler_test.go @@ -41,7 +41,7 @@ func TestGangScheduler(t *testing.T) { ExpectedScheduledIndices []int // Cumulative number of jobs we expect to schedule successfully. // Each index `i` is the expected value when processing gang `i`. - ExpectedScheduledJobs []int + ExpectedCumulativeScheduledJobs []int // If present, assert that gang `i` is scheduled on nodes with node // uniformity label `ExpectedNodeUniformity[i]`. ExpectedNodeUniformity map[int]string @@ -54,9 +54,9 @@ func TestGangScheduler(t *testing.T) { Gangs: [][]*jobdb.Job{ testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 32)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{32}, - ExpectedRuntimeGangCardinality: []int{32}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{32}, + ExpectedRuntimeGangCardinality: []int{32}, }, "simple failure": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -64,9 +64,9 @@ func TestGangScheduler(t *testing.T) { Gangs: [][]*jobdb.Job{ testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 33)), }, - ExpectedScheduledIndices: nil, - ExpectedScheduledJobs: []int{0}, - ExpectedRuntimeGangCardinality: []int{0}, + ExpectedScheduledIndices: nil, + ExpectedCumulativeScheduledJobs: []int{0}, + ExpectedRuntimeGangCardinality: []int{0}, }, "simple success where min cardinality is met": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -77,9 +77,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 40), ), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{32}, - ExpectedRuntimeGangCardinality: []int{32}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{32}, + ExpectedRuntimeGangCardinality: []int{32}, }, "simple failure where min cardinality is not met": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -90,9 +90,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 40), ), }, - ExpectedScheduledIndices: nil, - ExpectedScheduledJobs: []int{0}, - ExpectedRuntimeGangCardinality: []int{0}, + ExpectedScheduledIndices: nil, + ExpectedCumulativeScheduledJobs: []int{0}, + ExpectedRuntimeGangCardinality: []int{0}, }, "one success and one failure": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -101,9 +101,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 32)), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{32, 32}, - ExpectedRuntimeGangCardinality: []int{32, 0}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{32, 32}, + ExpectedRuntimeGangCardinality: []int{32, 0}, }, "one success and one failure using min cardinality": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -115,9 +115,9 @@ func TestGangScheduler(t *testing.T) { ), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{32, 32}, - ExpectedRuntimeGangCardinality: []int{32, 0}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{32, 32}, + ExpectedRuntimeGangCardinality: []int{32, 0}, }, "multiple nodes": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -125,9 +125,9 @@ func TestGangScheduler(t *testing.T) { Gangs: [][]*jobdb.Job{ testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 64)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{64}, - ExpectedRuntimeGangCardinality: []int{64}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{64}, + ExpectedRuntimeGangCardinality: []int{64}, }, "MaximumResourceFractionToSchedule": { SchedulingConfig: testfixtures.WithRoundLimitsConfig( @@ -140,9 +140,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 16)), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 8)), }, - ExpectedScheduledIndices: []int{0, 1}, - ExpectedScheduledJobs: []int{8, 24, 24}, - ExpectedRuntimeGangCardinality: []int{8, 16, 0}, + ExpectedScheduledIndices: []int{0, 1}, + ExpectedCumulativeScheduledJobs: []int{8, 24, 24}, + ExpectedRuntimeGangCardinality: []int{8, 16, 0}, }, "MaximumResourceFractionToScheduleByPool": { SchedulingConfig: testfixtures.WithRoundLimitsConfig( @@ -160,9 +160,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: []int{0, 1, 2}, - ExpectedScheduledJobs: []int{1, 2, 3, 3, 3}, - ExpectedRuntimeGangCardinality: []int{1, 1, 1, 0, 0}, + ExpectedScheduledIndices: []int{0, 1, 2}, + ExpectedCumulativeScheduledJobs: []int{1, 2, 3, 3, 3}, + ExpectedRuntimeGangCardinality: []int{1, 1, 1, 0, 0}, }, "MaximumResourceFractionToScheduleByPool non-existing pool": { SchedulingConfig: testfixtures.WithRoundLimitsConfig( @@ -180,9 +180,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: []int{0, 1, 2, 3}, - ExpectedScheduledJobs: []int{1, 2, 3, 4, 4}, - ExpectedRuntimeGangCardinality: []int{1, 1, 1, 1, 0}, + ExpectedScheduledIndices: []int{0, 1, 2, 3}, + ExpectedCumulativeScheduledJobs: []int{1, 2, 3, 4, 4}, + ExpectedRuntimeGangCardinality: []int{1, 1, 1, 1, 0}, }, "MaximumResourceFractionPerQueue": { SchedulingConfig: testfixtures.WithPerPriorityLimitsConfig( @@ -205,9 +205,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass3, 4)), testfixtures.WithGangAnnotationsJobs(testfixtures.N1Cpu4GiJobs("A", testfixtures.PriorityClass3, 5)), }, - ExpectedScheduledIndices: []int{0, 2, 4, 6}, - ExpectedScheduledJobs: []int{1, 1, 3, 3, 6, 6, 10, 10}, - ExpectedRuntimeGangCardinality: []int{1, 0, 2, 0, 3, 0, 4, 0}, + ExpectedScheduledIndices: []int{0, 2, 4, 6}, + ExpectedCumulativeScheduledJobs: []int{1, 1, 3, 3, 6, 6, 10, 10}, + ExpectedRuntimeGangCardinality: []int{1, 0, 2, 0, 3, 0, 4, 0}, }, "resolution has no impact on jobs of size a multiple of the resolution": { SchedulingConfig: testfixtures.WithIndexedResourcesConfig( @@ -226,9 +226,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1)), testfixtures.WithGangAnnotationsJobs(testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 5), - ExpectedScheduledJobs: testfixtures.IntRange(1, 6), - ExpectedRuntimeGangCardinality: []int{1, 1, 1, 1, 1, 1}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 5), + ExpectedCumulativeScheduledJobs: testfixtures.IntRange(1, 6), + ExpectedRuntimeGangCardinality: []int{1, 1, 1, 1, 1, 1}, }, "jobs of size not a multiple of the resolution blocks scheduling new jobs": { SchedulingConfig: testfixtures.WithIndexedResourcesConfig( @@ -245,9 +245,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithGangAnnotationsJobs(testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1)), testfixtures.WithGangAnnotationsJobs(testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 2), - ExpectedScheduledJobs: []int{1, 2, 3, 3}, - ExpectedRuntimeGangCardinality: []int{1, 1, 1, 0}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 2), + ExpectedCumulativeScheduledJobs: []int{1, 2, 3, 3}, + ExpectedRuntimeGangCardinality: []int{1, 1, 1, 0}, }, "consider all nodes in the bucket": { SchedulingConfig: testfixtures.WithIndexedResourcesConfig( @@ -283,9 +283,9 @@ func TestGangScheduler(t *testing.T) { Gangs: [][]*jobdb.Job{ testfixtures.WithGangAnnotationsJobs(testfixtures.N1GpuJobs("A", testfixtures.PriorityClass0, 1)), }, - ExpectedScheduledIndices: testfixtures.IntRange(0, 0), - ExpectedScheduledJobs: []int{1}, - ExpectedRuntimeGangCardinality: []int{1}, + ExpectedScheduledIndices: testfixtures.IntRange(0, 0), + ExpectedCumulativeScheduledJobs: []int{1}, + ExpectedRuntimeGangCardinality: []int{1}, }, "NodeUniformityLabel set but not indexed": { SchedulingConfig: testfixtures.TestSchedulingConfig(), @@ -300,9 +300,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1), )), }, - ExpectedScheduledIndices: nil, - ExpectedScheduledJobs: []int{0}, - ExpectedRuntimeGangCardinality: []int{0}, + ExpectedScheduledIndices: nil, + ExpectedCumulativeScheduledJobs: []int{0}, + ExpectedRuntimeGangCardinality: []int{0}, }, "NodeUniformityLabel not set": { SchedulingConfig: testfixtures.WithIndexedNodeLabelsConfig( @@ -317,9 +317,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 1), )), }, - ExpectedScheduledIndices: nil, - ExpectedScheduledJobs: []int{0}, - ExpectedRuntimeGangCardinality: []int{0}, + ExpectedScheduledIndices: nil, + ExpectedCumulativeScheduledJobs: []int{0}, + ExpectedRuntimeGangCardinality: []int{0}, }, "NodeUniformityLabel insufficient capacity": { SchedulingConfig: testfixtures.WithIndexedNodeLabelsConfig( @@ -341,9 +341,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.WithNodeUniformityLabelAnnotationJobs("foo", testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 3)), ), }, - ExpectedScheduledIndices: nil, - ExpectedScheduledJobs: []int{0}, - ExpectedRuntimeGangCardinality: []int{0}, + ExpectedScheduledIndices: nil, + ExpectedCumulativeScheduledJobs: []int{0}, + ExpectedRuntimeGangCardinality: []int{0}, }, "NodeUniformityLabel": { SchedulingConfig: testfixtures.WithIndexedNodeLabelsConfig( @@ -379,9 +379,9 @@ func TestGangScheduler(t *testing.T) { testfixtures.N16Cpu128GiJobs("A", testfixtures.PriorityClass0, 4), )), }, - ExpectedScheduledIndices: []int{0}, - ExpectedScheduledJobs: []int{4}, - ExpectedRuntimeGangCardinality: []int{4}, + ExpectedScheduledIndices: []int{0}, + ExpectedCumulativeScheduledJobs: []int{4}, + ExpectedRuntimeGangCardinality: []int{4}, }, "NodeUniformityLabel NumScheduled tiebreak": { SchedulingConfig: testfixtures.WithIndexedNodeLabelsConfig( @@ -407,10 +407,10 @@ func TestGangScheduler(t *testing.T) { ), ), }, - ExpectedScheduledIndices: []int{0}, - ExpectedScheduledJobs: []int{3}, - ExpectedNodeUniformity: map[int]string{0: "b"}, - ExpectedRuntimeGangCardinality: []int{3}, + ExpectedScheduledIndices: []int{0}, + ExpectedCumulativeScheduledJobs: []int{3}, + ExpectedNodeUniformity: map[int]string{0: "b"}, + ExpectedRuntimeGangCardinality: []int{3}, }, "NodeUniformityLabel PreemptedAtPriority tiebreak": { SchedulingConfig: testfixtures.WithIndexedNodeLabelsConfig( @@ -444,10 +444,10 @@ func TestGangScheduler(t *testing.T) { ), ), }, - ExpectedScheduledIndices: []int{0}, - ExpectedScheduledJobs: []int{2}, - ExpectedNodeUniformity: map[int]string{0: "b"}, - ExpectedRuntimeGangCardinality: []int{2}, + ExpectedScheduledIndices: []int{0}, + ExpectedCumulativeScheduledJobs: []int{2}, + ExpectedNodeUniformity: map[int]string{0: "b"}, + ExpectedRuntimeGangCardinality: []int{2}, }, "AwayNodeTypes": { SchedulingConfig: func() configuration.SchedulingConfig { @@ -513,9 +513,56 @@ func TestGangScheduler(t *testing.T) { gangs = append(gangs, []*jobdb.Job{testfixtures.TestJob("A", jobId, "armada-preemptible-away-both", testfixtures.Test1Cpu4GiPodReqs("A", jobId, 30000))}) return }(), - ExpectedScheduledIndices: []int{1}, - ExpectedScheduledJobs: []int{0, 1}, - ExpectedRuntimeGangCardinality: []int{0, 1}, + ExpectedScheduledIndices: []int{1}, + ExpectedCumulativeScheduledJobs: []int{0, 1}, + ExpectedRuntimeGangCardinality: []int{0, 1}, + }, + "Home-away scheduling": { + SchedulingConfig: func() configuration.SchedulingConfig { + config := testfixtures.TestSchedulingConfig() + config.Preemption.PriorityClasses = map[string]types.PriorityClass{ + "armada-preemptible": { + Priority: 30000, + Preemptible: true, + }, + "armada-preemptible-away": { + Priority: 30000, + Preemptible: true, + AwayNodeTypes: []types.AwayNodeType{ + {Priority: 29000, WellKnownNodeTypeName: "node-type-a"}, + }, + }, + } + config.Preemption.DefaultPriorityClass = "armada-preemptible" + config.WellKnownNodeTypes = []configuration.WellKnownNodeType{ + { + Name: "node-type-a", + Taints: []v1.Taint{ + {Key: "taint-a", Value: "true", Effect: v1.TaintEffectNoSchedule}, + }, + }, + } + return config + }(), + Nodes: func() []*schedulerobjects.Node { + nodes := testfixtures.N32CpuNodes(1, []int32{29000, 30000}) + for _, node := range nodes { + node.Taints = []v1.Taint{ + {Key: "taint-a", Value: "true", Effect: v1.TaintEffectNoSchedule}, + } + } + return nodes + }(), + Gangs: func() (gangs [][]*jobdb.Job) { + jobId := util.ULID() + gangs = append(gangs, []*jobdb.Job{testfixtures.TestJob("A", jobId, "armada-preemptible-away", testfixtures.Test32Cpu256GiPodReqs("A", jobId, 30000))}) + jobId = util.ULID() + gangs = append(gangs, []*jobdb.Job{testfixtures.TestJob("A", jobId, "armada-preemptible-away", testfixtures.Test32Cpu256GiPodReqs("A", jobId, 30000))}) + return + }(), + ExpectedScheduledIndices: []int{0}, + ExpectedCumulativeScheduledJobs: []int{1, 1}, + ExpectedRuntimeGangCardinality: []int{1, 0}, }, } for name, tc := range tests { @@ -641,7 +688,7 @@ func TestGangScheduler(t *testing.T) { // Verify accounting scheduledGangs++ require.Equal(t, scheduledGangs, sch.schedulingContext.NumScheduledGangs) - require.Equal(t, tc.ExpectedScheduledJobs[i], sch.schedulingContext.NumScheduledJobs) + require.Equal(t, tc.ExpectedCumulativeScheduledJobs[i], sch.schedulingContext.NumScheduledJobs) require.Equal(t, 0, sch.schedulingContext.NumEvictedJobs) } else { require.NotEmpty(t, reason) @@ -655,7 +702,7 @@ func TestGangScheduler(t *testing.T) { // Verify accounting require.Equal(t, scheduledGangs, sch.schedulingContext.NumScheduledGangs) - require.Equal(t, tc.ExpectedScheduledJobs[i], sch.schedulingContext.NumScheduledJobs) + require.Equal(t, tc.ExpectedCumulativeScheduledJobs[i], sch.schedulingContext.NumScheduledJobs) require.Equal(t, 0, sch.schedulingContext.NumEvictedJobs) } } diff --git a/internal/scheduler/nodedb/nodedb.go b/internal/scheduler/nodedb/nodedb.go index 4d737331909..49eb1ec4903 100644 --- a/internal/scheduler/nodedb/nodedb.go +++ b/internal/scheduler/nodedb/nodedb.go @@ -574,12 +574,17 @@ func deleteEvictedJobSchedulingContextIfExistsWithTxn(txn *memdb.Txn, jobId stri // SelectNodeForJobWithTxn selects a node on which the job can be scheduled. func (nodeDb *NodeDb) SelectNodeForJobWithTxn(txn *memdb.Txn, jctx *schedulercontext.JobSchedulingContext) (*Node, error) { req := jctx.PodRequirements - priorityClass := interfaces.PriorityClassFromLegacySchedulerJob(nodeDb.priorityClasses, nodeDb.defaultPriorityClass, jctx.Job) + // If the job has already been scheduled, get the priority at which it was scheduled. + // Otherwise, get the original priority the job was submitted with. + priority, ok := nodeDb.GetScheduledAtPriority(jctx.JobId) + if !ok { + priority = req.Priority + } pctx := &schedulercontext.PodSchedulingContext{ Created: time.Now(), - ScheduledAtPriority: -1, + ScheduledAtPriority: priority, PreemptedAtPriority: MinPriority, NumNodes: nodeDb.numNodes, NumExcludedNodesByReason: make(map[string]int), @@ -606,14 +611,6 @@ func (nodeDb *NodeDb) SelectNodeForJobWithTxn(txn *memdb.Txn, jctx *schedulercon if it, err := txn.Get("nodes", "id", nodeId); err != nil { return nil, errors.WithStack(err) } else { - priority, ok := nodeDb.GetScheduledAtPriority(jctx.JobId) - if !ok { - // We only get here if the node ID label was set by the user - // (instead of the scheduler); home-away preemption is ignored - // in that case. - priority = req.Priority - } - pctx.ScheduledAtPriority = priority if node, err := nodeDb.selectNodeForPodWithItAtPriority(it, jctx, priority, true); err != nil { return nil, err } else { @@ -622,7 +619,7 @@ func (nodeDb *NodeDb) SelectNodeForJobWithTxn(txn *memdb.Txn, jctx *schedulercon } } - node, err := nodeDb.selectNodeForJobWithTxnAtPriority(txn, jctx, req.Priority) + node, err := nodeDb.selectNodeForJobWithTxnAtPriority(txn, jctx) if err != nil { return nil, err } @@ -672,25 +669,22 @@ func (nodeDb *NodeDb) selectNodeForJobWithTxnAndAwayNodeType( jctx.AdditionalTolerations = append(jctx.AdditionalTolerations, v1.Toleration{Key: taint.Key, Value: taint.Value, Effect: taint.Effect}) } - node, err = nodeDb.selectNodeForJobWithTxnAtPriority(txn, jctx, awayNodeType.Priority) + jctx.PodSchedulingContext.ScheduledAtPriority = awayNodeType.Priority + node, err = nodeDb.selectNodeForJobWithTxnAtPriority(txn, jctx) return } func (nodeDb *NodeDb) selectNodeForJobWithTxnAtPriority( txn *memdb.Txn, jctx *schedulercontext.JobSchedulingContext, - priority int32, ) (*Node, error) { - req := jctx.PodRequirements + pctx := jctx.PodSchedulingContext matchingNodeTypeIds, numExcludedNodesByReason, err := nodeDb.NodeTypesMatchingJob(jctx) if err != nil { return nil, err } - pctx := jctx.PodSchedulingContext - pctx.ScheduledAtPriority = priority - // Try scheduling at evictedPriority. If this succeeds, no preemption is necessary. pctx.NumExcludedNodesByReason = maps.Clone(numExcludedNodesByReason) if node, err := nodeDb.selectNodeForPodAtPriority(txn, jctx, matchingNodeTypeIds, evictedPriority); err != nil { @@ -704,7 +698,7 @@ func (nodeDb *NodeDb) selectNodeForJobWithTxnAtPriority( // Try scheduling at the job priority. If this fails, scheduling is impossible and we return. // This is an optimisation to avoid looking for preemption targets for unschedulable jobs. pctx.NumExcludedNodesByReason = maps.Clone(numExcludedNodesByReason) - if node, err := nodeDb.selectNodeForPodAtPriority(txn, jctx, matchingNodeTypeIds, req.Priority); err != nil { + if node, err := nodeDb.selectNodeForPodAtPriority(txn, jctx, matchingNodeTypeIds, pctx.ScheduledAtPriority); err != nil { return nil, err } else if err := assertPodSchedulingContextNode(pctx, node); err != nil { return nil, err @@ -759,7 +753,6 @@ func (nodeDb *NodeDb) selectNodeForJobWithUrgencyPreemption( jctx *schedulercontext.JobSchedulingContext, matchingNodeTypeIds []uint64, ) (*Node, error) { - req := jctx.PodRequirements pctx := jctx.PodSchedulingContext numExcludedNodesByReason := pctx.NumExcludedNodesByReason for _, priority := range nodeDb.nodeDbPriorities { @@ -768,7 +761,9 @@ func (nodeDb *NodeDb) selectNodeForJobWithUrgencyPreemption( continue } - if priority > req.Priority { + // Using pctx.ScheduledAtPriority instead of jctx.PodRequirements.Priority, + // since the pctx.ScheduledAtPriority may differ, e.g., in case of home-away scheduling. + if priority > pctx.ScheduledAtPriority { break } @@ -923,13 +918,21 @@ func (nodeDb *NodeDb) selectNodeForJobWithFairPreemption(txn *memdb.Txn, jctx *s } nodesById[nodeId] = node evictedJobSchedulingContextsByNodeId[nodeId] = append(evictedJobSchedulingContextsByNodeId[nodeId], evictedJobSchedulingContext) - if priority := evictedJctx.PodRequirements.Priority; priority > maxPriority { + + priority, ok := nodeDb.GetScheduledAtPriority(evictedJctx.JobId) + if !ok { + priority = evictedJctx.PodRequirements.Priority + } + if priority > maxPriority { maxPriority = priority } matches, _, reason, err := JobRequirementsMet( node.Taints, node.Labels, node.TotalResources, + // At this point, we've unbound the jobs running on the node. + // Hence, we should check if the job is schedulable at evictedPriority, + // since that indicates the job can be scheduled without causing further preemptions. node.AllocatableByPriority[evictedPriority], jctx, ) diff --git a/internal/scheduler/preempting_queue_scheduler.go b/internal/scheduler/preempting_queue_scheduler.go index 2f1c0dadbaf..ee83023cbb0 100644 --- a/internal/scheduler/preempting_queue_scheduler.go +++ b/internal/scheduler/preempting_queue_scheduler.go @@ -9,6 +9,7 @@ import ( "github.com/pkg/errors" "golang.org/x/exp/maps" "golang.org/x/exp/slices" + v1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/resource" "github.com/armadaproject/armada/internal/common/armadacontext" @@ -852,12 +853,20 @@ func (evi *Evictor) Evict(ctx *armadacontext.Context, nodeDbTxn *memdb.Txn) (*Ev } for _, job := range evictedJobs { - // Create a scheduling context for when re-scheduling this job. - // Mark as evicted and add a node selector to ensure the job is re-scheduled onto the node it was evicted from. + // Create a scheduling context for the attempt to re-schedule the job, and: + // 1. Mark the job as evicted. This ensures total scheduled resources is calculated correctly. + // 2. Add a node selector ensuring the job can only be re-scheduled onto the node it was evicted from. + // 3. Add tolerations for all taints on the node. This to ensure that: + // - Adding taints to a node doesn't cause jobs already running on the node to be preempted. + // - Jobs scheduled as away jobs have the necessary tolerations to be re-scheduled. + // TODO(albin): We can remove the checkOnlyDynamicRequirements flag in the nodeDb now that we've added the tolerations. jctx := schedulercontext.JobSchedulingContextFromJob(evi.priorityClasses, job) jctx.IsEvicted = true jctx.AddNodeSelector(schedulerconfig.NodeIdLabel, node.Id) evictedJctxsByJobId[job.GetId()] = jctx + for _, taint := range node.Taints { + jctx.AdditionalTolerations = append(jctx.AdditionalTolerations, v1.Toleration{Key: taint.Key, Value: taint.Value, Effect: taint.Effect}) + } nodeIdByJobId[job.GetId()] = node.Id } diff --git a/internal/scheduler/preempting_queue_scheduler_test.go b/internal/scheduler/preempting_queue_scheduler_test.go index fb0aea8e4d4..915fa6268ca 100644 --- a/internal/scheduler/preempting_queue_scheduler_test.go +++ b/internal/scheduler/preempting_queue_scheduler_test.go @@ -1938,6 +1938,16 @@ func TestPreemptingQueueScheduler(t *testing.T) { assert.True(t, ok) assert.NotEmpty(t, nodeId) + node, err := nodeDb.GetNode(nodeId) + require.NoError(t, err) + assert.NotEmpty(t, node) + + // Check that the job can actually go onto this node. + matches, reason, err := nodedb.StaticJobRequirementsMet(node.Taints, node.Labels, node.TotalResources, jctx) + require.NoError(t, err) + assert.Empty(t, reason) + assert.True(t, matches) + // Check that scheduled jobs are consistently assigned to the same node. // (We don't allow moving jobs between nodes.) if expectedNodeId, ok := nodeIdByJobId[job.GetId()]; ok { diff --git a/internal/scheduler/queue_scheduler_test.go b/internal/scheduler/queue_scheduler_test.go index 4683b80ccb2..412ed6dc176 100644 --- a/internal/scheduler/queue_scheduler_test.go +++ b/internal/scheduler/queue_scheduler_test.go @@ -690,6 +690,16 @@ func TestQueueScheduler(t *testing.T) { nodeId, ok := result.NodeIdByJobId[jctx.JobId] assert.True(t, ok) assert.NotEmpty(t, nodeId) + + node, err := nodeDb.GetNode(nodeId) + require.NoError(t, err) + assert.NotEmpty(t, node) + + // Check that the job can actually go onto this node. + matches, reason, err := nodedb.StaticJobRequirementsMet(node.Taints, node.Labels, node.TotalResources, jctx) + require.NoError(t, err) + assert.Empty(t, reason) + assert.True(t, matches) } // For jobs that could not be scheduled, diff --git a/internal/scheduler/schedulerobjects/schedulerobjects.pb.go b/internal/scheduler/schedulerobjects/schedulerobjects.pb.go index 8319120e3de..bbbdfe705a8 100644 --- a/internal/scheduler/schedulerobjects/schedulerobjects.pb.go +++ b/internal/scheduler/schedulerobjects/schedulerobjects.pb.go @@ -835,7 +835,10 @@ type PodRequirements struct { Tolerations []v1.Toleration `protobuf:"bytes,3,rep,name=tolerations,proto3" json:"tolerations"` // Kubernetes annotations. Included here since we use annotations with special meaning. Annotations map[string]string `protobuf:"bytes,7,rep,name=annotations,proto3" json:"annotations,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` - // Pod priority. Should be mapped from the priority class name of the submitted pod. + // Priority class priority of the pod as submitted. Should be mapped from the priority class name of the submitted pod. + // + // During scheduling, the priority stored on the podSchedulingContext should be used instead, + // since a pod may be scheduled at a priority different from the priority it was submitted with. Priority int32 `protobuf:"varint,4,opt,name=priority,proto3" json:"priority,omitempty"` // One of Never, PreemptLowerPriority. // Defaults to PreemptLowerPriority if unset. diff --git a/internal/scheduler/schedulerobjects/schedulerobjects.proto b/internal/scheduler/schedulerobjects/schedulerobjects.proto index db8fbf64da2..ebfb8c2149d 100644 --- a/internal/scheduler/schedulerobjects/schedulerobjects.proto +++ b/internal/scheduler/schedulerobjects/schedulerobjects.proto @@ -159,7 +159,10 @@ message PodRequirements { repeated k8s.io.api.core.v1.Toleration tolerations = 3 [(gogoproto.nullable) = false]; // Kubernetes annotations. Included here since we use annotations with special meaning. map annotations = 7; - // Pod priority. Should be mapped from the priority class name of the submitted pod. + // Priority class priority of the pod as submitted. Should be mapped from the priority class name of the submitted pod. + // + // During scheduling, the priority stored on the podSchedulingContext should be used instead, + // since a pod may be scheduled at a priority different from the priority it was submitted with. int32 priority = 4; // One of Never, PreemptLowerPriority. // Defaults to PreemptLowerPriority if unset. diff --git a/pkg/armadaevents/events.pb.go b/pkg/armadaevents/events.pb.go index a38940932b6..6185d70f957 100644 --- a/pkg/armadaevents/events.pb.go +++ b/pkg/armadaevents/events.pb.go @@ -3526,233 +3526,233 @@ func init() { func init() { proto.RegisterFile("pkg/armadaevents/events.proto", fileDescriptor_6aab92ca59e015f8) } var fileDescriptor_6aab92ca59e015f8 = []byte{ - // 3615 bytes of a gzipped FileDescriptorProto + // 3604 bytes of a gzipped FileDescriptorProto 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x5b, 0x4b, 0x6c, 0x1b, 0xd7, - 0x7a, 0xf6, 0x90, 0x12, 0x1f, 0x3f, 0xf5, 0xa0, 0x8f, 0x25, 0x85, 0x56, 0x6c, 0x51, 0x1e, 0xa7, - 0x8d, 0x13, 0x24, 0x54, 0xe2, 0x3c, 0x90, 0x47, 0x91, 0x40, 0xb4, 0x15, 0xdb, 0x8a, 0x65, 0x2b, - 0x94, 0x95, 0xba, 0x41, 0x0a, 0x66, 0xc8, 0x39, 0xa2, 0xc6, 0x1a, 0xce, 0x4c, 0xe6, 0x21, 0x4b, - 0x40, 0x16, 0x4d, 0xd1, 0xa6, 0xbb, 0xd4, 0x40, 0xb3, 0x28, 0xd0, 0x45, 0xba, 0x6d, 0x80, 0xae, - 0xbb, 0xee, 0xaa, 0x59, 0x14, 0x45, 0xba, 0xeb, 0x8a, 0xbd, 0x48, 0x70, 0x17, 0x97, 0x8b, 0xbb, - 0xbe, 0xf7, 0x6e, 0xee, 0xc5, 0x79, 0xcd, 0x9c, 0x33, 0x1c, 0xda, 0xf2, 0xeb, 0x3a, 0x17, 0x5e, - 0x49, 0xf3, 0xfd, 0xcf, 0xf3, 0xfa, 0xe7, 0xff, 0xff, 0x39, 0x84, 0xd3, 0xde, 0x5e, 0x6f, 0xc5, - 0xf0, 0xfb, 0x86, 0x69, 0xe0, 0x7d, 0xec, 0x84, 0xc1, 0x0a, 0xfb, 0xd3, 0xf0, 0x7c, 0x37, 0x74, - 0xd1, 0x94, 0x4c, 0x5a, 0xd4, 0xf7, 0xde, 0x0a, 0x1a, 0x96, 0xbb, 0x62, 0x78, 0xd6, 0x4a, 0xd7, - 0xf5, 0xf1, 0xca, 0xfe, 0xab, 0x2b, 0x3d, 0xec, 0x60, 0xdf, 0x08, 0xb1, 0xc9, 0x24, 0x16, 0xcf, - 0x49, 0x3c, 0x0e, 0x0e, 0x6f, 0xbb, 0xfe, 0x9e, 0xe5, 0xf4, 0xb2, 0x38, 0xeb, 0x3d, 0xd7, 0xed, - 0xd9, 0x78, 0x85, 0x3e, 0x75, 0xa2, 0x9d, 0x95, 0xd0, 0xea, 0xe3, 0x20, 0x34, 0xfa, 0x1e, 0x67, - 0x58, 0x4a, 0x33, 0xdc, 0xf6, 0x0d, 0xcf, 0xc3, 0x3e, 0x77, 0x6e, 0xf1, 0xf5, 0xc4, 0x54, 0xdf, - 0xe8, 0xee, 0x5a, 0x0e, 0xf6, 0x0f, 0x57, 0xe8, 0x78, 0x3c, 0x6b, 0xc5, 0xc7, 0x81, 0x1b, 0xf9, - 0x5d, 0x3c, 0x62, 0xf6, 0xe5, 0x9e, 0x15, 0xee, 0x46, 0x9d, 0x46, 0xd7, 0xed, 0xaf, 0xf4, 0xdc, - 0x9e, 0x9b, 0xa8, 0x27, 0x4f, 0xf4, 0x81, 0xfe, 0xc7, 0xd9, 0xdf, 0xb1, 0x9c, 0x10, 0xfb, 0x8e, - 0x61, 0xaf, 0x04, 0xdd, 0x5d, 0x6c, 0x46, 0x36, 0xf6, 0x93, 0xff, 0xdc, 0xce, 0x2d, 0xdc, 0x0d, - 0x83, 0x11, 0x80, 0xc9, 0xea, 0x77, 0xe6, 0x60, 0x7a, 0x8d, 0x4c, 0xdd, 0x16, 0xfe, 0x3c, 0xc2, - 0x4e, 0x17, 0xa3, 0x17, 0x60, 0xf2, 0xf3, 0x08, 0x47, 0xb8, 0xa6, 0x2d, 0x6b, 0xe7, 0xca, 0xcd, - 0x13, 0xc3, 0x41, 0x7d, 0x96, 0x02, 0x2f, 0xb9, 0x7d, 0x2b, 0xc4, 0x7d, 0x2f, 0x3c, 0x6c, 0x31, - 0x0e, 0xf4, 0x0e, 0x4c, 0xdd, 0x72, 0x3b, 0xed, 0x00, 0x87, 0x6d, 0xc7, 0xe8, 0xe3, 0x5a, 0x8e, - 0x4a, 0xd4, 0x86, 0x83, 0xfa, 0xdc, 0x2d, 0xb7, 0xb3, 0x85, 0xc3, 0x6b, 0x46, 0x5f, 0x16, 0x83, - 0x04, 0x45, 0x2f, 0x43, 0x31, 0x0a, 0xb0, 0xdf, 0xb6, 0xcc, 0x5a, 0x9e, 0x8a, 0xcd, 0x0d, 0x07, - 0xf5, 0x2a, 0x81, 0xae, 0x98, 0x92, 0x48, 0x81, 0x21, 0xe8, 0x25, 0x28, 0xf4, 0x7c, 0x37, 0xf2, - 0x82, 0xda, 0xc4, 0x72, 0x5e, 0x70, 0x33, 0x44, 0xe6, 0x66, 0x08, 0xba, 0x0e, 0x05, 0xb6, 0x1f, - 0x6a, 0x93, 0xcb, 0xf9, 0x73, 0x95, 0xf3, 0x67, 0x1a, 0xf2, 0x26, 0x69, 0x28, 0x03, 0x66, 0x4f, - 0x4c, 0x21, 0xa3, 0xcb, 0x0a, 0xf9, 0xb6, 0xfa, 0xd5, 0x71, 0x98, 0xa4, 0x7c, 0xe8, 0x3a, 0x14, - 0xbb, 0x3e, 0x26, 0x8b, 0x55, 0x43, 0xcb, 0xda, 0xb9, 0xca, 0xf9, 0xc5, 0x06, 0xdb, 0x03, 0x0d, - 0xb1, 0x48, 0x8d, 0x1b, 0x62, 0x93, 0x34, 0x4f, 0x0e, 0x07, 0xf5, 0xe3, 0x9c, 0x3d, 0xd1, 0x7a, - 0xe7, 0xff, 0xeb, 0x5a, 0x4b, 0x68, 0x41, 0x9b, 0x50, 0x0e, 0xa2, 0x4e, 0xdf, 0x0a, 0xd7, 0xdd, - 0x0e, 0x9d, 0xf3, 0xca, 0xf9, 0x67, 0x54, 0x77, 0xb7, 0x04, 0xb9, 0xf9, 0xcc, 0x70, 0x50, 0x3f, - 0x11, 0x73, 0x27, 0x1a, 0x2f, 0x1f, 0x6b, 0x25, 0x4a, 0xd0, 0x2e, 0xcc, 0xfa, 0xd8, 0xf3, 0x2d, - 0xd7, 0xb7, 0x42, 0x2b, 0xc0, 0x44, 0x6f, 0x8e, 0xea, 0x3d, 0xad, 0xea, 0x6d, 0xa9, 0x4c, 0xcd, - 0xd3, 0xc3, 0x41, 0xfd, 0x64, 0x4a, 0x52, 0xb1, 0x91, 0x56, 0x8b, 0x42, 0x40, 0x29, 0x68, 0x0b, - 0x87, 0x74, 0x3d, 0x2b, 0xe7, 0x97, 0xef, 0x6a, 0x6c, 0x0b, 0x87, 0xcd, 0xe5, 0xe1, 0xa0, 0x7e, - 0x6a, 0x54, 0x5e, 0x31, 0x99, 0xa1, 0x1f, 0xd9, 0x50, 0x95, 0x51, 0x93, 0x0c, 0x70, 0x82, 0xda, - 0x5c, 0x1a, 0x6f, 0x93, 0x70, 0x35, 0x97, 0x86, 0x83, 0xfa, 0x62, 0x5a, 0x56, 0xb1, 0x37, 0xa2, - 0x99, 0xac, 0x4f, 0xd7, 0x70, 0xba, 0xd8, 0x26, 0x66, 0x26, 0xb3, 0xd6, 0xe7, 0x82, 0x20, 0xb3, - 0xf5, 0x89, 0xb9, 0xd5, 0xf5, 0x89, 0x61, 0xf4, 0x29, 0x4c, 0xc5, 0x0f, 0x64, 0xbe, 0x0a, 0x7c, - 0x1f, 0x65, 0x2b, 0x25, 0x33, 0xb5, 0x38, 0x1c, 0xd4, 0x17, 0x64, 0x19, 0x45, 0xb5, 0xa2, 0x2d, - 0xd1, 0x6e, 0xb3, 0x99, 0x29, 0x8e, 0xd7, 0xce, 0x38, 0x64, 0xed, 0xf6, 0xe8, 0x8c, 0x28, 0xda, - 0x88, 0x76, 0x72, 0x88, 0xa3, 0x6e, 0x17, 0x63, 0x13, 0x9b, 0xb5, 0x52, 0x96, 0xf6, 0x75, 0x89, - 0x83, 0x69, 0x97, 0x65, 0x54, 0xed, 0x32, 0x85, 0xcc, 0xf5, 0x2d, 0xb7, 0xb3, 0xe6, 0xfb, 0xae, - 0x1f, 0xd4, 0xca, 0x59, 0x73, 0xbd, 0x2e, 0xc8, 0x6c, 0xae, 0x63, 0x6e, 0x75, 0xae, 0x63, 0x98, - 0xfb, 0xdb, 0x8a, 0x9c, 0xab, 0xd8, 0x08, 0xb0, 0x59, 0x83, 0x31, 0xfe, 0xc6, 0x1c, 0xb1, 0xbf, - 0x31, 0x32, 0xe2, 0x6f, 0x4c, 0x41, 0x26, 0xcc, 0xb0, 0xe7, 0xd5, 0x20, 0xb0, 0x7a, 0x0e, 0x36, - 0x6b, 0x15, 0xaa, 0xff, 0x54, 0x96, 0x7e, 0xc1, 0xd3, 0x3c, 0x35, 0x1c, 0xd4, 0x6b, 0xaa, 0x9c, - 0x62, 0x23, 0xa5, 0x13, 0x7d, 0x06, 0xd3, 0x0c, 0x69, 0x45, 0x8e, 0x63, 0x39, 0xbd, 0xda, 0x14, - 0x35, 0xf2, 0x6c, 0x96, 0x11, 0xce, 0xd2, 0x7c, 0x76, 0x38, 0xa8, 0x3f, 0xa3, 0x48, 0x29, 0x26, - 0x54, 0x85, 0x24, 0x62, 0x30, 0x20, 0x59, 0xd8, 0xe9, 0xac, 0x88, 0xb1, 0xae, 0x32, 0xb1, 0x88, - 0x91, 0x92, 0x54, 0x23, 0x46, 0x8a, 0x98, 0xac, 0x07, 0x5f, 0xe4, 0x99, 0xf1, 0xeb, 0xc1, 0xd7, - 0x59, 0x5a, 0x8f, 0x8c, 0xa5, 0x56, 0xb4, 0xa1, 0x2f, 0x80, 0xbc, 0x78, 0x2e, 0x46, 0x9e, 0x6d, - 0x75, 0x8d, 0x10, 0x5f, 0xc4, 0x21, 0xee, 0x92, 0x48, 0x3d, 0x4b, 0xad, 0xe8, 0x23, 0x56, 0x46, - 0x38, 0x9b, 0xfa, 0x70, 0x50, 0x5f, 0xca, 0xd2, 0xa1, 0x58, 0xcd, 0xb4, 0x82, 0xfe, 0x46, 0x83, - 0xf9, 0x20, 0x34, 0x1c, 0xd3, 0xb0, 0x5d, 0x07, 0x5f, 0x71, 0x7a, 0x3e, 0x0e, 0x82, 0x2b, 0xce, - 0x8e, 0x5b, 0xab, 0x52, 0xfb, 0x67, 0x53, 0x61, 0x3d, 0x8b, 0xb5, 0x79, 0x76, 0x38, 0xa8, 0xd7, - 0x33, 0xb5, 0x28, 0x1e, 0x64, 0x1b, 0x42, 0x07, 0x70, 0x42, 0x64, 0x15, 0xdb, 0xa1, 0x65, 0x5b, - 0x81, 0x11, 0x5a, 0xae, 0x53, 0x3b, 0x4e, 0xed, 0x9f, 0x49, 0x47, 0xc7, 0x11, 0xc6, 0xe6, 0x99, - 0xe1, 0xa0, 0x7e, 0x3a, 0x43, 0x83, 0x62, 0x3b, 0xcb, 0x44, 0xb2, 0x85, 0x36, 0x7d, 0x4c, 0x18, - 0xb1, 0x59, 0x3b, 0x31, 0x7e, 0x0b, 0xc5, 0x4c, 0xf2, 0x16, 0x8a, 0xc1, 0xac, 0x2d, 0x14, 0x13, - 0x89, 0x25, 0xcf, 0xf0, 0x43, 0x8b, 0x98, 0xdd, 0x30, 0xfc, 0x3d, 0xec, 0xd7, 0xe6, 0xb2, 0x2c, - 0x6d, 0xaa, 0x4c, 0xcc, 0x52, 0x4a, 0x52, 0xb5, 0x94, 0x22, 0xa2, 0x3b, 0x1a, 0xa8, 0xae, 0x59, - 0xae, 0xd3, 0x22, 0x69, 0x43, 0x40, 0x86, 0x37, 0x4f, 0x8d, 0x3e, 0x7f, 0x97, 0xe1, 0xc9, 0xec, - 0xcd, 0xe7, 0x87, 0x83, 0xfa, 0xd9, 0xb1, 0xda, 0x14, 0x47, 0xc6, 0x1b, 0x45, 0x37, 0xa1, 0x42, - 0x88, 0x98, 0x26, 0x60, 0x66, 0x6d, 0x81, 0xfa, 0x70, 0x72, 0xd4, 0x07, 0xce, 0x40, 0x33, 0x90, - 0x79, 0x49, 0x42, 0xb1, 0x23, 0xab, 0x6a, 0x16, 0x61, 0x92, 0xca, 0xeb, 0xc3, 0x02, 0x9c, 0xc8, - 0xd8, 0x1b, 0xe8, 0x3d, 0x28, 0xf8, 0x91, 0x43, 0x12, 0x36, 0x96, 0xa5, 0x20, 0xd5, 0xea, 0x76, - 0x64, 0x99, 0x2c, 0x5b, 0xf4, 0x23, 0x47, 0xc9, 0xe1, 0x26, 0x29, 0x40, 0xe4, 0x49, 0xb6, 0x68, - 0x99, 0x3c, 0x1b, 0x19, 0x2b, 0x7f, 0xcb, 0xed, 0xa8, 0xf2, 0x14, 0x40, 0x18, 0xa6, 0xc5, 0xc6, - 0x6b, 0x5b, 0xe4, 0x54, 0xb1, 0x3c, 0xe3, 0x39, 0x55, 0xcd, 0x87, 0x51, 0x07, 0xfb, 0x0e, 0x0e, - 0x71, 0x20, 0xc6, 0x40, 0x8f, 0x15, 0x8d, 0x22, 0xbe, 0x84, 0x48, 0xfa, 0xa7, 0x64, 0x1c, 0x7d, - 0xa3, 0x41, 0xad, 0x6f, 0x1c, 0xb4, 0x05, 0x18, 0xb4, 0x77, 0x5c, 0xbf, 0xed, 0x61, 0xdf, 0x72, - 0x4d, 0x9a, 0x7c, 0x56, 0xce, 0xff, 0xc5, 0x3d, 0x0f, 0x52, 0x63, 0xc3, 0x38, 0x10, 0x70, 0xf0, - 0x81, 0xeb, 0x6f, 0x52, 0xf1, 0x35, 0x27, 0xf4, 0x0f, 0x9b, 0xa7, 0xbf, 0x1f, 0xd4, 0x8f, 0x91, - 0x65, 0xe9, 0x67, 0xf1, 0xb4, 0xb2, 0x61, 0xf4, 0x8f, 0x1a, 0x2c, 0x84, 0x6e, 0x68, 0xd8, 0xed, - 0x6e, 0xd4, 0x8f, 0x6c, 0x23, 0xb4, 0xf6, 0x71, 0x3b, 0x0a, 0x8c, 0x1e, 0xe6, 0x39, 0xee, 0xbb, - 0xf7, 0x76, 0xea, 0x06, 0x91, 0xbf, 0x10, 0x8b, 0x6f, 0x13, 0x69, 0xe6, 0xd3, 0x29, 0xee, 0xd3, - 0x5c, 0x98, 0xc1, 0xd2, 0xca, 0x44, 0x17, 0xff, 0x55, 0x83, 0xc5, 0xf1, 0xc3, 0x44, 0x67, 0x21, - 0xbf, 0x87, 0x0f, 0x79, 0x15, 0x71, 0x7c, 0x38, 0xa8, 0x4f, 0xef, 0xe1, 0x43, 0x69, 0xd6, 0x09, - 0x15, 0xfd, 0x15, 0x4c, 0xee, 0x1b, 0x76, 0x84, 0xf9, 0x96, 0x68, 0x34, 0x58, 0xbd, 0xd4, 0x90, - 0xeb, 0xa5, 0x86, 0xb7, 0xd7, 0x23, 0x40, 0x43, 0xac, 0x48, 0xe3, 0xa3, 0xc8, 0x70, 0x42, 0x2b, - 0x3c, 0x64, 0xdb, 0x85, 0x2a, 0x90, 0xb7, 0x0b, 0x05, 0xde, 0xc9, 0xbd, 0xa5, 0x2d, 0x7e, 0xab, - 0xc1, 0xc9, 0xb1, 0x83, 0xfe, 0x39, 0x78, 0xa8, 0xb7, 0x61, 0x82, 0x6c, 0x7c, 0x52, 0xdf, 0xec, - 0x5a, 0xbd, 0xdd, 0x37, 0x5f, 0xa7, 0xee, 0x14, 0x58, 0x39, 0xc2, 0x10, 0xb9, 0x1c, 0x61, 0x08, - 0xa9, 0xd1, 0x6c, 0xf7, 0xf6, 0x9b, 0xaf, 0x53, 0xa7, 0x0a, 0xcc, 0x08, 0x05, 0x64, 0x23, 0x14, - 0xd0, 0x7f, 0x5f, 0x80, 0x72, 0x5c, 0x40, 0x48, 0x67, 0x50, 0x7b, 0xa0, 0x33, 0x78, 0x19, 0xaa, - 0x26, 0x36, 0xf9, 0x9b, 0xcf, 0x72, 0x1d, 0x71, 0x9a, 0xcb, 0x2c, 0xba, 0x2a, 0x34, 0x45, 0x7e, - 0x36, 0x45, 0x42, 0xe7, 0xa1, 0xc4, 0x13, 0xed, 0x43, 0x7a, 0x90, 0xa7, 0x9b, 0x0b, 0xc3, 0x41, - 0x1d, 0x09, 0x4c, 0x12, 0x8d, 0xf9, 0x50, 0x0b, 0x80, 0x55, 0xaf, 0x1b, 0x38, 0x34, 0x78, 0xca, - 0x5f, 0x53, 0x47, 0x70, 0x3d, 0xa6, 0xb3, 0x3a, 0x34, 0xe1, 0x97, 0xeb, 0xd0, 0x04, 0x45, 0x9f, - 0x02, 0xf4, 0x0d, 0xcb, 0x61, 0x72, 0x3c, 0xbf, 0xd7, 0xc7, 0x85, 0x94, 0x8d, 0x98, 0x93, 0x69, - 0x4f, 0x24, 0x65, 0xed, 0x09, 0x4a, 0xaa, 0x45, 0x5e, 0x6f, 0xd7, 0x0a, 0xf4, 0x94, 0x2e, 0x8d, - 0x53, 0xcd, 0xd5, 0xce, 0x93, 0x8a, 0x91, 0x8b, 0x48, 0x3a, 0x85, 0x16, 0x32, 0x6d, 0xb6, 0xb5, - 0x83, 0x43, 0xab, 0x8f, 0x69, 0x66, 0xcf, 0xa7, 0x4d, 0x60, 0xf2, 0xb4, 0x09, 0x0c, 0xbd, 0x05, - 0x60, 0x84, 0x1b, 0x6e, 0x10, 0x5e, 0x77, 0xba, 0x98, 0x66, 0xec, 0x25, 0xe6, 0x7e, 0x82, 0xca, - 0xee, 0x27, 0x28, 0x7a, 0x17, 0x2a, 0x1e, 0x7f, 0x09, 0x75, 0x6c, 0x4c, 0x33, 0xf2, 0x12, 0x7b, - 0xa5, 0x48, 0xb0, 0x24, 0x2b, 0x73, 0xa3, 0x4b, 0x30, 0xdb, 0x75, 0x9d, 0x6e, 0xe4, 0xfb, 0xd8, - 0xe9, 0x1e, 0x6e, 0x19, 0x3b, 0x98, 0x66, 0xdf, 0x25, 0xb6, 0x55, 0x52, 0x24, 0x79, 0xab, 0xa4, - 0x48, 0xe8, 0x0d, 0x28, 0xc7, 0xdd, 0x0b, 0x9a, 0x60, 0x97, 0x79, 0x21, 0x2c, 0x40, 0x49, 0x38, - 0xe1, 0x24, 0xce, 0x5b, 0x41, 0x9c, 0xa5, 0xd1, 0xa4, 0x99, 0x3b, 0x2f, 0xc1, 0xb2, 0xf3, 0x12, - 0x8c, 0xae, 0xc0, 0x71, 0xfa, 0x5e, 0x6c, 0x87, 0xa1, 0xdd, 0x0e, 0x70, 0xd7, 0x75, 0xcc, 0x80, - 0xe6, 0xc4, 0x79, 0xe6, 0x3e, 0x25, 0xde, 0x08, 0xed, 0x2d, 0x46, 0x92, 0xdd, 0x4f, 0x91, 0xf4, - 0xff, 0xd6, 0x60, 0x2e, 0x6b, 0x0b, 0xa5, 0xb6, 0xb3, 0xf6, 0x48, 0xb6, 0xf3, 0xc7, 0x50, 0xf2, - 0x5c, 0xb3, 0x1d, 0x78, 0xb8, 0xcb, 0x23, 0x56, 0x6a, 0x33, 0x6f, 0xba, 0xe6, 0x96, 0x87, 0xbb, - 0x7f, 0x69, 0x85, 0xbb, 0xab, 0xfb, 0xae, 0x65, 0x5e, 0xb5, 0x02, 0xbe, 0xeb, 0x3c, 0x46, 0x51, - 0x32, 0x84, 0x22, 0x07, 0x9b, 0x25, 0x28, 0x30, 0x2b, 0xfa, 0xff, 0xe4, 0xa1, 0x9a, 0xde, 0xb6, - 0x7f, 0x4a, 0x43, 0x41, 0x37, 0xa1, 0x68, 0xb1, 0x94, 0x99, 0x67, 0x10, 0x7f, 0x26, 0xc5, 0xf4, - 0x46, 0xd2, 0x10, 0x6c, 0xec, 0xbf, 0xda, 0xe0, 0xb9, 0x35, 0x9d, 0x02, 0xaa, 0x99, 0x4b, 0xaa, - 0x9a, 0x39, 0x88, 0x5a, 0x50, 0x0c, 0xb0, 0xbf, 0x6f, 0x75, 0x31, 0x0f, 0x4e, 0x75, 0x59, 0x73, - 0xd7, 0xf5, 0x31, 0xd1, 0xb9, 0xc5, 0x58, 0x12, 0x9d, 0x5c, 0x46, 0xd5, 0xc9, 0x41, 0xf4, 0x31, - 0x94, 0xbb, 0xae, 0xb3, 0x63, 0xf5, 0x36, 0x0c, 0x8f, 0x87, 0xa7, 0xd3, 0x59, 0x5a, 0x2f, 0x08, - 0x26, 0xde, 0x84, 0x10, 0x8f, 0xa9, 0x26, 0x44, 0xcc, 0x95, 0x2c, 0xe8, 0xaf, 0x27, 0x00, 0x92, - 0xc5, 0x41, 0x6f, 0x43, 0x05, 0x1f, 0xe0, 0x6e, 0x14, 0xba, 0xbe, 0x78, 0x4f, 0xf0, 0x9e, 0x9e, - 0x80, 0x95, 0xc0, 0x0e, 0x09, 0x4a, 0x0e, 0xaa, 0x63, 0xf4, 0x71, 0xe0, 0x19, 0x5d, 0xd1, 0x0c, - 0xa4, 0xce, 0xc4, 0xa0, 0x7c, 0x50, 0x63, 0x10, 0xfd, 0x39, 0x4c, 0xd0, 0xf6, 0x21, 0xeb, 0x03, - 0xa2, 0xe1, 0xa0, 0x3e, 0xe3, 0xa8, 0x8d, 0x43, 0x4a, 0x47, 0xef, 0xc3, 0xf4, 0x5e, 0xbc, 0xf1, - 0x88, 0x6f, 0x13, 0x54, 0x80, 0xa6, 0x76, 0x09, 0x41, 0xf1, 0x6e, 0x4a, 0xc6, 0xd1, 0x0e, 0x54, - 0x0c, 0xc7, 0x71, 0x43, 0xfa, 0x0e, 0x12, 0xbd, 0xc1, 0x17, 0xc6, 0x6d, 0xd3, 0xc6, 0x6a, 0xc2, - 0xcb, 0xb2, 0x24, 0x1a, 0x3c, 0x24, 0x0d, 0x72, 0xf0, 0x90, 0x60, 0xd4, 0x82, 0x82, 0x6d, 0x74, - 0xb0, 0x2d, 0x82, 0xfe, 0x73, 0x63, 0x4d, 0x5c, 0xa5, 0x6c, 0x4c, 0x3b, 0x7d, 0xe5, 0x33, 0x39, - 0xf9, 0x95, 0xcf, 0x90, 0xc5, 0x1d, 0xa8, 0xa6, 0xfd, 0x39, 0x5a, 0x02, 0xf3, 0x82, 0x9c, 0xc0, - 0x94, 0xef, 0x99, 0x32, 0x19, 0x50, 0x91, 0x9c, 0x7a, 0x1c, 0x26, 0xf4, 0x7f, 0xd3, 0x60, 0x2e, - 0xeb, 0xec, 0xa2, 0x0d, 0xe9, 0xc4, 0x6b, 0xbc, 0xc7, 0x91, 0xb1, 0xd5, 0xb9, 0xec, 0x98, 0xa3, - 0x9e, 0x1c, 0xf4, 0x26, 0xcc, 0x38, 0xae, 0x89, 0xdb, 0x06, 0x31, 0x60, 0x5b, 0x41, 0x58, 0xcb, - 0xd1, 0xde, 0x31, 0xed, 0x8d, 0x10, 0xca, 0xaa, 0x20, 0x48, 0xd2, 0xd3, 0x0a, 0x41, 0xff, 0x7b, - 0x0d, 0x66, 0x53, 0xad, 0xcb, 0x87, 0x4e, 0xa2, 0xe4, 0xd4, 0x27, 0x77, 0xb4, 0xd4, 0x47, 0xff, - 0xa7, 0x1c, 0x54, 0xa4, 0xba, 0xee, 0xa1, 0x7d, 0xb8, 0x05, 0xb3, 0xfc, 0x4d, 0x69, 0x39, 0x3d, - 0x56, 0x4e, 0xe5, 0x78, 0x93, 0x62, 0xe4, 0x4b, 0xc1, 0xba, 0xdb, 0xd9, 0x8a, 0x79, 0x69, 0x35, - 0x45, 0x3b, 0x58, 0x81, 0x82, 0x49, 0x26, 0x66, 0x54, 0x0a, 0xba, 0x09, 0x0b, 0x91, 0x67, 0x1a, - 0x21, 0x6e, 0x07, 0xbc, 0xe7, 0xde, 0x76, 0xa2, 0x7e, 0x07, 0xfb, 0xf4, 0xc4, 0x4f, 0xb2, 0x9e, - 0x0b, 0xe3, 0x10, 0x4d, 0xf9, 0x6b, 0x94, 0x2e, 0xe9, 0x9c, 0xcb, 0xa2, 0xeb, 0x97, 0x01, 0x8d, - 0xf6, 0x95, 0x95, 0xf9, 0xd5, 0x8e, 0x38, 0xbf, 0x5f, 0x69, 0x50, 0x4d, 0xb7, 0x8b, 0x9f, 0xc8, - 0x42, 0x1f, 0x42, 0x39, 0x6e, 0xfd, 0x3e, 0xb4, 0x03, 0x2f, 0x41, 0xc1, 0xc7, 0x46, 0xe0, 0x3a, - 0xfc, 0x64, 0xd2, 0x10, 0xc3, 0x10, 0x39, 0xc4, 0x30, 0x44, 0xbf, 0x01, 0x53, 0x6c, 0x06, 0x3f, - 0xb0, 0xec, 0x10, 0xfb, 0xe8, 0x22, 0x14, 0x82, 0xd0, 0x08, 0x71, 0x50, 0xd3, 0x96, 0xf3, 0xe7, - 0x66, 0xce, 0x2f, 0x8c, 0x76, 0x79, 0x09, 0x99, 0x69, 0x65, 0x9c, 0xb2, 0x56, 0x86, 0xe8, 0x7f, - 0xab, 0xc1, 0x94, 0xdc, 0xcc, 0x7e, 0x34, 0x6a, 0xef, 0x73, 0x68, 0x5f, 0x08, 0x1f, 0xec, 0x47, - 0xb3, 0xb2, 0xf7, 0x67, 0xfd, 0x3f, 0x34, 0x36, 0xb3, 0x71, 0x17, 0xf4, 0x61, 0xcd, 0xf7, 0x92, - 0x56, 0x08, 0x39, 0x61, 0x01, 0x0d, 0x6c, 0x47, 0x6d, 0x85, 0xd0, 0xf0, 0xa7, 0x88, 0xcb, 0xe1, - 0x4f, 0x21, 0xe8, 0xdf, 0x4c, 0x52, 0xcf, 0x93, 0x8e, 0xf7, 0x93, 0x6e, 0x02, 0xa5, 0xb2, 0x93, - 0xfc, 0x7d, 0x64, 0x27, 0x2f, 0x43, 0x91, 0xbe, 0x0e, 0xe2, 0xc4, 0x81, 0x2e, 0x1a, 0x81, 0xd4, - 0x2f, 0x8e, 0x0c, 0xb9, 0x4b, 0xd4, 0x9a, 0x7c, 0xb8, 0xa8, 0x85, 0xda, 0x70, 0x72, 0xd7, 0x08, - 0xda, 0x22, 0xce, 0x9a, 0x6d, 0x23, 0x6c, 0xc7, 0x71, 0xa2, 0x40, 0xcb, 0x94, 0xe7, 0x86, 0x83, - 0xfa, 0xf2, 0xae, 0x11, 0x6c, 0x09, 0x9e, 0xd5, 0x70, 0x73, 0x34, 0x6a, 0x2c, 0x64, 0x73, 0xa0, - 0x6d, 0x98, 0xcf, 0x56, 0x5e, 0xa4, 0x9e, 0xd3, 0x26, 0x6f, 0x70, 0x57, 0xcd, 0x27, 0x32, 0xc8, - 0xe8, 0x4b, 0x0d, 0x6a, 0xe4, 0xfd, 0xec, 0xe3, 0xcf, 0x23, 0xcb, 0xc7, 0x7d, 0xb2, 0x62, 0x6d, - 0x77, 0x1f, 0xfb, 0xb6, 0x71, 0xc8, 0xbf, 0xd6, 0x9c, 0x19, 0x7d, 0x7b, 0x6c, 0xba, 0x66, 0x4b, - 0x12, 0x60, 0x43, 0xf3, 0x54, 0xf0, 0x3a, 0x53, 0x22, 0x0f, 0x2d, 0x9b, 0x63, 0x7d, 0xa2, 0x54, - 0xaa, 0x96, 0xf5, 0xdf, 0x6a, 0x30, 0xa3, 0x7e, 0x54, 0x79, 0xe2, 0x1b, 0x73, 0xe4, 0x48, 0xe6, - 0x1f, 0xd3, 0x91, 0xfc, 0x8d, 0x06, 0xd3, 0xca, 0xb7, 0x9e, 0xa7, 0x67, 0xe8, 0xff, 0x9c, 0x83, - 0x85, 0x6c, 0x35, 0x8f, 0xa5, 0x00, 0xbd, 0x0c, 0x24, 0x95, 0xbc, 0x92, 0xe4, 0x46, 0xf3, 0x23, - 0xf5, 0x27, 0x1d, 0x82, 0xc8, 0x43, 0x47, 0x3e, 0xd2, 0x08, 0x71, 0x74, 0x13, 0x2a, 0x96, 0xf4, - 0x39, 0x28, 0x9f, 0xd5, 0xb5, 0x97, 0x3f, 0x02, 0xb1, 0x2e, 0xc5, 0x98, 0x4f, 0x3f, 0xb2, 0xaa, - 0x66, 0x01, 0x26, 0x48, 0xf2, 0xa6, 0xef, 0x43, 0x91, 0xbb, 0x83, 0x5e, 0x83, 0x32, 0x8d, 0x73, - 0xb4, 0xa6, 0x62, 0x89, 0x3b, 0x4d, 0x3b, 0x08, 0x98, 0xba, 0x90, 0x51, 0x12, 0x18, 0x7a, 0x13, - 0x80, 0x1c, 0x6d, 0x1e, 0xe1, 0x72, 0x34, 0x4e, 0xd0, 0xda, 0xcd, 0x73, 0xcd, 0x91, 0xb0, 0x56, - 0x8e, 0x41, 0xfd, 0xdf, 0x73, 0x50, 0x91, 0x3f, 0x40, 0x3d, 0x90, 0xf1, 0x2f, 0x40, 0xd4, 0xd5, - 0x6d, 0xc3, 0x34, 0xc9, 0x5f, 0x2c, 0x5e, 0x69, 0x2b, 0x63, 0x27, 0x49, 0xfc, 0xbf, 0x2a, 0x24, - 0x58, 0x15, 0x45, 0x3f, 0xf1, 0x5b, 0x29, 0x92, 0x64, 0xb5, 0x9a, 0xa6, 0x2d, 0xee, 0xc1, 0x7c, - 0xa6, 0x2a, 0xb9, 0xf6, 0x99, 0x7c, 0x54, 0xb5, 0xcf, 0x7f, 0x4e, 0xc2, 0x7c, 0xe6, 0x87, 0xbf, - 0x27, 0x7e, 0x8a, 0xd5, 0x13, 0x94, 0x7f, 0x24, 0x27, 0xe8, 0x2b, 0x2d, 0x6b, 0x65, 0xd9, 0x47, - 0x94, 0xb7, 0x8f, 0xf0, 0x35, 0xf4, 0x51, 0xad, 0xb1, 0xba, 0x2d, 0x27, 0x1f, 0xe8, 0x4c, 0x14, - 0x8e, 0x7a, 0x26, 0xd0, 0x2b, 0xac, 0x8c, 0xa5, 0xb6, 0x8a, 0xd4, 0x96, 0x88, 0x10, 0x29, 0x53, - 0x45, 0x0e, 0xa1, 0xf7, 0x61, 0x5a, 0x48, 0xb0, 0xe6, 0x49, 0x29, 0xe9, 0x6c, 0x70, 0x9e, 0x74, - 0xff, 0x64, 0x4a, 0xc6, 0xff, 0xb8, 0x7b, 0xf8, 0x77, 0x1a, 0xcc, 0xa6, 0x6e, 0x02, 0x3c, 0x3d, - 0xef, 0xa0, 0xaf, 0x35, 0x28, 0xc7, 0x97, 0x50, 0x1e, 0x3a, 0x91, 0x5f, 0x85, 0x02, 0x66, 0x17, - 0x21, 0x58, 0xb8, 0x3b, 0x91, 0xba, 0xa8, 0x46, 0x68, 0xfc, 0x6a, 0x5a, 0xea, 0xee, 0x43, 0x8b, - 0x0b, 0xea, 0xff, 0xab, 0x89, 0x14, 0x3d, 0xf1, 0xe9, 0x89, 0x2e, 0x45, 0x32, 0xa6, 0xfc, 0x83, - 0x8e, 0xe9, 0xbf, 0xca, 0x30, 0x49, 0xf9, 0x48, 0x09, 0x1d, 0x62, 0xbf, 0x6f, 0x39, 0x86, 0x4d, - 0x87, 0x53, 0x62, 0xe7, 0x56, 0x60, 0xf2, 0xb9, 0x15, 0x18, 0xda, 0x85, 0xd9, 0xa4, 0xed, 0x47, - 0xd5, 0x64, 0xdf, 0x7f, 0xfb, 0x50, 0x65, 0x62, 0x8d, 0xfd, 0x94, 0xa4, 0x7a, 0x41, 0x20, 0x45, - 0x44, 0x26, 0xcc, 0x74, 0x5d, 0x27, 0x34, 0x2c, 0x07, 0xfb, 0xcc, 0x50, 0x3e, 0xeb, 0xfe, 0xcf, - 0x05, 0x85, 0x87, 0x75, 0x4f, 0x54, 0x39, 0xf5, 0xfe, 0x8f, 0x4a, 0x43, 0x9f, 0xc1, 0xb4, 0x28, - 0x63, 0x98, 0x91, 0x89, 0xac, 0xfb, 0x3f, 0x6b, 0x32, 0x0b, 0xdb, 0xd2, 0x8a, 0x94, 0x7a, 0xff, - 0x47, 0x21, 0x21, 0x1b, 0xaa, 0x9e, 0x6b, 0x6e, 0x3b, 0x3c, 0x79, 0x37, 0x3a, 0x36, 0xe6, 0xbd, - 0xe6, 0xa5, 0x91, 0x94, 0x47, 0xe1, 0x62, 0xa1, 0x38, 0x2d, 0xab, 0xde, 0xa8, 0x4b, 0x53, 0xd1, - 0xa7, 0x30, 0x65, 0x93, 0x6a, 0x72, 0xed, 0xc0, 0xb3, 0x7c, 0x6c, 0x66, 0xdf, 0x7f, 0xbb, 0x2a, - 0x71, 0xb0, 0x40, 0x28, 0xcb, 0xa8, 0x77, 0x80, 0x64, 0x0a, 0x59, 0xfd, 0xbe, 0x71, 0xd0, 0x8a, - 0x9c, 0x60, 0xed, 0x80, 0xdf, 0x65, 0x2a, 0x66, 0xad, 0xfe, 0x86, 0xca, 0xc4, 0x56, 0x3f, 0x25, - 0xa9, 0xae, 0x7e, 0x8a, 0x88, 0xae, 0xd2, 0x38, 0xcf, 0x96, 0x84, 0xdd, 0x83, 0x5b, 0x18, 0x99, - 0x2d, 0xb6, 0x1a, 0xac, 0xed, 0xc3, 0x9f, 0x14, 0xa5, 0xb1, 0x06, 0xbe, 0x06, 0x74, 0xd8, 0x2d, - 0x1c, 0x46, 0xbe, 0x83, 0x4d, 0x5e, 0x54, 0x8d, 0xae, 0x81, 0xc2, 0x15, 0xaf, 0x81, 0x82, 0x8e, - 0xac, 0x81, 0x42, 0x25, 0x7b, 0xca, 0x73, 0xcd, 0x1b, 0xec, 0xc8, 0x84, 0xf1, 0xc5, 0xb8, 0x67, - 0x47, 0x4c, 0x25, 0x2c, 0x6c, 0x4f, 0x29, 0x52, 0xea, 0x9e, 0x52, 0x48, 0xfc, 0x2e, 0x96, 0x7c, - 0x73, 0x87, 0xcd, 0x54, 0x65, 0xcc, 0x5d, 0xac, 0x11, 0xce, 0xf8, 0x2e, 0xd6, 0x08, 0x65, 0xe4, - 0x2e, 0xd6, 0x08, 0x07, 0xb1, 0xde, 0x33, 0x9c, 0xde, 0xba, 0xdb, 0x51, 0x77, 0xf5, 0x54, 0x96, - 0xf5, 0x4b, 0x19, 0x9c, 0xcc, 0x7a, 0x96, 0x0e, 0xd5, 0x7a, 0x16, 0x47, 0xb3, 0x24, 0xda, 0x43, - 0xfa, 0xb7, 0x1a, 0xcc, 0xa6, 0xe2, 0x0c, 0x7a, 0x0f, 0xe2, 0x1b, 0x27, 0x37, 0x0e, 0x3d, 0x91, - 0x26, 0x2b, 0x37, 0x54, 0x08, 0x9e, 0x75, 0x43, 0x85, 0xe0, 0xe8, 0x2a, 0x40, 0xfc, 0x4e, 0xba, - 0x5b, 0x90, 0xa6, 0x39, 0x5a, 0xc2, 0x29, 0xe7, 0x68, 0x09, 0xaa, 0xff, 0x90, 0x87, 0x92, 0xd8, - 0xa8, 0x8f, 0xa5, 0x8c, 0x5a, 0x81, 0x62, 0x1f, 0x07, 0xf4, 0xa6, 0x4a, 0x2e, 0xc9, 0x86, 0x38, - 0x24, 0x67, 0x43, 0x1c, 0x52, 0x93, 0xb5, 0xfc, 0x03, 0x25, 0x6b, 0x13, 0x47, 0x4e, 0xd6, 0x30, - 0xfd, 0x4a, 0x2d, 0x85, 0x5b, 0xf1, 0x5d, 0xe8, 0xee, 0x31, 0x5c, 0x7c, 0xc3, 0x96, 0x05, 0x53, - 0xdf, 0xb0, 0x65, 0x12, 0xda, 0x83, 0xe3, 0xd2, 0xb7, 0x2b, 0xde, 0x3b, 0x24, 0x81, 0x6f, 0x66, - 0xfc, 0x95, 0x80, 0x16, 0xe5, 0x62, 0xc7, 0x7b, 0x2f, 0x85, 0xca, 0xd9, 0x6e, 0x9a, 0xa6, 0xff, - 0x32, 0x07, 0x33, 0xaa, 0xbf, 0x8f, 0x65, 0x61, 0x5f, 0x83, 0x32, 0x3e, 0xb0, 0xc2, 0x76, 0xd7, - 0x35, 0x31, 0x2f, 0x19, 0xe9, 0x3a, 0x11, 0xf0, 0x82, 0x6b, 0x2a, 0xeb, 0x24, 0x30, 0x79, 0x37, - 0xe4, 0x8f, 0xb4, 0x1b, 0x92, 0x56, 0xeb, 0xc4, 0xbd, 0x5b, 0xad, 0xd9, 0xf3, 0x5c, 0x7e, 0x4c, - 0xf3, 0x7c, 0x27, 0x07, 0xd5, 0x74, 0x34, 0xfe, 0x79, 0x1c, 0x21, 0xf5, 0x34, 0xe4, 0x8f, 0x7c, - 0x1a, 0xde, 0x87, 0x69, 0x92, 0x3b, 0x1a, 0x61, 0xc8, 0xef, 0x70, 0x4e, 0xd0, 0x9c, 0x8b, 0xc5, - 0xa6, 0xc8, 0x59, 0x15, 0xb8, 0x12, 0x9b, 0x24, 0x5c, 0xff, 0x32, 0x07, 0xd3, 0xca, 0x5b, 0xe3, - 0xe9, 0x0b, 0x29, 0xfa, 0x2c, 0x4c, 0x2b, 0xc9, 0x98, 0xfe, 0x77, 0x6c, 0x9f, 0xa8, 0x59, 0xd0, - 0xd3, 0x37, 0x2f, 0x33, 0x30, 0x25, 0x67, 0x75, 0x7a, 0x13, 0x66, 0x53, 0x49, 0x98, 0x3c, 0x00, - 0xed, 0x28, 0x03, 0xd0, 0x17, 0x60, 0x2e, 0x2b, 0x77, 0xd0, 0x2f, 0xc1, 0x5c, 0xd6, 0x5b, 0xfd, - 0xfe, 0x0d, 0x7c, 0xa7, 0x51, 0x0b, 0xa3, 0xb7, 0xbd, 0x2f, 0x03, 0x38, 0xf8, 0x76, 0xfb, 0x9e, - 0xe5, 0x1f, 0x9b, 0x4f, 0x7c, 0x7b, 0x3d, 0x55, 0x2d, 0x95, 0x04, 0x46, 0x34, 0xb9, 0xb6, 0xd9, - 0xbe, 0x67, 0xd1, 0x45, 0x35, 0xb9, 0xb6, 0x39, 0xa2, 0x49, 0x60, 0xfa, 0x3f, 0xe4, 0x45, 0x65, - 0x9e, 0x5c, 0x97, 0xfe, 0x04, 0xaa, 0x9e, 0x78, 0xb8, 0xb7, 0xb7, 0xb4, 0x36, 0x89, 0xf9, 0xd3, - 0x96, 0x66, 0x54, 0x8a, 0xaa, 0x9b, 0x17, 0x9d, 0xb9, 0x23, 0xea, 0x6e, 0xa5, 0xaa, 0xcf, 0x19, - 0x95, 0x82, 0xfe, 0x1a, 0x8e, 0x8b, 0xdb, 0x64, 0xfb, 0x58, 0x38, 0x9e, 0x1f, 0xab, 0x9c, 0xdd, - 0xee, 0x8e, 0x05, 0xd2, 0x9e, 0xcf, 0xa6, 0x48, 0x29, 0xf5, 0xdc, 0xf7, 0x89, 0xa3, 0xaa, 0x4f, - 0x3b, 0x3f, 0x9b, 0x22, 0xe9, 0x5f, 0x6b, 0x30, 0x9b, 0xba, 0x80, 0x8e, 0x2e, 0x42, 0x89, 0xfe, - 0x3e, 0xed, 0xee, 0x2b, 0x40, 0x37, 0x24, 0xe5, 0x53, 0x2c, 0x14, 0x39, 0x84, 0xde, 0x80, 0x72, - 0x7c, 0x4f, 0x9d, 0x7f, 0x55, 0x66, 0x87, 0x4f, 0x80, 0xca, 0xe1, 0x13, 0xa0, 0xfe, 0x2f, 0x1a, - 0x9c, 0x1c, 0x7b, 0x39, 0xfd, 0x49, 0xf7, 0x0c, 0x5e, 0x7c, 0x05, 0x4a, 0xe2, 0xbb, 0x2f, 0x02, - 0x28, 0x7c, 0xb4, 0xbd, 0xb6, 0xbd, 0x76, 0xb1, 0x7a, 0x0c, 0x55, 0xa0, 0xb8, 0xb9, 0x76, 0xed, - 0xe2, 0x95, 0x6b, 0x97, 0xaa, 0x1a, 0x79, 0x68, 0x6d, 0x5f, 0xbb, 0x46, 0x1e, 0x72, 0x2f, 0x5e, - 0x95, 0x6f, 0xa1, 0xb1, 0xf7, 0x31, 0x9a, 0x82, 0xd2, 0xaa, 0xe7, 0xd1, 0x00, 0xc0, 0x64, 0xd7, - 0xf6, 0x2d, 0x72, 0x56, 0xab, 0x1a, 0x2a, 0x42, 0xfe, 0xfa, 0xf5, 0x8d, 0x6a, 0x0e, 0xcd, 0x41, - 0xf5, 0x22, 0x36, 0x4c, 0xdb, 0x72, 0xb0, 0x88, 0x3a, 0xd5, 0x7c, 0xf3, 0xd6, 0xf7, 0x3f, 0x2e, - 0x69, 0x3f, 0xfc, 0xb8, 0xa4, 0xfd, 0xe2, 0xc7, 0x25, 0xed, 0xce, 0x4f, 0x4b, 0xc7, 0x7e, 0xf8, - 0x69, 0xe9, 0xd8, 0xff, 0xfd, 0xb4, 0x74, 0xec, 0x93, 0x57, 0xa4, 0xdf, 0x62, 0xb2, 0x31, 0x79, - 0xbe, 0x4b, 0x02, 0x2e, 0x7f, 0x5a, 0x49, 0xff, 0x3a, 0xf5, 0xbb, 0xdc, 0xe9, 0x55, 0xfa, 0xb8, - 0xc9, 0xf8, 0x1a, 0x57, 0xdc, 0x06, 0x03, 0xe8, 0x0f, 0x08, 0x83, 0x4e, 0x81, 0xfe, 0x50, 0xf0, - 0xb5, 0x3f, 0x04, 0x00, 0x00, 0xff, 0xff, 0xe4, 0xb3, 0x51, 0x43, 0xd8, 0x3a, 0x00, 0x00, + 0xb9, 0xf6, 0x90, 0x12, 0x1f, 0x3f, 0xf5, 0xa0, 0x8f, 0x25, 0x85, 0x56, 0x6c, 0x51, 0x1e, 0xe7, + 0xde, 0x38, 0x41, 0x42, 0x26, 0xce, 0x03, 0x79, 0x5c, 0x24, 0x10, 0x6d, 0xc5, 0xb6, 0x62, 0xd9, + 0x0a, 0x65, 0xe5, 0xfa, 0x06, 0xb9, 0x60, 0x86, 0x9c, 0x23, 0x6a, 0xac, 0xe1, 0x0c, 0x33, 0x0f, + 0x59, 0x02, 0xb2, 0xb8, 0xb9, 0xb8, 0x37, 0xdd, 0xa5, 0x06, 0x9a, 0x45, 0x81, 0x2e, 0xd2, 0x6d, + 0x03, 0x74, 0xdd, 0x75, 0x57, 0xcd, 0xa2, 0x28, 0xd2, 0x5d, 0x57, 0x6c, 0x91, 0xa0, 0x8b, 0x72, + 0xd1, 0x75, 0xdb, 0x4d, 0x8b, 0xf3, 0x9a, 0x39, 0x67, 0x38, 0xb4, 0xe5, 0x57, 0x9d, 0xc2, 0x2b, + 0x69, 0xbe, 0xff, 0x39, 0xe7, 0xf1, 0xcf, 0xff, 0xff, 0xe7, 0x10, 0x4e, 0xf6, 0x77, 0xbb, 0x75, + 0xc3, 0xeb, 0x19, 0xa6, 0x81, 0xf7, 0xb0, 0x13, 0xf8, 0x75, 0xf6, 0xa7, 0xd6, 0xf7, 0xdc, 0xc0, + 0x45, 0x53, 0x32, 0x69, 0x51, 0xdf, 0x7d, 0xcd, 0xaf, 0x59, 0x6e, 0xdd, 0xe8, 0x5b, 0xf5, 0x8e, + 0xeb, 0xe1, 0xfa, 0xde, 0x8b, 0xf5, 0x2e, 0x76, 0xb0, 0x67, 0x04, 0xd8, 0x64, 0x12, 0x8b, 0x67, + 0x24, 0x1e, 0x07, 0x07, 0x37, 0x5d, 0x6f, 0xd7, 0x72, 0xba, 0x69, 0x9c, 0xd5, 0xae, 0xeb, 0x76, + 0x6d, 0x5c, 0xa7, 0x4f, 0xed, 0x70, 0xbb, 0x1e, 0x58, 0x3d, 0xec, 0x07, 0x46, 0xaf, 0xcf, 0x19, + 0x5e, 0x8e, 0x55, 0xf5, 0x8c, 0xce, 0x8e, 0xe5, 0x60, 0xef, 0xa0, 0x4e, 0xfd, 0xed, 0x5b, 0x75, + 0x0f, 0xfb, 0x6e, 0xe8, 0x75, 0xf0, 0x88, 0xda, 0xe7, 0xbb, 0x56, 0xb0, 0x13, 0xb6, 0x6b, 0x1d, + 0xb7, 0x57, 0xef, 0xba, 0x5d, 0x37, 0xd6, 0x4f, 0x9e, 0xe8, 0x03, 0xfd, 0x8f, 0xb3, 0xbf, 0x61, + 0x39, 0x01, 0xf6, 0x1c, 0xc3, 0xae, 0xfb, 0x9d, 0x1d, 0x6c, 0x86, 0x36, 0xf6, 0xe2, 0xff, 0xdc, + 0xf6, 0x0d, 0xdc, 0x09, 0xfc, 0x11, 0x80, 0xc9, 0xea, 0xb7, 0xe6, 0x60, 0x7a, 0x95, 0x0c, 0xcd, + 0x26, 0xfe, 0x38, 0xc4, 0x4e, 0x07, 0xa3, 0x67, 0x60, 0xf2, 0xe3, 0x10, 0x87, 0xb8, 0xa2, 0x2d, + 0x6b, 0x67, 0x8a, 0x8d, 0x63, 0xc3, 0x41, 0x75, 0x96, 0x02, 0xcf, 0xb9, 0x3d, 0x2b, 0xc0, 0xbd, + 0x7e, 0x70, 0xd0, 0x64, 0x1c, 0xe8, 0x0d, 0x98, 0xba, 0xe1, 0xb6, 0x5b, 0x3e, 0x0e, 0x5a, 0x8e, + 0xd1, 0xc3, 0x95, 0x0c, 0x95, 0xa8, 0x0c, 0x07, 0xd5, 0xb9, 0x1b, 0x6e, 0x7b, 0x13, 0x07, 0x57, + 0x8c, 0x9e, 0x2c, 0x06, 0x31, 0x8a, 0x9e, 0x87, 0x7c, 0xe8, 0x63, 0xaf, 0x65, 0x99, 0x95, 0x2c, + 0x15, 0x9b, 0x1b, 0x0e, 0xaa, 0x65, 0x02, 0x5d, 0x32, 0x25, 0x91, 0x1c, 0x43, 0xd0, 0x73, 0x90, + 0xeb, 0x7a, 0x6e, 0xd8, 0xf7, 0x2b, 0x13, 0xcb, 0x59, 0xc1, 0xcd, 0x10, 0x99, 0x9b, 0x21, 0xe8, + 0x2a, 0xe4, 0xd8, 0x7c, 0x57, 0x26, 0x97, 0xb3, 0x67, 0x4a, 0x67, 0x4f, 0xd5, 0xe4, 0x45, 0x50, + 0x53, 0x5e, 0x98, 0x3d, 0x31, 0x85, 0x8c, 0x2e, 0x2b, 0xe4, 0xcb, 0xe6, 0x4f, 0x47, 0x61, 0x92, + 0xf2, 0xa1, 0xab, 0x90, 0xef, 0x78, 0x98, 0x4c, 0x56, 0x05, 0x2d, 0x6b, 0x67, 0x4a, 0x67, 0x17, + 0x6b, 0x6c, 0x11, 0xd4, 0xc4, 0x24, 0xd5, 0xae, 0x89, 0x45, 0xd0, 0x38, 0x3e, 0x1c, 0x54, 0x8f, + 0x72, 0xf6, 0x58, 0xeb, 0xad, 0xdf, 0x57, 0xb5, 0xa6, 0xd0, 0x82, 0x36, 0xa0, 0xe8, 0x87, 0xed, + 0x9e, 0x15, 0xac, 0xb9, 0x6d, 0x3a, 0xe6, 0xa5, 0xb3, 0x4f, 0xa8, 0xee, 0x6e, 0x0a, 0x72, 0xe3, + 0x89, 0xe1, 0xa0, 0x7a, 0x2c, 0xe2, 0x8e, 0x35, 0x5e, 0x3c, 0xd2, 0x8c, 0x95, 0xa0, 0x1d, 0x98, + 0xf5, 0x70, 0xdf, 0xb3, 0x5c, 0xcf, 0x0a, 0x2c, 0x1f, 0x13, 0xbd, 0x19, 0xaa, 0xf7, 0xa4, 0xaa, + 0xb7, 0xa9, 0x32, 0x35, 0x4e, 0x0e, 0x07, 0xd5, 0xe3, 0x09, 0x49, 0xc5, 0x46, 0x52, 0x2d, 0x0a, + 0x00, 0x25, 0xa0, 0x4d, 0x1c, 0xd0, 0xf9, 0x2c, 0x9d, 0x5d, 0xbe, 0xad, 0xb1, 0x4d, 0x1c, 0x34, + 0x96, 0x87, 0x83, 0xea, 0x89, 0x51, 0x79, 0xc5, 0x64, 0x8a, 0x7e, 0x64, 0x43, 0x59, 0x46, 0x4d, + 0xf2, 0x82, 0x13, 0xd4, 0xe6, 0xd2, 0x78, 0x9b, 0x84, 0xab, 0xb1, 0x34, 0x1c, 0x54, 0x17, 0x93, + 0xb2, 0x8a, 0xbd, 0x11, 0xcd, 0x64, 0x7e, 0x3a, 0x86, 0xd3, 0xc1, 0x36, 0x31, 0x33, 0x99, 0x36, + 0x3f, 0xe7, 0x04, 0x99, 0xcd, 0x4f, 0xc4, 0xad, 0xce, 0x4f, 0x04, 0xa3, 0x0f, 0x61, 0x2a, 0x7a, + 0x20, 0xe3, 0x95, 0xe3, 0xeb, 0x28, 0x5d, 0x29, 0x19, 0xa9, 0xc5, 0xe1, 0xa0, 0xba, 0x20, 0xcb, + 0x28, 0xaa, 0x15, 0x6d, 0xb1, 0x76, 0x9b, 0x8d, 0x4c, 0x7e, 0xbc, 0x76, 0xc6, 0x21, 0x6b, 0xb7, + 0x47, 0x47, 0x44, 0xd1, 0x46, 0xb4, 0x93, 0x4d, 0x1c, 0x76, 0x3a, 0x18, 0x9b, 0xd8, 0xac, 0x14, + 0xd2, 0xb4, 0xaf, 0x49, 0x1c, 0x4c, 0xbb, 0x2c, 0xa3, 0x6a, 0x97, 0x29, 0x64, 0xac, 0x6f, 0xb8, + 0xed, 0x55, 0xcf, 0x73, 0x3d, 0xbf, 0x52, 0x4c, 0x1b, 0xeb, 0x35, 0x41, 0x66, 0x63, 0x1d, 0x71, + 0xab, 0x63, 0x1d, 0xc1, 0xdc, 0xdf, 0x66, 0xe8, 0x5c, 0xc6, 0x86, 0x8f, 0xcd, 0x0a, 0x8c, 0xf1, + 0x37, 0xe2, 0x88, 0xfc, 0x8d, 0x90, 0x11, 0x7f, 0x23, 0x0a, 0x32, 0x61, 0x86, 0x3d, 0xaf, 0xf8, + 0xbe, 0xd5, 0x75, 0xb0, 0x59, 0x29, 0x51, 0xfd, 0x27, 0xd2, 0xf4, 0x0b, 0x9e, 0xc6, 0x89, 0xe1, + 0xa0, 0x5a, 0x51, 0xe5, 0x14, 0x1b, 0x09, 0x9d, 0xe8, 0x23, 0x98, 0x66, 0x48, 0x33, 0x74, 0x1c, + 0xcb, 0xe9, 0x56, 0xa6, 0xa8, 0x91, 0x27, 0xd3, 0x8c, 0x70, 0x96, 0xc6, 0x93, 0xc3, 0x41, 0xf5, + 0x09, 0x45, 0x4a, 0x31, 0xa1, 0x2a, 0x24, 0x11, 0x83, 0x01, 0xf1, 0xc4, 0x4e, 0xa7, 0x45, 0x8c, + 0x35, 0x95, 0x89, 0x45, 0x8c, 0x84, 0xa4, 0x1a, 0x31, 0x12, 0xc4, 0x78, 0x3e, 0xf8, 0x24, 0xcf, + 0x8c, 0x9f, 0x0f, 0x3e, 0xcf, 0xd2, 0x7c, 0xa4, 0x4c, 0xb5, 0xa2, 0x0d, 0x7d, 0x02, 0xe4, 0xc3, + 0x73, 0x3e, 0xec, 0xdb, 0x56, 0xc7, 0x08, 0xf0, 0x79, 0x1c, 0xe0, 0x0e, 0x89, 0xd4, 0xb3, 0xd4, + 0x8a, 0x3e, 0x62, 0x65, 0x84, 0xb3, 0xa1, 0x0f, 0x07, 0xd5, 0xa5, 0x34, 0x1d, 0x8a, 0xd5, 0x54, + 0x2b, 0xe8, 0x7f, 0x34, 0x98, 0xf7, 0x03, 0xc3, 0x31, 0x0d, 0xdb, 0x75, 0xf0, 0x25, 0xa7, 0xeb, + 0x61, 0xdf, 0xbf, 0xe4, 0x6c, 0xbb, 0x95, 0x32, 0xb5, 0x7f, 0x3a, 0x11, 0xd6, 0xd3, 0x58, 0x1b, + 0xa7, 0x87, 0x83, 0x6a, 0x35, 0x55, 0x8b, 0xe2, 0x41, 0xba, 0x21, 0xb4, 0x0f, 0xc7, 0x44, 0x56, + 0xb1, 0x15, 0x58, 0xb6, 0xe5, 0x1b, 0x81, 0xe5, 0x3a, 0x95, 0xa3, 0xd4, 0xfe, 0xa9, 0x64, 0x74, + 0x1c, 0x61, 0x6c, 0x9c, 0x1a, 0x0e, 0xaa, 0x27, 0x53, 0x34, 0x28, 0xb6, 0xd3, 0x4c, 0xc4, 0x4b, + 0x68, 0xc3, 0xc3, 0x84, 0x11, 0x9b, 0x95, 0x63, 0xe3, 0x97, 0x50, 0xc4, 0x24, 0x2f, 0xa1, 0x08, + 0x4c, 0x5b, 0x42, 0x11, 0x91, 0x58, 0xea, 0x1b, 0x5e, 0x60, 0x11, 0xb3, 0xeb, 0x86, 0xb7, 0x8b, + 0xbd, 0xca, 0x5c, 0x9a, 0xa5, 0x0d, 0x95, 0x89, 0x59, 0x4a, 0x48, 0xaa, 0x96, 0x12, 0x44, 0x74, + 0x4b, 0x03, 0xd5, 0x35, 0xcb, 0x75, 0x9a, 0x24, 0x6d, 0xf0, 0xc9, 0xeb, 0xcd, 0x53, 0xa3, 0x4f, + 0xdf, 0xe6, 0xf5, 0x64, 0xf6, 0xc6, 0xd3, 0xc3, 0x41, 0xf5, 0xf4, 0x58, 0x6d, 0x8a, 0x23, 0xe3, + 0x8d, 0xa2, 0xeb, 0x50, 0x22, 0x44, 0x4c, 0x13, 0x30, 0xb3, 0xb2, 0x40, 0x7d, 0x38, 0x3e, 0xea, + 0x03, 0x67, 0xa0, 0x19, 0xc8, 0xbc, 0x24, 0xa1, 0xd8, 0x91, 0x55, 0x35, 0xf2, 0x30, 0x49, 0xe5, + 0xf5, 0x61, 0x0e, 0x8e, 0xa5, 0xac, 0x0d, 0xf4, 0x16, 0xe4, 0xbc, 0xd0, 0x21, 0x09, 0x1b, 0xcb, + 0x52, 0x90, 0x6a, 0x75, 0x2b, 0xb4, 0x4c, 0x96, 0x2d, 0x7a, 0xa1, 0xa3, 0xe4, 0x70, 0x93, 0x14, + 0x20, 0xf2, 0x24, 0x5b, 0xb4, 0x4c, 0x9e, 0x8d, 0x8c, 0x95, 0xbf, 0xe1, 0xb6, 0x55, 0x79, 0x0a, + 0x20, 0x0c, 0xd3, 0x62, 0xe1, 0xb5, 0x2c, 0xb2, 0xab, 0x58, 0x9e, 0xf1, 0x94, 0xaa, 0xe6, 0xdd, + 0xb0, 0x8d, 0x3d, 0x07, 0x07, 0xd8, 0x17, 0xef, 0x40, 0xb7, 0x15, 0x8d, 0x22, 0x9e, 0x84, 0x48, + 0xfa, 0xa7, 0x64, 0x1c, 0x7d, 0xa1, 0x41, 0xa5, 0x67, 0xec, 0xb7, 0x04, 0xe8, 0xb7, 0xb6, 0x5d, + 0xaf, 0xd5, 0xc7, 0x9e, 0xe5, 0x9a, 0x34, 0xf9, 0x2c, 0x9d, 0xfd, 0x8f, 0x3b, 0x6e, 0xa4, 0xda, + 0xba, 0xb1, 0x2f, 0x60, 0xff, 0x1d, 0xd7, 0xdb, 0xa0, 0xe2, 0xab, 0x4e, 0xe0, 0x1d, 0x34, 0x4e, + 0x7e, 0x3d, 0xa8, 0x1e, 0x21, 0xd3, 0xd2, 0x4b, 0xe3, 0x69, 0xa6, 0xc3, 0xe8, 0x87, 0x1a, 0x2c, + 0x04, 0x6e, 0x60, 0xd8, 0xad, 0x4e, 0xd8, 0x0b, 0x6d, 0x23, 0xb0, 0xf6, 0x70, 0x2b, 0xf4, 0x8d, + 0x2e, 0xe6, 0x39, 0xee, 0x9b, 0x77, 0x76, 0xea, 0x1a, 0x91, 0x3f, 0x17, 0x89, 0x6f, 0x11, 0x69, + 0xe6, 0xd3, 0x09, 0xee, 0xd3, 0x5c, 0x90, 0xc2, 0xd2, 0x4c, 0x45, 0x17, 0x7f, 0xaa, 0xc1, 0xe2, + 0xf8, 0xd7, 0x44, 0xa7, 0x21, 0xbb, 0x8b, 0x0f, 0x78, 0x15, 0x71, 0x74, 0x38, 0xa8, 0x4e, 0xef, + 0xe2, 0x03, 0x69, 0xd4, 0x09, 0x15, 0xfd, 0x17, 0x4c, 0xee, 0x19, 0x76, 0x88, 0xf9, 0x92, 0xa8, + 0xd5, 0x58, 0xbd, 0x54, 0x93, 0xeb, 0xa5, 0x5a, 0x7f, 0xb7, 0x4b, 0x80, 0x9a, 0x98, 0x91, 0xda, + 0x7b, 0xa1, 0xe1, 0x04, 0x56, 0x70, 0xc0, 0x96, 0x0b, 0x55, 0x20, 0x2f, 0x17, 0x0a, 0xbc, 0x91, + 0x79, 0x4d, 0x5b, 0xfc, 0x52, 0x83, 0xe3, 0x63, 0x5f, 0xfa, 0xfb, 0xe0, 0xa1, 0xde, 0x82, 0x09, + 0xb2, 0xf0, 0x49, 0x7d, 0xb3, 0x63, 0x75, 0x77, 0x5e, 0x7d, 0x99, 0xba, 0x93, 0x63, 0xe5, 0x08, + 0x43, 0xe4, 0x72, 0x84, 0x21, 0xa4, 0x46, 0xb3, 0xdd, 0x9b, 0xaf, 0xbe, 0x4c, 0x9d, 0xca, 0x31, + 0x23, 0x14, 0x90, 0x8d, 0x50, 0x40, 0xff, 0x7b, 0x0e, 0x8a, 0x51, 0x01, 0x21, 0xed, 0x41, 0xed, + 0x9e, 0xf6, 0xe0, 0x45, 0x28, 0x9b, 0xd8, 0xe4, 0x5f, 0x3e, 0xcb, 0x75, 0xc4, 0x6e, 0x2e, 0xb2, + 0xe8, 0xaa, 0xd0, 0x14, 0xf9, 0xd9, 0x04, 0x09, 0x9d, 0x85, 0x02, 0x4f, 0xb4, 0x0f, 0xe8, 0x46, + 0x9e, 0x6e, 0x2c, 0x0c, 0x07, 0x55, 0x24, 0x30, 0x49, 0x34, 0xe2, 0x43, 0x4d, 0x00, 0x56, 0xbd, + 0xae, 0xe3, 0xc0, 0xe0, 0x29, 0x7f, 0x45, 0x7d, 0x83, 0xab, 0x11, 0x9d, 0xd5, 0xa1, 0x31, 0xbf, + 0x5c, 0x87, 0xc6, 0x28, 0xfa, 0x10, 0xa0, 0x67, 0x58, 0x0e, 0x93, 0xe3, 0xf9, 0xbd, 0x3e, 0x2e, + 0xa4, 0xac, 0x47, 0x9c, 0x4c, 0x7b, 0x2c, 0x29, 0x6b, 0x8f, 0x51, 0x52, 0x2d, 0xf2, 0x7a, 0xbb, + 0x92, 0xa3, 0xbb, 0x74, 0x69, 0x9c, 0x6a, 0xae, 0x76, 0x9e, 0x54, 0x8c, 0x5c, 0x44, 0xd2, 0x29, + 0xb4, 0x90, 0x61, 0xb3, 0xad, 0x6d, 0x1c, 0x58, 0x3d, 0x4c, 0x33, 0x7b, 0x3e, 0x6c, 0x02, 0x93, + 0x87, 0x4d, 0x60, 0xe8, 0x35, 0x00, 0x23, 0x58, 0x77, 0xfd, 0xe0, 0xaa, 0xd3, 0xc1, 0x34, 0x63, + 0x2f, 0x30, 0xf7, 0x63, 0x54, 0x76, 0x3f, 0x46, 0xd1, 0x9b, 0x50, 0xea, 0xf3, 0x8f, 0x50, 0xdb, + 0xc6, 0x34, 0x23, 0x2f, 0xb0, 0x4f, 0x8a, 0x04, 0x4b, 0xb2, 0x32, 0x37, 0xba, 0x00, 0xb3, 0x1d, + 0xd7, 0xe9, 0x84, 0x9e, 0x87, 0x9d, 0xce, 0xc1, 0xa6, 0xb1, 0x8d, 0x69, 0xf6, 0x5d, 0x60, 0x4b, + 0x25, 0x41, 0x92, 0x97, 0x4a, 0x82, 0x84, 0x5e, 0x81, 0x62, 0xd4, 0xbd, 0xa0, 0x09, 0x76, 0x91, + 0x17, 0xc2, 0x02, 0x94, 0x84, 0x63, 0x4e, 0xe2, 0xbc, 0xe5, 0x47, 0x59, 0x1a, 0x4d, 0x9a, 0xb9, + 0xf3, 0x12, 0x2c, 0x3b, 0x2f, 0xc1, 0xe8, 0x12, 0x1c, 0xa5, 0xdf, 0xc5, 0x56, 0x10, 0xd8, 0x2d, + 0x1f, 0x77, 0x5c, 0xc7, 0xf4, 0x69, 0x4e, 0x9c, 0x65, 0xee, 0x53, 0xe2, 0xb5, 0xc0, 0xde, 0x64, + 0x24, 0xd9, 0xfd, 0x04, 0x49, 0xff, 0xb5, 0x06, 0x73, 0x69, 0x4b, 0x28, 0xb1, 0x9c, 0xb5, 0x07, + 0xb2, 0x9c, 0xdf, 0x87, 0x42, 0xdf, 0x35, 0x5b, 0x7e, 0x1f, 0x77, 0x78, 0xc4, 0x4a, 0x2c, 0xe6, + 0x0d, 0xd7, 0xdc, 0xec, 0xe3, 0xce, 0x7f, 0x5a, 0xc1, 0xce, 0xca, 0x9e, 0x6b, 0x99, 0x97, 0x2d, + 0x9f, 0xaf, 0xba, 0x3e, 0xa3, 0x28, 0x19, 0x42, 0x9e, 0x83, 0x8d, 0x02, 0xe4, 0x98, 0x15, 0xfd, + 0x37, 0x59, 0x28, 0x27, 0x97, 0xed, 0xbf, 0xd2, 0xab, 0xa0, 0xeb, 0x90, 0xb7, 0x58, 0xca, 0xcc, + 0x33, 0x88, 0x7f, 0x93, 0x62, 0x7a, 0x2d, 0x6e, 0xf8, 0xd5, 0xf6, 0x5e, 0xac, 0xf1, 0xdc, 0x9a, + 0x0e, 0x01, 0xd5, 0xcc, 0x25, 0x55, 0xcd, 0x1c, 0x44, 0x4d, 0xc8, 0xfb, 0xd8, 0xdb, 0xb3, 0x3a, + 0x98, 0x07, 0xa7, 0xaa, 0xac, 0xb9, 0xe3, 0x7a, 0x98, 0xe8, 0xdc, 0x64, 0x2c, 0xb1, 0x4e, 0x2e, + 0xa3, 0xea, 0xe4, 0x20, 0x7a, 0x1f, 0x8a, 0x1d, 0xd7, 0xd9, 0xb6, 0xba, 0xeb, 0x46, 0x9f, 0x87, + 0xa7, 0x93, 0x69, 0x5a, 0xcf, 0x09, 0x26, 0xde, 0x84, 0x10, 0x8f, 0x89, 0x26, 0x44, 0xc4, 0x15, + 0x4f, 0xe8, 0x9f, 0x27, 0x00, 0xe2, 0xc9, 0x41, 0xaf, 0x43, 0x09, 0xef, 0xe3, 0x4e, 0x18, 0xb8, + 0x9e, 0xf8, 0x4e, 0xf0, 0x9e, 0x9e, 0x80, 0x95, 0xc0, 0x0e, 0x31, 0x4a, 0x36, 0xaa, 0x63, 0xf4, + 0xb0, 0xdf, 0x37, 0x3a, 0xa2, 0x19, 0x48, 0x9d, 0x89, 0x40, 0x79, 0xa3, 0x46, 0x20, 0xfa, 0x77, + 0x98, 0xa0, 0xed, 0x43, 0xd6, 0x07, 0x44, 0xc3, 0x41, 0x75, 0xc6, 0x51, 0x1b, 0x87, 0x94, 0x8e, + 0xde, 0x86, 0xe9, 0xdd, 0x68, 0xe1, 0x11, 0xdf, 0x26, 0xa8, 0x00, 0x4d, 0xed, 0x62, 0x82, 0xe2, + 0xdd, 0x94, 0x8c, 0xa3, 0x6d, 0x28, 0x19, 0x8e, 0xe3, 0x06, 0xf4, 0x1b, 0x24, 0x7a, 0x83, 0xcf, + 0x8c, 0x5b, 0xa6, 0xb5, 0x95, 0x98, 0x97, 0x65, 0x49, 0x34, 0x78, 0x48, 0x1a, 0xe4, 0xe0, 0x21, + 0xc1, 0xa8, 0x09, 0x39, 0xdb, 0x68, 0x63, 0x5b, 0x04, 0xfd, 0xa7, 0xc6, 0x9a, 0xb8, 0x4c, 0xd9, + 0x98, 0x76, 0xfa, 0xc9, 0x67, 0x72, 0xf2, 0x27, 0x9f, 0x21, 0x8b, 0xdb, 0x50, 0x4e, 0xfa, 0x73, + 0xb8, 0x04, 0xe6, 0x19, 0x39, 0x81, 0x29, 0xde, 0x31, 0x65, 0x32, 0xa0, 0x24, 0x39, 0xf5, 0x30, + 0x4c, 0xe8, 0x3f, 0xd3, 0x60, 0x2e, 0x6d, 0xef, 0xa2, 0x75, 0x69, 0xc7, 0x6b, 0xbc, 0xc7, 0x91, + 0xb2, 0xd4, 0xb9, 0xec, 0x98, 0xad, 0x1e, 0x6f, 0xf4, 0x06, 0xcc, 0x38, 0xae, 0x89, 0x5b, 0x06, + 0x31, 0x60, 0x5b, 0x7e, 0x50, 0xc9, 0xd0, 0xde, 0x31, 0xed, 0x8d, 0x10, 0xca, 0x8a, 0x20, 0x48, + 0xd2, 0xd3, 0x0a, 0x41, 0xff, 0x7f, 0x0d, 0x66, 0x13, 0xad, 0xcb, 0xfb, 0x4e, 0xa2, 0xe4, 0xd4, + 0x27, 0x73, 0xb8, 0xd4, 0x47, 0xff, 0x51, 0x06, 0x4a, 0x52, 0x5d, 0x77, 0xdf, 0x3e, 0xdc, 0x80, + 0x59, 0xfe, 0xa5, 0xb4, 0x9c, 0x2e, 0x2b, 0xa7, 0x32, 0xbc, 0x49, 0x31, 0x72, 0x52, 0xb0, 0xe6, + 0xb6, 0x37, 0x23, 0x5e, 0x5a, 0x4d, 0xd1, 0x0e, 0x96, 0xaf, 0x60, 0x92, 0x89, 0x19, 0x95, 0x82, + 0xae, 0xc3, 0x42, 0xd8, 0x37, 0x8d, 0x00, 0xb7, 0x7c, 0xde, 0x73, 0x6f, 0x39, 0x61, 0xaf, 0x8d, + 0x3d, 0xba, 0xe3, 0x27, 0x59, 0xcf, 0x85, 0x71, 0x88, 0xa6, 0xfc, 0x15, 0x4a, 0x97, 0x74, 0xce, + 0xa5, 0xd1, 0xf5, 0x8b, 0x80, 0x46, 0xfb, 0xca, 0xca, 0xf8, 0x6a, 0x87, 0x1c, 0xdf, 0xcf, 0x34, + 0x28, 0x27, 0xdb, 0xc5, 0x8f, 0x64, 0xa2, 0x0f, 0xa0, 0x18, 0xb5, 0x7e, 0xef, 0xdb, 0x81, 0xe7, + 0x20, 0xe7, 0x61, 0xc3, 0x77, 0x1d, 0xbe, 0x33, 0x69, 0x88, 0x61, 0x88, 0x1c, 0x62, 0x18, 0xa2, + 0x5f, 0x83, 0x29, 0x36, 0x82, 0xef, 0x58, 0x76, 0x80, 0x3d, 0x74, 0x1e, 0x72, 0x7e, 0x60, 0x04, + 0xd8, 0xaf, 0x68, 0xcb, 0xd9, 0x33, 0x33, 0x67, 0x17, 0x46, 0xbb, 0xbc, 0x84, 0xcc, 0xb4, 0x32, + 0x4e, 0x59, 0x2b, 0x43, 0xf4, 0xff, 0xd5, 0x60, 0x4a, 0x6e, 0x66, 0x3f, 0x18, 0xb5, 0x77, 0xf9, + 0x6a, 0x9f, 0x08, 0x1f, 0xec, 0x07, 0x33, 0xb3, 0x77, 0x67, 0xfd, 0x17, 0x1a, 0x1b, 0xd9, 0xa8, + 0x0b, 0x7a, 0xbf, 0xe6, 0xbb, 0x71, 0x2b, 0x84, 0xec, 0x30, 0x9f, 0x06, 0xb6, 0xc3, 0xb6, 0x42, + 0x68, 0xf8, 0x53, 0xc4, 0xe5, 0xf0, 0xa7, 0x10, 0xf4, 0x2f, 0x26, 0xa9, 0xe7, 0x71, 0xc7, 0xfb, + 0x51, 0x37, 0x81, 0x12, 0xd9, 0x49, 0xf6, 0x2e, 0xb2, 0x93, 0xe7, 0x21, 0x4f, 0x3f, 0x07, 0x51, + 0xe2, 0x40, 0x27, 0x8d, 0x40, 0xea, 0x89, 0x23, 0x43, 0x6e, 0x13, 0xb5, 0x26, 0xef, 0x2f, 0x6a, + 0xa1, 0x16, 0x1c, 0xdf, 0x31, 0xfc, 0x96, 0x88, 0xb3, 0x66, 0xcb, 0x08, 0x5a, 0x51, 0x9c, 0xc8, + 0xd1, 0x32, 0xe5, 0xa9, 0xe1, 0xa0, 0xba, 0xbc, 0x63, 0xf8, 0x9b, 0x82, 0x67, 0x25, 0xd8, 0x18, + 0x8d, 0x1a, 0x0b, 0xe9, 0x1c, 0x68, 0x0b, 0xe6, 0xd3, 0x95, 0xe7, 0xa9, 0xe7, 0xb4, 0xc9, 0xeb, + 0xdf, 0x56, 0xf3, 0xb1, 0x14, 0x32, 0xfa, 0x54, 0x83, 0x0a, 0xf9, 0x3e, 0x7b, 0xf8, 0xe3, 0xd0, + 0xf2, 0x70, 0x8f, 0xcc, 0x58, 0xcb, 0xdd, 0xc3, 0x9e, 0x6d, 0x1c, 0xf0, 0xd3, 0x9a, 0x53, 0xa3, + 0x5f, 0x8f, 0x0d, 0xd7, 0x6c, 0x4a, 0x02, 0xec, 0xd5, 0xfa, 0x2a, 0x78, 0x95, 0x29, 0x91, 0x5f, + 0x2d, 0x9d, 0x63, 0x6d, 0xa2, 0x50, 0x28, 0x17, 0xf5, 0xbf, 0x6a, 0x30, 0xa3, 0x1e, 0xaa, 0x3c, + 0xf2, 0x85, 0x39, 0xb2, 0x25, 0xb3, 0x0f, 0x69, 0x4b, 0xfe, 0x45, 0x83, 0x69, 0xe5, 0xac, 0xe7, + 0xf1, 0x79, 0xf5, 0x1f, 0x67, 0x60, 0x21, 0x5d, 0xcd, 0x43, 0x29, 0x40, 0x2f, 0x02, 0x49, 0x25, + 0x2f, 0xc5, 0xb9, 0xd1, 0xfc, 0x48, 0xfd, 0x49, 0x5f, 0x41, 0xe4, 0xa1, 0x23, 0x87, 0x34, 0x42, + 0x1c, 0x5d, 0x87, 0x92, 0x25, 0x1d, 0x07, 0x65, 0xd3, 0xba, 0xf6, 0xf2, 0x21, 0x10, 0xeb, 0x52, + 0x8c, 0x39, 0xfa, 0x91, 0x55, 0x35, 0x72, 0x30, 0x41, 0x92, 0x37, 0x7d, 0x0f, 0xf2, 0xdc, 0x1d, + 0xf4, 0x12, 0x14, 0x69, 0x9c, 0xa3, 0x35, 0x15, 0x4b, 0xdc, 0x69, 0xda, 0x41, 0xc0, 0xc4, 0x85, + 0x8c, 0x82, 0xc0, 0xd0, 0xab, 0x00, 0x64, 0x6b, 0xf3, 0x08, 0x97, 0xa1, 0x71, 0x82, 0xd6, 0x6e, + 0x7d, 0xd7, 0x1c, 0x09, 0x6b, 0xc5, 0x08, 0xd4, 0x7f, 0x9e, 0x81, 0x92, 0x7c, 0x00, 0x75, 0x4f, + 0xc6, 0x3f, 0x01, 0x51, 0x57, 0xb7, 0x0c, 0xd3, 0x24, 0x7f, 0xb1, 0xf8, 0xa4, 0xd5, 0xc7, 0x0e, + 0x92, 0xf8, 0x7f, 0x45, 0x48, 0xb0, 0x2a, 0x8a, 0x1e, 0xf1, 0x5b, 0x09, 0x92, 0x64, 0xb5, 0x9c, + 0xa4, 0x2d, 0xee, 0xc2, 0x7c, 0xaa, 0x2a, 0xb9, 0xf6, 0x99, 0x7c, 0x50, 0xb5, 0xcf, 0x2f, 0x27, + 0x61, 0x3e, 0xf5, 0xe0, 0xef, 0x91, 0xef, 0x62, 0x75, 0x07, 0x65, 0x1f, 0xc8, 0x0e, 0xfa, 0x4c, + 0x4b, 0x9b, 0x59, 0x76, 0x88, 0xf2, 0xfa, 0x21, 0x4e, 0x43, 0x1f, 0xd4, 0x1c, 0xab, 0xcb, 0x72, + 0xf2, 0x9e, 0xf6, 0x44, 0xee, 0xb0, 0x7b, 0x02, 0xbd, 0xc0, 0xca, 0x58, 0x6a, 0x2b, 0x4f, 0x6d, + 0x89, 0x08, 0x91, 0x30, 0x95, 0xe7, 0x10, 0x7a, 0x1b, 0xa6, 0x85, 0x04, 0x6b, 0x9e, 0x14, 0xe2, + 0xce, 0x06, 0xe7, 0x49, 0xf6, 0x4f, 0xa6, 0x64, 0xfc, 0x9f, 0xbb, 0x86, 0xff, 0xa6, 0xc1, 0x6c, + 0xe2, 0x26, 0xc0, 0xe3, 0xf3, 0x0d, 0xfa, 0x5c, 0x83, 0x62, 0x74, 0x09, 0xe5, 0xbe, 0x13, 0xf9, + 0x15, 0xc8, 0x61, 0x76, 0x11, 0x82, 0x85, 0xbb, 0x63, 0x89, 0x8b, 0x6a, 0x84, 0xc6, 0xaf, 0xa6, + 0x25, 0xee, 0x3e, 0x34, 0xb9, 0xa0, 0xfe, 0x5b, 0x4d, 0xa4, 0xe8, 0xb1, 0x4f, 0x8f, 0x74, 0x2a, + 0xe2, 0x77, 0xca, 0xde, 0xeb, 0x3b, 0xfd, 0xaa, 0x08, 0x93, 0x94, 0x8f, 0x94, 0xd0, 0x01, 0xf6, + 0x7a, 0x96, 0x63, 0xd8, 0xf4, 0x75, 0x0a, 0x6c, 0xdf, 0x0a, 0x4c, 0xde, 0xb7, 0x02, 0x43, 0x3b, + 0x30, 0x1b, 0xb7, 0xfd, 0xa8, 0x9a, 0xf4, 0xfb, 0x6f, 0xef, 0xaa, 0x4c, 0xac, 0xb1, 0x9f, 0x90, + 0x54, 0x2f, 0x08, 0x24, 0x88, 0xc8, 0x84, 0x99, 0x8e, 0xeb, 0x04, 0x86, 0xe5, 0x60, 0x8f, 0x19, + 0xca, 0xa6, 0xdd, 0xff, 0x39, 0xa7, 0xf0, 0xb0, 0xee, 0x89, 0x2a, 0xa7, 0xde, 0xff, 0x51, 0x69, + 0xe8, 0x23, 0x98, 0x16, 0x65, 0x0c, 0x33, 0x32, 0x91, 0x76, 0xff, 0x67, 0x55, 0x66, 0x61, 0x4b, + 0x5a, 0x91, 0x52, 0xef, 0xff, 0x28, 0x24, 0x64, 0x43, 0xb9, 0xef, 0x9a, 0x5b, 0x0e, 0x4f, 0xde, + 0x8d, 0xb6, 0x8d, 0x79, 0xaf, 0x79, 0x69, 0x24, 0xe5, 0x51, 0xb8, 0x58, 0x28, 0x4e, 0xca, 0xaa, + 0x37, 0xea, 0x92, 0x54, 0xf4, 0x21, 0x4c, 0xd9, 0xa4, 0x9a, 0x5c, 0xdd, 0xef, 0x5b, 0x1e, 0x36, + 0xd3, 0xef, 0xbf, 0x5d, 0x96, 0x38, 0x58, 0x20, 0x94, 0x65, 0xd4, 0x3b, 0x40, 0x32, 0x85, 0xcc, + 0x7e, 0xcf, 0xd8, 0x6f, 0x86, 0x8e, 0xbf, 0xba, 0xcf, 0xef, 0x32, 0xe5, 0xd3, 0x66, 0x7f, 0x5d, + 0x65, 0x62, 0xb3, 0x9f, 0x90, 0x54, 0x67, 0x3f, 0x41, 0x44, 0x97, 0x69, 0x9c, 0x67, 0x53, 0xc2, + 0xee, 0xc1, 0x2d, 0x8c, 0x8c, 0x16, 0x9b, 0x0d, 0xd6, 0xf6, 0xe1, 0x4f, 0x8a, 0xd2, 0x48, 0x03, + 0x9f, 0x03, 0xfa, 0xda, 0x4d, 0x1c, 0x84, 0x9e, 0x83, 0x4d, 0x5e, 0x54, 0x8d, 0xce, 0x81, 0xc2, + 0x15, 0xcd, 0x81, 0x82, 0x8e, 0xcc, 0x81, 0x42, 0x25, 0x6b, 0xaa, 0xef, 0x9a, 0xd7, 0xd8, 0x96, + 0x09, 0xa2, 0x8b, 0x71, 0x4f, 0x8e, 0x98, 0x8a, 0x59, 0xd8, 0x9a, 0x52, 0xa4, 0xd4, 0x35, 0xa5, + 0x90, 0xf8, 0x5d, 0x2c, 0xf9, 0xe6, 0x0e, 0x1b, 0xa9, 0xd2, 0x98, 0xbb, 0x58, 0x23, 0x9c, 0xd1, + 0x5d, 0xac, 0x11, 0xca, 0xc8, 0x5d, 0xac, 0x11, 0x0e, 0x62, 0xbd, 0x6b, 0x38, 0xdd, 0x35, 0xb7, + 0xad, 0xae, 0xea, 0xa9, 0x34, 0xeb, 0x17, 0x52, 0x38, 0x99, 0xf5, 0x34, 0x1d, 0xaa, 0xf5, 0x34, + 0x8e, 0x46, 0x41, 0xb4, 0x87, 0xf4, 0x2f, 0x35, 0x98, 0x4d, 0xc4, 0x19, 0xf4, 0x16, 0x44, 0x37, + 0x4e, 0xae, 0x1d, 0xf4, 0x45, 0x9a, 0xac, 0xdc, 0x50, 0x21, 0x78, 0xda, 0x0d, 0x15, 0x82, 0xa3, + 0xcb, 0x00, 0xd1, 0x37, 0xe9, 0x76, 0x41, 0x9a, 0xe6, 0x68, 0x31, 0xa7, 0x9c, 0xa3, 0xc5, 0xa8, + 0xfe, 0x4d, 0x16, 0x0a, 0x62, 0xa1, 0x3e, 0x94, 0x32, 0xaa, 0x0e, 0xf9, 0x1e, 0xf6, 0xe9, 0x4d, + 0x95, 0x4c, 0x9c, 0x0d, 0x71, 0x48, 0xce, 0x86, 0x38, 0xa4, 0x26, 0x6b, 0xd9, 0x7b, 0x4a, 0xd6, + 0x26, 0x0e, 0x9d, 0xac, 0x61, 0x7a, 0x4a, 0x2d, 0x85, 0x5b, 0x71, 0x2e, 0x74, 0xfb, 0x18, 0x2e, + 0xce, 0xb0, 0x65, 0xc1, 0xc4, 0x19, 0xb6, 0x4c, 0x42, 0xbb, 0x70, 0x54, 0x3a, 0xbb, 0xe2, 0xbd, + 0x43, 0x12, 0xf8, 0x66, 0xc6, 0x5f, 0x09, 0x68, 0x52, 0x2e, 0xb6, 0xbd, 0x77, 0x13, 0xa8, 0x9c, + 0xed, 0x26, 0x69, 0xfa, 0x1f, 0x33, 0x30, 0xa3, 0xfa, 0xfb, 0x50, 0x26, 0xf6, 0x25, 0x28, 0xe2, + 0x7d, 0x2b, 0x68, 0x75, 0x5c, 0x13, 0xf3, 0x92, 0x91, 0xce, 0x13, 0x01, 0xcf, 0xb9, 0xa6, 0x32, + 0x4f, 0x02, 0x93, 0x57, 0x43, 0xf6, 0x50, 0xab, 0x21, 0x6e, 0xb5, 0x4e, 0xdc, 0xb9, 0xd5, 0x9a, + 0x3e, 0xce, 0xc5, 0x87, 0x34, 0xce, 0xb7, 0x32, 0x50, 0x4e, 0x46, 0xe3, 0xef, 0xc7, 0x16, 0x52, + 0x77, 0x43, 0xf6, 0xd0, 0xbb, 0xe1, 0x6d, 0x98, 0x26, 0xb9, 0xa3, 0x11, 0x04, 0xfc, 0x0e, 0xe7, + 0x04, 0xcd, 0xb9, 0x58, 0x6c, 0x0a, 0x9d, 0x15, 0x81, 0x2b, 0xb1, 0x49, 0xc2, 0xf5, 0x4f, 0x33, + 0x30, 0xad, 0x7c, 0x35, 0x1e, 0xbf, 0x90, 0xa2, 0xcf, 0xc2, 0xb4, 0x92, 0x8c, 0xe9, 0xff, 0xc7, + 0xd6, 0x89, 0x9a, 0x05, 0x3d, 0x7e, 0xe3, 0x32, 0x03, 0x53, 0x72, 0x56, 0xa7, 0x37, 0x60, 0x36, + 0x91, 0x84, 0xc9, 0x2f, 0xa0, 0x1d, 0xe6, 0x05, 0xf4, 0x05, 0x98, 0x4b, 0xcb, 0x1d, 0xf4, 0x0b, + 0x30, 0x97, 0xf6, 0x55, 0xbf, 0x7b, 0x03, 0x5f, 0x69, 0xd4, 0xc2, 0xe8, 0x6d, 0xef, 0x8b, 0x00, + 0x0e, 0xbe, 0xd9, 0xba, 0x63, 0xf9, 0xc7, 0xc6, 0x13, 0xdf, 0x5c, 0x4b, 0x54, 0x4b, 0x05, 0x81, + 0x11, 0x4d, 0xae, 0x6d, 0xb6, 0xee, 0x58, 0x74, 0x51, 0x4d, 0xae, 0x6d, 0x8e, 0x68, 0x12, 0x98, + 0xfe, 0x83, 0xac, 0xa8, 0xcc, 0xe3, 0xeb, 0xd2, 0x1f, 0x40, 0xb9, 0x2f, 0x1e, 0xee, 0xec, 0x2d, + 0xad, 0x4d, 0x22, 0xfe, 0xa4, 0xa5, 0x19, 0x95, 0xa2, 0xea, 0xe6, 0x45, 0x67, 0xe6, 0x90, 0xba, + 0x9b, 0x89, 0xea, 0x73, 0x46, 0xa5, 0xa0, 0xff, 0x86, 0xa3, 0xe2, 0x36, 0xd9, 0x1e, 0x16, 0x8e, + 0x67, 0xc7, 0x2a, 0x67, 0xb7, 0xbb, 0x23, 0x81, 0xa4, 0xe7, 0xb3, 0x09, 0x52, 0x42, 0x3d, 0xf7, + 0x7d, 0xe2, 0xb0, 0xea, 0x93, 0xce, 0xcf, 0x26, 0x48, 0xfa, 0xe7, 0x1a, 0xcc, 0x26, 0x2e, 0xa0, + 0xa3, 0xf3, 0x50, 0xa0, 0xbf, 0x4f, 0xbb, 0xfd, 0x0c, 0xd0, 0x05, 0x49, 0xf9, 0x14, 0x0b, 0x79, + 0x0e, 0xa1, 0x57, 0xa0, 0x18, 0xdd, 0x53, 0xe7, 0xa7, 0xca, 0x6c, 0xf3, 0x09, 0x50, 0xd9, 0x7c, + 0x02, 0xd4, 0x7f, 0xa2, 0xc1, 0xf1, 0xb1, 0x97, 0xd3, 0x1f, 0x75, 0xcf, 0xe0, 0xd9, 0x17, 0xa0, + 0x20, 0xce, 0x7d, 0x11, 0x40, 0xee, 0xbd, 0xad, 0xd5, 0xad, 0xd5, 0xf3, 0xe5, 0x23, 0xa8, 0x04, + 0xf9, 0x8d, 0xd5, 0x2b, 0xe7, 0x2f, 0x5d, 0xb9, 0x50, 0xd6, 0xc8, 0x43, 0x73, 0xeb, 0xca, 0x15, + 0xf2, 0x90, 0x79, 0xf6, 0xb2, 0x7c, 0x0b, 0x8d, 0x7d, 0x8f, 0xd1, 0x14, 0x14, 0x56, 0xfa, 0x7d, + 0x1a, 0x00, 0x98, 0xec, 0xea, 0x9e, 0x45, 0xf6, 0x6a, 0x59, 0x43, 0x79, 0xc8, 0x5e, 0xbd, 0xba, + 0x5e, 0xce, 0xa0, 0x39, 0x28, 0x9f, 0xc7, 0x86, 0x69, 0x5b, 0x0e, 0x16, 0x51, 0xa7, 0x9c, 0x6d, + 0xdc, 0xf8, 0xfa, 0xdb, 0x25, 0xed, 0x9b, 0x6f, 0x97, 0xb4, 0x3f, 0x7c, 0xbb, 0xa4, 0xdd, 0xfa, + 0x6e, 0xe9, 0xc8, 0x37, 0xdf, 0x2d, 0x1d, 0xf9, 0xdd, 0x77, 0x4b, 0x47, 0x3e, 0x78, 0x41, 0xfa, + 0x2d, 0x26, 0x7b, 0xa7, 0xbe, 0xe7, 0x92, 0x80, 0xcb, 0x9f, 0xea, 0xc9, 0x5f, 0x9f, 0x7e, 0x95, + 0x39, 0xb9, 0x42, 0x1f, 0x37, 0x18, 0x5f, 0xed, 0x92, 0x5b, 0x63, 0x00, 0xfd, 0x01, 0xa1, 0xdf, + 0xce, 0xd1, 0x1f, 0x0a, 0xbe, 0xf4, 0x8f, 0x00, 0x00, 0x00, 0xff, 0xff, 0xdf, 0x71, 0x86, 0xa5, + 0xb8, 0x3a, 0x00, 0x00, } func (m *EventSequence) Marshal() (dAtA []byte, err error) { diff --git a/pkg/armadaevents/events.proto b/pkg/armadaevents/events.proto index f78e8bb5839..d68759a6d75 100644 --- a/pkg/armadaevents/events.proto +++ b/pkg/armadaevents/events.proto @@ -6,10 +6,10 @@ option csharp_namespace = "ArmadaProject.Io.ArmadaEvents"; import "k8s.io/api/core/v1/generated.proto"; import "k8s.io/api/networking/v1/generated.proto"; import "google/protobuf/timestamp.proto"; -import "google/protobuf/wrappers.proto"; import "k8s.io/apimachinery/pkg/api/resource/generated.proto"; import "github.com/gogo/protobuf/gogoproto/gogo.proto"; import "internal/scheduler/schedulerobjects/schedulerobjects.proto"; + // Armada state transition model // // The central concepts are: