From 0b4186d4482766ad749b4eba58838358727e5601 Mon Sep 17 00:00:00 2001 From: Future-Outlier Date: Wed, 11 Oct 2023 01:20:37 +0800 Subject: [PATCH 01/18] Flyte Add Enable/Disable Options for Agent Services (#4041) * enable agent Signed-off-by: Future Outlier * update flyteagent kustomization Signed-off-by: Future Outlier * add agent-servie key in enabled plugins Signed-off-by: Future Outlier * add flyte agent patch Signed-off-by: Future Outlier * Update helm chart and remove patch Signed-off-by: Future Outlier * add agentService Signed-off-by: Future Outlier * add task timeout Signed-off-by: Future Outlier * make helm Signed-off-by: Future Outlier * update helm README.md Signed-off-by: Future Outlier * checksum Signed-off-by: Future Outlier * update supported task type Signed-off-by: Future Outlier * update key value Signed-off-by: Future Outlier --------- Signed-off-by: Future Outlier Signed-off-by: Future-Outlier Co-authored-by: Future Outlier --- charts/flyte-binary/README.md | 12 +- charts/flyte-binary/values.yaml | 16 +- .../flyte_sandbox_binary_helm_generated.yaml | 12 +- docker/sandbox-bundled/Makefile | 5 + .../bootstrap/cmd/bootstrap/main.go | 14 +- .../complete-agent/kustomization.yaml | 12 + .../manifests/complete-agent.yaml | 1912 +++++++++++++++++ .../sandbox-bundled/manifests/complete.yaml | 16 +- docker/sandbox-bundled/manifests/dev.yaml | 4 +- 9 files changed, 1983 insertions(+), 20 deletions(-) create mode 100644 docker/sandbox-bundled/kustomize/complete-agent/kustomization.yaml create mode 100644 docker/sandbox-bundled/manifests/complete-agent.yaml diff --git a/charts/flyte-binary/README.md b/charts/flyte-binary/README.md index 3ea5cd2f6cd..d6d9a3bb6a8 100644 --- a/charts/flyte-binary/README.md +++ b/charts/flyte-binary/README.md @@ -21,7 +21,11 @@ Chart for basic single Flyte executable deployment | clusterResourceTemplates.labels | object | `{}` | | | commonAnnotations | object | `{}` | | | commonLabels | object | `{}` | | -| configuration.agentService | object | `{}` | | +| configuration.agentService.defaultAgent.defaultTimeout | string | `"10s"` | | +| configuration.agentService.defaultAgent.endpoint | string | `"dns:///flyteagent.flyte.svc.cluster.local:8000"` | | +| configuration.agentService.defaultAgent.insecure | bool | `true` | | +| configuration.agentService.defaultAgent.timeouts.GetTask | string | `"10s"` | | +| configuration.agentService.supportedTaskTypes[0] | string | `"default_task"` | | | configuration.annotations | object | `{}` | | | configuration.auth.authorizedUris | list | `[]` | | | configuration.auth.clientSecretsExternalSecretRef | string | `""` | | @@ -103,9 +107,9 @@ Chart for basic single Flyte executable deployment | deployment.waitForDB.image.pullPolicy | string | `"IfNotPresent"` | | | deployment.waitForDB.image.repository | string | `"postgres"` | | | deployment.waitForDB.image.tag | string | `"15-alpine"` | | -| enabled_plugins.tasks | object | `{"task-plugins":{"default-for-task-types":{"container":"container","container_array":"k8s-array","sidecar":"sidecar"},"enabled-plugins":["container","sidecar","k8s-array"]}}` | Tasks specific configuration [structure](https://pkg.go.dev/github.com/flyteorg/flytepropeller/pkg/controller/nodes/task/config#GetConfig) | -| enabled_plugins.tasks.task-plugins | object | `{"default-for-task-types":{"container":"container","container_array":"k8s-array","sidecar":"sidecar"},"enabled-plugins":["container","sidecar","k8s-array"]}` | Plugins configuration, [structure](https://pkg.go.dev/github.com/flyteorg/flytepropeller/pkg/controller/nodes/task/config#TaskPluginConfig) | -| enabled_plugins.tasks.task-plugins.enabled-plugins | list | `["container","sidecar","k8s-array"]` | [Enabled Plugins](https://pkg.go.dev/github.com/lyft/flyteplugins/go/tasks/config#Config). Enable sagemaker*, athena if you install the backend plugins | +| enabled_plugins.tasks | object | `{"task-plugins":{"default-for-task-types":{"container":"container","container_array":"k8s-array","sidecar":"sidecar"},"enabled-plugins":["container","sidecar","k8s-array","agent-service"]}}` | Tasks specific configuration [structure](https://pkg.go.dev/github.com/flyteorg/flytepropeller/pkg/controller/nodes/task/config#GetConfig) | +| enabled_plugins.tasks.task-plugins | object | `{"default-for-task-types":{"container":"container","container_array":"k8s-array","sidecar":"sidecar"},"enabled-plugins":["container","sidecar","k8s-array","agent-service"]}` | Plugins configuration, [structure](https://pkg.go.dev/github.com/flyteorg/flytepropeller/pkg/controller/nodes/task/config#TaskPluginConfig) | +| enabled_plugins.tasks.task-plugins.enabled-plugins | list | `["container","sidecar","k8s-array","agent-service"]` | [Enabled Plugins](https://pkg.go.dev/github.com/lyft/flyteplugins/go/tasks/config#Config). Enable sagemaker*, athena if you install the backend plugins | | flyte-core-components.admin.disableClusterResourceManager | bool | `false` | | | flyte-core-components.admin.disableScheduler | bool | `false` | | | flyte-core-components.admin.disabled | bool | `false` | | diff --git a/charts/flyte-binary/values.yaml b/charts/flyte-binary/values.yaml index 5e555ac8909..84ffe00112e 100644 --- a/charts/flyte-binary/values.yaml +++ b/charts/flyte-binary/values.yaml @@ -151,11 +151,16 @@ configuration: # tag CoPilot sidecar image tag tag: v1.9.4 # FLYTECOPILOT_TAG # agentService Flyte Agent configuration - agentService: {} + agentService: + defaultAgent: + endpoint: "dns:///flyteagent.flyte.svc.cluster.local:8000" + insecure: true + timeouts: + GetTask: 10s + defaultTimeout: 10s # Uncomment and modify to include configuration for Flyte Agent - # defaultGrpcEndpoint: agent-service.agent-namespace:8000 - # supportedTaskTypes: - # - custom_task_type + supportedTaskTypes: + - default_task # externalConfigMap Specify an existing, external ConfigMap to use as configuration for Flyte # If set, no Flyte configuration will be generated by this chart externalConfigMap: "" @@ -382,8 +387,7 @@ enabled_plugins: - container - sidecar - k8s-array - # -- Uncomment to enable agent service - # - agent-service + - agent-service default-for-task-types: container: container sidecar: sidecar diff --git a/deployment/sandbox-binary/flyte_sandbox_binary_helm_generated.yaml b/deployment/sandbox-binary/flyte_sandbox_binary_helm_generated.yaml index 7c1524af060..bb5e0cd675e 100644 --- a/deployment/sandbox-binary/flyte_sandbox_binary_helm_generated.yaml +++ b/deployment/sandbox-binary/flyte_sandbox_binary_helm_generated.yaml @@ -108,6 +108,7 @@ data: - container - sidecar - k8s-array + - agent-service plugins: logs: kubernetes-enabled: false @@ -122,6 +123,15 @@ data: kubernetes-enabled: false cloudwatch-enabled: false stackdriver-enabled: false + agent-service: + defaultAgent: + defaultTimeout: 10s + endpoint: dns:///flyteagent.flyte.svc.cluster.local:8000 + insecure: true + timeouts: + GetTask: 10s + supportedTaskTypes: + - default_task 002-database.yaml: | database: postgres: @@ -357,7 +367,7 @@ spec: app.kubernetes.io/instance: flyte app.kubernetes.io/component: flyte-binary annotations: - checksum/configuration: 528ce4a42638a7810c99802dfd49525967db2a99dbc1019544b7799de2490b61 + checksum/configuration: d220769393e7acbe0372fdccbf3d588797864ec934661f08912e88ec084cdfde checksum/configuration-secret: d5d93f4e67780b21593dc3799f0f6682aab0765e708e4020939975d14d44f929 checksum/cluster-resource-templates: 7dfa59f3d447e9c099b8f8ffad3af466fecbc9cf9f8c97295d9634254a55d4ae spec: diff --git a/docker/sandbox-bundled/Makefile b/docker/sandbox-bundled/Makefile index 709c04caf2a..9ae4197673c 100644 --- a/docker/sandbox-bundled/Makefile +++ b/docker/sandbox-bundled/Makefile @@ -19,6 +19,7 @@ flyte: manifests: mkdir -p manifests helm dependency update ../../charts/flyte-sandbox + helm dependency update ../../charts/flyteagent kustomize build \ --enable-helm \ --load-restrictor=LoadRestrictionsNone \ @@ -27,6 +28,10 @@ manifests: --enable-helm \ --load-restrictor=LoadRestrictionsNone \ kustomize/dev > manifests/dev.yaml + kustomize build \ + --enable-helm \ + --load-restrictor=LoadRestrictionsNone \ + kustomize/complete-agent > manifests/complete-agent.yaml .PHONY: build build: flyte manifests diff --git a/docker/sandbox-bundled/bootstrap/cmd/bootstrap/main.go b/docker/sandbox-bundled/bootstrap/cmd/bootstrap/main.go index 82bae881515..6afee0e82c5 100644 --- a/docker/sandbox-bundled/bootstrap/cmd/bootstrap/main.go +++ b/docker/sandbox-bundled/bootstrap/cmd/bootstrap/main.go @@ -17,13 +17,15 @@ const ( clusterResourceTemplatesConfigMapName = "flyte-sandbox-extra-cluster-resource-templates" deploymentName = "flyte-sandbox" devModeEnvVar = "FLYTE_DEV" + disableAgentModeEnvVar = "DISABLE_AGENT" dockerHost = "host.docker.internal" namespace = "flyte" // Template paths - devTemplatePath = "/var/lib/rancher/k3s/server/manifests-staging/dev.yaml" - fullTemplatePath = "/var/lib/rancher/k3s/server/manifests-staging/complete.yaml" - renderedManifestPath = "/var/lib/rancher/k3s/server/manifests/flyte.yaml" + devTemplatePath = "/var/lib/rancher/k3s/server/manifests-staging/dev.yaml" + fullTemplatePath = "/var/lib/rancher/k3s/server/manifests-staging/complete.yaml" + fullAgentTemplatePath = "/var/lib/rancher/k3s/server/manifests-staging/complete-agent.yaml" + renderedManifestPath = "/var/lib/rancher/k3s/server/manifests/flyte.yaml" ) func main() { @@ -35,7 +37,11 @@ func main() { } else { // If we are not running in dev mode, look for user-specified configuration // to load into the sandbox deployment - tmplPath = fullTemplatePath + tmplPath = fullAgentTemplatePath + if os.Getenv(disableAgentModeEnvVar) == "True" { + tmplPath = fullTemplatePath + } + cOpts := config.LoaderOpts{ ConfigurationConfigMapName: configurationConfigMapName, ClusterResourceTemplatesConfigMapName: clusterResourceTemplatesConfigMapName, diff --git a/docker/sandbox-bundled/kustomize/complete-agent/kustomization.yaml b/docker/sandbox-bundled/kustomize/complete-agent/kustomization.yaml new file mode 100644 index 00000000000..3c6d5c6e5e5 --- /dev/null +++ b/docker/sandbox-bundled/kustomize/complete-agent/kustomization.yaml @@ -0,0 +1,12 @@ +helmGlobals: + chartHome: ../../../../charts +helmCharts: +- name: flyte-sandbox + releaseName: flyte-sandbox + namespace: flyte +- name: flyteagent + releaseName: flyteagent + namespace: flyte +namespace: flyte +resources: +- ../namespace.yaml diff --git a/docker/sandbox-bundled/manifests/complete-agent.yaml b/docker/sandbox-bundled/manifests/complete-agent.yaml new file mode 100644 index 00000000000..b18c5ac84e5 --- /dev/null +++ b/docker/sandbox-bundled/manifests/complete-agent.yaml @@ -0,0 +1,1912 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: flyte +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox + namespace: flyte +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +--- +apiVersion: v1 +automountServiceAccountToken: true +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: minio + helm.sh/chart: minio-12.1.1 + name: flyte-sandbox-minio + namespace: flyte +secrets: +- name: flyte-sandbox-minio +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyteagent + helm.sh/chart: flyteagent-v0.1.10 + name: flyteagent + namespace: flyte +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +rules: +- apiGroups: + - "" + resourceNames: + - kubernetes-dashboard-key-holder + - kubernetes-dashboard-certs + - kubernetes-dashboard-csrf + resources: + - secrets + verbs: + - get + - update + - delete +- apiGroups: + - "" + resourceNames: + - kubernetes-dashboard-settings + resources: + - configmaps + verbs: + - get + - update +- apiGroups: + - "" + resourceNames: + - heapster + - dashboard-metrics-scraper + resources: + - services + verbs: + - proxy +- apiGroups: + - "" + resourceNames: + - heapster + - 'http:heapster:' + - 'https:heapster:' + - dashboard-metrics-scraper + - http:dashboard-metrics-scraper + resources: + - services/proxy + verbs: + - get +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-cluster-role + namespace: flyte +rules: +- apiGroups: + - "" + resources: + - namespaces + - resourcequotas + - secrets + verbs: + - create + - get + - list + - patch + - update +- apiGroups: + - "" + resources: + - pods + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - "" + resources: + - events + verbs: + - create + - delete + - patch + - update +- apiGroups: + - "" + resources: + - podtemplates + verbs: + - get + - list + - watch +- apiGroups: + - flyte.lyft.com + resources: + - flyteworkflows + verbs: + - create + - delete + - deletecollection + - get + - list + - patch + - post + - update + - watch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - create + - get + - list +- apiGroups: + - admissionregistration.k8s.io + resources: + - mutatingwebhookconfigurations + verbs: + - create + - get + - list + - patch + - update +- apiGroups: + - '*' + resources: + - '*' + verbs: + - '*' +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard-readonly +rules: +- apiGroups: + - "" + resources: + - configmaps + - endpoints + - persistentvolumeclaims + - pods + - replicationcontrollers + - replicationcontrollers/scale + - serviceaccounts + - services + - nodes + - persistentvolumeclaims + - persistentvolumes + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - bindings + - events + - limitranges + - namespaces/status + - pods/log + - pods/status + - replicationcontrollers/status + - resourcequotas + - resourcequotas/status + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - apps + resources: + - daemonsets + - deployments + - deployments/scale + - replicasets + - replicasets/scale + - statefulsets + verbs: + - get + - list + - watch +- apiGroups: + - autoscaling + resources: + - horizontalpodautoscalers + verbs: + - get + - list + - watch +- apiGroups: + - batch + resources: + - cronjobs + - jobs + verbs: + - get + - list + - watch +- apiGroups: + - extensions + resources: + - daemonsets + - deployments + - deployments/scale + - ingresses + - networkpolicies + - replicasets + - replicasets/scale + - replicationcontrollers/scale + verbs: + - get + - list + - watch +- apiGroups: + - policy + resources: + - poddisruptionbudgets + verbs: + - get + - list + - watch +- apiGroups: + - networking.k8s.io + resources: + - networkpolicies + - ingresses + verbs: + - get + - list + - watch +- apiGroups: + - storage.k8s.io + resources: + - storageclasses + - volumeattachments + verbs: + - get + - list + - watch +- apiGroups: + - rbac.authorization.k8s.io + resources: + - clusterrolebindings + - clusterroles + - roles + - rolebindings + verbs: + - get + - list + - watch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: flyte-sandbox-kubernetes-dashboard +subjects: +- kind: ServiceAccount + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-cluster-role-binding + namespace: flyte +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: flyte-sandbox-cluster-role +subjects: +- kind: ServiceAccount + name: flyte-sandbox + namespace: flyte +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard-readonly +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: flyte-sandbox-kubernetes-dashboard-readonly +subjects: +- kind: ServiceAccount + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +--- +apiVersion: v1 +data: + namespace.yaml: | + apiVersion: v1 + kind: Namespace + metadata: + name: '{{ namespace }}' +kind: ConfigMap +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-cluster-resource-templates + namespace: flyte +--- +apiVersion: v1 +data: + 000-core.yaml: | + admin: + endpoint: localhost:8089 + insecure: true + catalog-cache: + endpoint: localhost:8081 + insecure: true + type: datacatalog + cluster_resources: + standaloneDeployment: false + templatePath: /etc/flyte/cluster-resource-templates + logger: + show-source: true + level: 6 + propeller: + create-flyteworkflow-crd: true + webhook: + certDir: /var/run/flyte/certs + localCert: true + secretName: flyte-sandbox-webhook-secret + serviceName: flyte-sandbox-webhook + servicePort: 443 + flyte: + admin: + disableClusterResourceManager: false + disableScheduler: false + disabled: false + seedProjects: + - flytesnacks + dataCatalog: + disabled: false + propeller: + disableWebhook: false + disabled: false + 001-plugins.yaml: | + tasks: + task-plugins: + default-for-task-types: + container: container + container_array: k8s-array + sidecar: sidecar + enabled-plugins: + - container + - sidecar + - k8s-array + - agent-service + plugins: + logs: + kubernetes-enabled: true + kubernetes-template-uri: http://localhost:30080/kubernetes-dashboard/#/log/{{.namespace }}/{{ .podName }}/pod?namespace={{ .namespace }} + cloudwatch-enabled: false + stackdriver-enabled: false + k8s: + co-pilot: + image: "cr.flyte.org/flyteorg/flytecopilot:v1.9.4" + k8s-array: + logs: + config: + kubernetes-enabled: true + kubernetes-template-uri: http://localhost:30080/kubernetes-dashboard/#/log/{{.namespace }}/{{ .podName }}/pod?namespace={{ .namespace }} + cloudwatch-enabled: false + stackdriver-enabled: false + agent-service: + defaultAgent: + defaultTimeout: 10s + endpoint: dns:///flyteagent.flyte.svc.cluster.local:8000 + insecure: true + timeouts: + GetTask: 10s + supportedTaskTypes: + - default_task + 002-database.yaml: | + database: + postgres: + username: postgres + host: flyte-sandbox-postgresql + port: 5432 + dbname: flyte + options: "sslmode=disable" + 003-storage.yaml: | + propeller: + rawoutput-prefix: s3://my-s3-bucket/data + storage: + type: stow + stow: + kind: s3 + config: + region: us-east-1 + disable_ssl: true + v2_signing: true + endpoint: http://flyte-sandbox-minio.flyte:9000 + auth_type: accesskey + container: my-s3-bucket + 100-inline-config.yaml: | + plugins: + k8s: + default-env-vars: + - FLYTE_AWS_ENDPOINT: http://flyte-sandbox-minio.flyte:9000 + - FLYTE_AWS_ACCESS_KEY_ID: minio + - FLYTE_AWS_SECRET_ACCESS_KEY: miniostorage + storage: + signedURL: + stowConfigOverride: + endpoint: http://localhost:30002 + task_resources: + defaults: + cpu: 500m + ephemeralStorage: 0 + gpu: 0 + memory: 1Gi + limits: + cpu: 0 + ephemeralStorage: 0 + gpu: 0 + memory: 0 +kind: ConfigMap +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-config + namespace: flyte +--- +apiVersion: v1 +data: + config.yml: |- + health: + storagedriver: + enabled: true + interval: 10s + threshold: 3 + http: + addr: :5000 + debug: + addr: :5001 + prometheus: + enabled: false + path: /metrics + headers: + X-Content-Type-Options: + - nosniff + log: + fields: + service: registry + storage: + cache: + blobdescriptor: inmemory + version: 0.1 +kind: ConfigMap +metadata: + labels: + app: docker-registry + chart: docker-registry-2.2.2 + heritage: Helm + release: flyte-sandbox + name: flyte-sandbox-docker-registry-config + namespace: flyte +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: flyte-sandbox-extra-cluster-resource-templates + namespace: flyte +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: flyte-sandbox-extra-config + namespace: flyte +--- +apiVersion: v1 +data: + envoy.yaml: | + admin: + access_log_path: /dev/stdout + static_resources: + listeners: + - address: + socket_address: + address: 0.0.0.0 + port_value: 8000 + filter_chains: + - filters: + - name: envoy.filters.network.http_connection_manager + typed_config: + "@type": type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager + stat_prefix: ingress_http + codec_type: AUTO + upgrade_configs: + - upgrade_type: websocket + route_config: + name: local_route + virtual_hosts: + - name: backend + domains: + - "*" + routes: + - match: + path: "/" + redirect: + path_redirect: "/console/" + - match: + prefix: "/.well-known" + route: + cluster: flyte + - match: + prefix: "/__webpack_hmr" + route: + cluster: flyte + - match: + prefix: "/api" + route: + cluster: flyte + - match: + prefix: "/callback" + route: + cluster: flyte + - match: + prefix: "/config" + route: + cluster: flyte + - match: + prefix: "/console" + route: + cluster: flyte + - match: + prefix: "/healthcheck" + route: + cluster: flyte + - match: + prefix: "/login" + route: + cluster: flyte + - match: + prefix: "/logout" + route: + cluster: flyte + - match: + prefix: "/me" + route: + cluster: flyte + - match: + prefix: "/oauth2" + route: + cluster: flyte + - match: + prefix: "/v1" + route: + cluster: flyte + - match: + prefix: "/flyteidl.service.AdminService" + route: + cluster: flyte_grpc + - match: + prefix: "/flyteidl.service.AuthMetadataService" + route: + cluster: flyte_grpc + - match: + prefix: "/flyteidl.service.DataProxyService" + route: + cluster: flyte_grpc + - match: + prefix: "/flyteidl.service.IdentityService" + route: + cluster: flyte_grpc + - match: + prefix: "/grpc.health.v1.Health" + route: + cluster: flyte_grpc + - match: + prefix: "/flyteidl.service.SignalService" + route: + cluster: flyte_grpc + - match: + path: "/kubernetes-dashboard" + redirect: + path_redirect: "/kubernetes-dashboard/" + - match: + prefix: "/kubernetes-dashboard/" + route: + cluster: kubernetes-dashboard + prefix_rewrite: / + - match: + path: "/minio" + redirect: + path_redirect: "/minio/" + - match: + prefix: "/minio/" + route: + cluster: minio + prefix_rewrite: / + http_filters: + - name: envoy.filters.http.router + typed_config: + "@type": type.googleapis.com/envoy.extensions.filters.http.router.v3.Router + clusters: + - name: flyte + connect_timeout: 0.25s + type: STRICT_DNS + lb_policy: ROUND_ROBIN + load_assignment: + cluster_name: flyte + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: flyte-sandbox-http + port_value: 8088 + - name: flyte_grpc + connect_timeout: 0.25s + type: STRICT_DNS + lb_policy: ROUND_ROBIN + http2_protocol_options: {} + load_assignment: + cluster_name: flyte_grpc + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: flyte-sandbox-grpc + port_value: 8089 + - name: kubernetes-dashboard + connect_timeout: 0.25s + type: STRICT_DNS + lb_policy: ROUND_ROBIN + load_assignment: + cluster_name: kubernetes-dashboard + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: flyte-sandbox-kubernetes-dashboard + port_value: 80 + - name: minio + connect_timeout: 0.25s + type: STRICT_DNS + lb_policy: ROUND_ROBIN + load_assignment: + cluster_name: minio + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: flyte-sandbox-minio + port_value: 9001 +kind: ConfigMap +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-proxy-config + namespace: flyte +--- +apiVersion: v1 +data: null +kind: ConfigMap +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: kubernetes-dashboard-settings + namespace: flyte +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-config-secret + namespace: flyte +stringData: + 012-database-secrets.yaml: | + database: + postgres: + password: "postgres" + 013-storage-secrets.yaml: | + storage: + stow: + config: + access_key_id: "minio" + secret_key: "miniostorage" +type: Opaque +--- +apiVersion: v1 +data: + haSharedSecret: R2JRWFVRYThnRFVLbHpuSA== + proxyPassword: "" + proxyUsername: "" +kind: Secret +metadata: + labels: + app: docker-registry + chart: docker-registry-2.2.2 + heritage: Helm + release: flyte-sandbox + name: flyte-sandbox-docker-registry-secret + namespace: flyte +type: Opaque +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard-certs + namespace: flyte +type: Opaque +--- +apiVersion: v1 +data: + root-password: bWluaW9zdG9yYWdl + root-user: bWluaW8= +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: minio + helm.sh/chart: minio-12.1.1 + name: flyte-sandbox-minio + namespace: flyte +type: Opaque +--- +apiVersion: v1 +data: + postgres-password: cG9zdGdyZXM= +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: postgresql + helm.sh/chart: postgresql-12.1.9 + name: flyte-sandbox-postgresql + namespace: flyte +type: Opaque +--- +apiVersion: v1 +data: + username: User +kind: Secret +metadata: + name: flyteagent + namespace: flyte +type: Opaque +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: kubernetes-dashboard-csrf + namespace: flyte +type: Opaque +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: kubernetes-dashboard-key-holder + namespace: flyte +type: Opaque +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app: docker-registry + chart: docker-registry-2.2.2 + heritage: Helm + release: flyte-sandbox + name: flyte-sandbox-docker-registry + namespace: flyte +spec: + ports: + - name: http-5000 + nodePort: 30000 + port: 5000 + protocol: TCP + targetPort: 5000 + selector: + app: docker-registry + release: flyte-sandbox + type: NodePort +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-grpc + namespace: flyte +spec: + ports: + - name: grpc + nodePort: null + port: 8089 + targetPort: grpc + selector: + app.kubernetes.io/component: flyte-binary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + type: ClusterIP +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-http + namespace: flyte +spec: + ports: + - name: http + nodePort: null + port: 8088 + targetPort: http + selector: + app.kubernetes.io/component: flyte-binary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + type: ClusterIP +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/component: kubernetes-dashboard + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + kubernetes.io/cluster-service: "true" + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +spec: + ports: + - name: http + port: 80 + targetPort: http + selector: + app.kubernetes.io/component: kubernetes-dashboard + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: kubernetes-dashboard + type: ClusterIP +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: minio + helm.sh/chart: minio-12.1.1 + name: flyte-sandbox-minio + namespace: flyte +spec: + externalTrafficPolicy: Cluster + ports: + - name: minio-api + nodePort: 30002 + port: 9000 + targetPort: minio-api + - name: minio-console + port: 9001 + targetPort: minio-console + selector: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: minio + type: NodePort +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: postgresql + helm.sh/chart: postgresql-12.1.9 + name: flyte-sandbox-postgresql + namespace: flyte +spec: + externalTrafficPolicy: Cluster + ports: + - name: tcp-postgresql + nodePort: 30001 + port: 5432 + targetPort: tcp-postgresql + selector: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: postgresql + sessionAffinity: None + type: NodePort +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: postgresql + helm.sh/chart: postgresql-12.1.9 + service.alpha.kubernetes.io/tolerate-unready-endpoints: "true" + name: flyte-sandbox-postgresql-hl + namespace: flyte +spec: + clusterIP: None + ports: + - name: tcp-postgresql + port: 5432 + targetPort: tcp-postgresql + publishNotReadyAddresses: true + selector: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: postgresql + type: ClusterIP +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-proxy + namespace: flyte +spec: + ports: + - name: http + nodePort: 30080 + port: 8000 + protocol: TCP + selector: + app.kubernetes.io/component: proxy + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + type: NodePort +--- +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox-webhook + namespace: flyte +spec: + ports: + - name: webhook + port: 443 + targetPort: webhook + selector: + app.kubernetes.io/component: flyte-binary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + type: ClusterIP +--- +apiVersion: v1 +kind: Service +metadata: + annotations: + projectcontour.io/upstream-protocol.h2c: grpc + labels: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyteagent + helm.sh/chart: flyteagent-v0.1.10 + name: flyteagent + namespace: flyte +spec: + ports: + - name: agent-grpc + port: 8000 + protocol: TCP + targetPort: agent-grpc + selector: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/name: flyteagent + type: ClusterIP +--- +apiVersion: v1 +kind: PersistentVolume +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-db-storage + namespace: flyte +spec: + accessModes: + - ReadWriteOnce + capacity: + storage: 1Gi + hostPath: + path: /var/lib/flyte/storage/db + storageClassName: manual +--- +apiVersion: v1 +kind: PersistentVolume +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-minio-storage + namespace: flyte +spec: + accessModes: + - ReadWriteOnce + capacity: + storage: 1Gi + hostPath: + path: /var/lib/flyte/storage/minio + storageClassName: manual +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-db-storage + namespace: flyte +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi + storageClassName: manual + volumeName: flyte-sandbox-db-storage +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-minio-storage + namespace: flyte +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi + storageClassName: manual + volumeName: flyte-sandbox-minio-storage +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-binary-v0.1.10 + name: flyte-sandbox + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/component: flyte-binary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + strategy: + type: Recreate + template: + metadata: + annotations: + checksum/cluster-resource-templates: 6fd9b172465e3089fcc59f738b92b8dc4d8939360c19de8ee65f68b0e7422035 + checksum/configuration: b765a68950c83acd0c069dac2a6569cf2b0f0f76a2760eea3561d1d04d6be831 + checksum/configuration-secret: 09216ffaa3d29e14f88b1f30af580d02a2a5e014de4d750b7f275cc07ed4e914 + labels: + app.kubernetes.io/component: flyte-binary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + spec: + containers: + - args: + - start + - --config + - /etc/flyte/config.d/*.yaml + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + image: flyte-binary:sandbox + imagePullPolicy: Never + livenessProbe: + httpGet: + path: /healthcheck + port: http + name: flyte + ports: + - containerPort: 8088 + name: http + - containerPort: 8089 + name: grpc + - containerPort: 9443 + name: webhook + readinessProbe: + httpGet: + path: /healthcheck + port: http + volumeMounts: + - mountPath: /etc/flyte/cluster-resource-templates + name: cluster-resource-templates + - mountPath: /etc/flyte/config.d + name: config + - mountPath: /var/run/flyte + name: state + initContainers: + - args: + - | + until pg_isready \ + -h flyte-sandbox-postgresql \ + -p 5432 \ + -U postgres + do + echo waiting for database + sleep 0.1 + done + command: + - sh + - -ec + image: bitnami/postgresql:sandbox + imagePullPolicy: Never + name: wait-for-db + serviceAccountName: flyte-sandbox + volumes: + - name: cluster-resource-templates + projected: + sources: + - configMap: + name: flyte-sandbox-cluster-resource-templates + - configMap: + name: flyte-sandbox-extra-cluster-resource-templates + - name: config + projected: + sources: + - configMap: + name: flyte-sandbox-config + - secret: + name: flyte-sandbox-config-secret + - configMap: + name: flyte-sandbox-extra-config + - emptyDir: {} + name: state +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-buildkit + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/component: buildkit + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + template: + metadata: + labels: + app.kubernetes.io/component: buildkit + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + spec: + containers: + - args: + - --addr + - unix:///run/buildkit/buildkitd.sock + - --addr + - tcp://0.0.0.0:30003 + image: moby/buildkit:sandbox + imagePullPolicy: Never + livenessProbe: + exec: + command: + - buildctl + - debug + - workers + initialDelaySeconds: 5 + periodSeconds: 30 + name: buildkit + ports: + - containerPort: 30003 + name: tcp + protocol: TCP + readinessProbe: + exec: + command: + - buildctl + - debug + - workers + initialDelaySeconds: 5 + periodSeconds: 30 + securityContext: + privileged: true + hostNetwork: true +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app: docker-registry + chart: docker-registry-2.2.2 + heritage: Helm + release: flyte-sandbox + name: flyte-sandbox-docker-registry + namespace: flyte +spec: + minReadySeconds: 5 + replicas: 1 + selector: + matchLabels: + app: docker-registry + release: flyte-sandbox + template: + metadata: + annotations: + checksum/config: 8f50e768255a87f078ba8b9879a0c174c3e045ffb46ac8723d2eedbe293c8d81 + checksum/secret: 0ee1553aec7c03152a0a44e7b1a82985795774412a779f7b607a57e59f42c8ef + labels: + app: docker-registry + release: flyte-sandbox + spec: + containers: + - command: + - /bin/registry + - serve + - /etc/docker/registry/config.yml + env: + - name: REGISTRY_HTTP_SECRET + valueFrom: + secretKeyRef: + key: haSharedSecret + name: flyte-sandbox-docker-registry-secret + - name: REGISTRY_STORAGE_FILESYSTEM_ROOTDIRECTORY + value: /var/lib/registry + image: registry:sandbox + imagePullPolicy: Never + livenessProbe: + httpGet: + path: / + port: 5000 + name: docker-registry + ports: + - containerPort: 5000 + readinessProbe: + httpGet: + path: / + port: 5000 + resources: {} + volumeMounts: + - mountPath: /etc/docker/registry + name: flyte-sandbox-docker-registry-config + - mountPath: /var/lib/registry/ + name: data + securityContext: + fsGroup: 1000 + runAsUser: 1000 + volumes: + - configMap: + name: flyte-sandbox-docker-registry-config + name: flyte-sandbox-docker-registry-config + - emptyDir: {} + name: data +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/component: kubernetes-dashboard + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + name: flyte-sandbox-kubernetes-dashboard + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/component: kubernetes-dashboard + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: kubernetes-dashboard + strategy: + rollingUpdate: + maxSurge: 0 + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: + annotations: null + labels: + app.kubernetes.io/component: kubernetes-dashboard + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: kubernetes-dashboard + app.kubernetes.io/version: 2.7.0 + helm.sh/chart: kubernetes-dashboard-6.0.0 + spec: + containers: + - args: + - --namespace=flyte + - --metrics-provider=none + - --enable-insecure-login + - --enable-skip-login + image: kubernetesui/dashboard:sandbox + imagePullPolicy: Never + livenessProbe: + httpGet: + path: / + port: 9090 + scheme: HTTP + initialDelaySeconds: 30 + timeoutSeconds: 30 + name: kubernetes-dashboard + ports: + - containerPort: 9090 + name: http + protocol: TCP + resources: + limits: + cpu: 2 + memory: 200Mi + requests: + cpu: 100m + memory: 200Mi + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + runAsGroup: 2001 + runAsUser: 1001 + volumeMounts: + - mountPath: /certs + name: kubernetes-dashboard-certs + - mountPath: /tmp + name: tmp-volume + securityContext: + seccompProfile: + type: RuntimeDefault + serviceAccountName: flyte-sandbox-kubernetes-dashboard + volumes: + - name: kubernetes-dashboard-certs + secret: + secretName: flyte-sandbox-kubernetes-dashboard-certs + - emptyDir: {} + name: tmp-volume +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: minio + helm.sh/chart: minio-12.1.1 + name: flyte-sandbox-minio + namespace: flyte +spec: + selector: + matchLabels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: minio + strategy: + type: Recreate + template: + metadata: + annotations: + checksum/credentials-secret: c199ac45f9d95d97966921c814d6c8b38cbf7416458e19cbe6d001a04c264448 + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: minio + helm.sh/chart: minio-12.1.1 + spec: + affinity: + nodeAffinity: null + podAffinity: null + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchLabels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: minio + topologyKey: kubernetes.io/hostname + weight: 1 + containers: + - env: + - name: BITNAMI_DEBUG + value: "false" + - name: MINIO_SCHEME + value: http + - name: MINIO_FORCE_NEW_KEYS + value: "no" + - name: MINIO_ROOT_USER + valueFrom: + secretKeyRef: + key: root-user + name: flyte-sandbox-minio + - name: MINIO_ROOT_PASSWORD + valueFrom: + secretKeyRef: + key: root-password + name: flyte-sandbox-minio + - name: MINIO_DEFAULT_BUCKETS + value: my-s3-bucket + - name: MINIO_BROWSER + value: "on" + - name: MINIO_PROMETHEUS_AUTH_TYPE + value: public + - name: MINIO_CONSOLE_PORT_NUMBER + value: "9001" + - name: MINIO_BROWSER_REDIRECT_URL + value: http://localhost:30080/minio + envFrom: null + image: docker.io/bitnami/minio:sandbox + imagePullPolicy: Never + livenessProbe: + failureThreshold: 5 + httpGet: + path: /minio/health/live + port: minio-api + scheme: HTTP + initialDelaySeconds: 5 + periodSeconds: 5 + successThreshold: 1 + timeoutSeconds: 5 + name: minio + ports: + - containerPort: 9000 + name: minio-api + protocol: TCP + - containerPort: 9001 + name: minio-console + protocol: TCP + readinessProbe: + failureThreshold: 5 + initialDelaySeconds: 5 + periodSeconds: 5 + successThreshold: 1 + tcpSocket: + port: minio-api + timeoutSeconds: 1 + resources: + limits: {} + requests: {} + securityContext: + runAsNonRoot: true + runAsUser: 1001 + volumeMounts: + - mountPath: /data + name: data + initContainers: + - command: + - /bin/bash + - -ec + - | + chown -R 1001:1001 /data + image: docker.io/bitnami/bitnami-shell:sandbox + imagePullPolicy: Never + name: volume-permissions + resources: + limits: {} + requests: {} + securityContext: + runAsUser: 0 + volumeMounts: + - mountPath: /data + name: data + securityContext: + fsGroup: 1001 + serviceAccountName: flyte-sandbox-minio + volumes: + - name: data + persistentVolumeClaim: + claimName: flyte-sandbox-minio-storage +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyte-sandbox + app.kubernetes.io/version: 1.16.0 + helm.sh/chart: flyte-sandbox-0.1.0 + name: flyte-sandbox-proxy + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/component: proxy + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + template: + metadata: + labels: + app.kubernetes.io/component: proxy + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: flyte-sandbox + spec: + containers: + - image: envoyproxy/envoy:sandbox + imagePullPolicy: Never + name: proxy + ports: + - containerPort: 8000 + name: http + volumeMounts: + - mountPath: /etc/envoy + name: config + volumes: + - configMap: + name: flyte-sandbox-proxy-config + name: config +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyteagent + helm.sh/chart: flyteagent-v0.1.10 + name: flyteagent + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/name: flyteagent + template: + metadata: + annotations: null + labels: + app.kubernetes.io/instance: flyteagent + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: flyteagent + helm.sh/chart: flyteagent-v0.1.10 + spec: + containers: + - command: + - pyflyte + - serve + image: ghcr.io/flyteorg/flyteagent:1.9.1 + imagePullPolicy: IfNotPresent + name: flyteagent + ports: + - containerPort: 8000 + name: agent-grpc + resources: + limits: + cpu: 500m + ephemeral-storage: 200Mi + memory: 200Mi + requests: + cpu: 500m + ephemeral-storage: 200Mi + memory: 200Mi + volumeMounts: + - mountPath: /etc/secrets + name: flyteagent + serviceAccountName: flyteagent + volumes: + - name: flyteagent + secret: + secretName: flyteagent +--- +apiVersion: apps/v1 +kind: StatefulSet +metadata: + labels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: postgresql + helm.sh/chart: postgresql-12.1.9 + name: flyte-sandbox-postgresql + namespace: flyte +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: postgresql + serviceName: flyte-sandbox-postgresql-hl + template: + metadata: + annotations: null + labels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/managed-by: Helm + app.kubernetes.io/name: postgresql + helm.sh/chart: postgresql-12.1.9 + name: flyte-sandbox-postgresql + spec: + affinity: + nodeAffinity: null + podAffinity: null + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchLabels: + app.kubernetes.io/component: primary + app.kubernetes.io/instance: flyte-sandbox + app.kubernetes.io/name: postgresql + topologyKey: kubernetes.io/hostname + weight: 1 + containers: + - env: + - name: BITNAMI_DEBUG + value: "false" + - name: POSTGRESQL_PORT_NUMBER + value: "5432" + - name: POSTGRESQL_VOLUME_DIR + value: /bitnami/postgresql + - name: PGDATA + value: /bitnami/postgresql/data + - name: POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + key: postgres-password + name: flyte-sandbox-postgresql + - name: POSTGRESQL_ENABLE_LDAP + value: "no" + - name: POSTGRESQL_ENABLE_TLS + value: "no" + - name: POSTGRESQL_LOG_HOSTNAME + value: "false" + - name: POSTGRESQL_LOG_CONNECTIONS + value: "false" + - name: POSTGRESQL_LOG_DISCONNECTIONS + value: "false" + - name: POSTGRESQL_PGAUDIT_LOG_CATALOG + value: "off" + - name: POSTGRESQL_CLIENT_MIN_MESSAGES + value: error + - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES + value: pgaudit + image: docker.io/bitnami/postgresql:sandbox + imagePullPolicy: Never + livenessProbe: + exec: + command: + - /bin/sh + - -c + - exec pg_isready -U "postgres" -h 127.0.0.1 -p 5432 + failureThreshold: 6 + initialDelaySeconds: 30 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 5 + name: postgresql + ports: + - containerPort: 5432 + name: tcp-postgresql + readinessProbe: + exec: + command: + - /bin/sh + - -c + - -e + - | + exec pg_isready -U "postgres" -h 127.0.0.1 -p 5432 + [ -f /opt/bitnami/postgresql/tmp/.initialized ] || [ -f /bitnami/postgresql/.initialized ] + failureThreshold: 6 + initialDelaySeconds: 5 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 5 + resources: + limits: {} + requests: + cpu: 250m + memory: 256Mi + securityContext: + runAsUser: 1001 + volumeMounts: + - mountPath: /bitnami/postgresql + name: data + hostIPC: false + hostNetwork: false + initContainers: + - command: + - /bin/sh + - -ec + - | + chown 1001:1001 /bitnami/postgresql + mkdir -p /bitnami/postgresql/data + chmod 700 /bitnami/postgresql/data + find /bitnami/postgresql -mindepth 1 -maxdepth 1 -not -name "conf" -not -name ".snapshot" -not -name "lost+found" | \ + xargs -r chown -R 1001:1001 + image: docker.io/bitnami/bitnami-shell:sandbox + imagePullPolicy: Never + name: init-chmod-data + resources: + limits: {} + requests: {} + securityContext: + runAsUser: 0 + volumeMounts: + - mountPath: /bitnami/postgresql + name: data + securityContext: + fsGroup: 1001 + serviceAccountName: default + volumes: + - name: data + persistentVolumeClaim: + claimName: flyte-sandbox-db-storage + updateStrategy: + rollingUpdate: {} + type: RollingUpdate diff --git a/docker/sandbox-bundled/manifests/complete.yaml b/docker/sandbox-bundled/manifests/complete.yaml index 175e7aece8c..d7f6e8b0cc2 100644 --- a/docker/sandbox-bundled/manifests/complete.yaml +++ b/docker/sandbox-bundled/manifests/complete.yaml @@ -448,6 +448,7 @@ data: - container - sidecar - k8s-array + - agent-service plugins: logs: kubernetes-enabled: true @@ -464,6 +465,15 @@ data: kubernetes-template-uri: http://localhost:30080/kubernetes-dashboard/#/log/{{.namespace }}/{{ .podName }}/pod?namespace={{ .namespace }} cloudwatch-enabled: false stackdriver-enabled: false + agent-service: + defaultAgent: + defaultTimeout: 10s + endpoint: dns:///flyteagent.flyte.svc.cluster.local:8000 + insecure: true + timeouts: + GetTask: 10s + supportedTaskTypes: + - default_task 002-database.yaml: | database: postgres: @@ -795,7 +805,7 @@ type: Opaque --- apiVersion: v1 data: - haSharedSecret: bzd6QlVrSG9ya1c0MUxBWg== + haSharedSecret: d1l6eWRCOXBJcFhiNEo5QQ== proxyPassword: "" proxyUsername: "" kind: Secret @@ -1193,7 +1203,7 @@ spec: metadata: annotations: checksum/cluster-resource-templates: 6fd9b172465e3089fcc59f738b92b8dc4d8939360c19de8ee65f68b0e7422035 - checksum/configuration: 91f9c46efb44022473a71c6c25bc6ef20190610644a48f81a9c0e1ae01c2a73d + checksum/configuration: b765a68950c83acd0c069dac2a6569cf2b0f0f76a2760eea3561d1d04d6be831 checksum/configuration-secret: 09216ffaa3d29e14f88b1f30af580d02a2a5e014de4d750b7f275cc07ed4e914 labels: app.kubernetes.io/component: flyte-binary @@ -1356,7 +1366,7 @@ spec: metadata: annotations: checksum/config: 8f50e768255a87f078ba8b9879a0c174c3e045ffb46ac8723d2eedbe293c8d81 - checksum/secret: c60195b739184d9ad0f4dd231ec9b2bdbedcbc835c4651806c1fa32d29279994 + checksum/secret: 2f5b6d46fd3276b5b25c8a537298beb6943b13b0b21900db8b2da23e166f0593 labels: app: docker-registry release: flyte-sandbox diff --git a/docker/sandbox-bundled/manifests/dev.yaml b/docker/sandbox-bundled/manifests/dev.yaml index 164a867eb75..4f3f0592e87 100644 --- a/docker/sandbox-bundled/manifests/dev.yaml +++ b/docker/sandbox-bundled/manifests/dev.yaml @@ -499,7 +499,7 @@ metadata: --- apiVersion: v1 data: - haSharedSecret: S3hhYmcwb1E0enNmZXpHQw== + haSharedSecret: UkFsUVRMRndZeTNJUVNFSA== proxyPassword: "" proxyUsername: "" kind: Secret @@ -933,7 +933,7 @@ spec: metadata: annotations: checksum/config: 8f50e768255a87f078ba8b9879a0c174c3e045ffb46ac8723d2eedbe293c8d81 - checksum/secret: 32e8e4864e56d8e05e03763b1e04dc6c1821c30c5079087b39a02c1348560d34 + checksum/secret: 25a046ef1aaf34ffb59f7b92554e1cfd0015b9a11f7f165ce06bba31e3bced1b labels: app: docker-registry release: flyte-sandbox From 4717c8d0a1f511e1b0cf220f8321b2ff52cadb0e Mon Sep 17 00:00:00 2001 From: Future-Outlier Date: Wed, 11 Oct 2023 01:27:35 +0800 Subject: [PATCH 02/18] Agent Get Pending Message (#4106) * agent get pending message Signed-off-by: Future Outlier * tidy Signed-off-by: Future Outlier * tidy Signed-off-by: Future Outlier * tidy Signed-off-by: Future Outlier * new flyteidl Signed-off-by: Future Outlier * spell Signed-off-by: Future Outlier * flyteidl import issue Signed-off-by: Future Outlier --------- Signed-off-by: Future Outlier Co-authored-by: Future Outlier --- .../gen/pb-cpp/flyteidl/admin/agent.pb.cc | 101 ++++++++- flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.h | 68 +++++++ flyteidl/gen/pb-go/flyteidl/admin/agent.pb.go | 112 +++++----- .../pb-go/flyteidl/admin/agent.pb.validate.go | 2 + .../pb-go/flyteidl/service/agent.swagger.json | 4 + .../gen/pb-java/flyteidl/admin/Agent.java | 192 +++++++++++++++++- flyteidl/gen/pb-js/flyteidl.d.ts | 6 + flyteidl/gen/pb-js/flyteidl.js | 17 ++ .../gen/pb_python/flyteidl/admin/agent_pb2.py | 18 +- .../pb_python/flyteidl/admin/agent_pb2.pyi | 6 +- flyteidl/gen/pb_rust/flyteidl.admin.rs | 3 + flyteidl/protos/flyteidl/admin/agent.proto | 2 + .../go/tasks/pluginmachinery/core/phase.go | 1 - .../go/tasks/plugins/webapi/agent/plugin.go | 5 + .../tasks/plugins/webapi/agent/plugin_test.go | 68 +++++++ 15 files changed, 522 insertions(+), 83 deletions(-) diff --git a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc index 3ca2f983e26..d9fe2fed164 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc @@ -324,6 +324,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fadmin_2fagent_2eproto::o ~0u, // no _weak_field_map_ PROTOBUF_FIELD_OFFSET(::flyteidl::admin::Resource, state_), PROTOBUF_FIELD_OFFSET(::flyteidl::admin::Resource, outputs_), + PROTOBUF_FIELD_OFFSET(::flyteidl::admin::Resource, message_), ~0u, // no _has_bits_ PROTOBUF_FIELD_OFFSET(::flyteidl::admin::DeleteTaskRequest, _internal_metadata_), ~0u, // no _extensions_ @@ -347,8 +348,8 @@ static const ::google::protobuf::internal::MigrationSchema schemas[] PROTOBUF_SE { 53, -1, sizeof(::flyteidl::admin::GetTaskRequest)}, { 60, -1, sizeof(::flyteidl::admin::GetTaskResponse)}, { 66, -1, sizeof(::flyteidl::admin::Resource)}, - { 73, -1, sizeof(::flyteidl::admin::DeleteTaskRequest)}, - { 80, -1, sizeof(::flyteidl::admin::DeleteTaskResponse)}, + { 74, -1, sizeof(::flyteidl::admin::DeleteTaskRequest)}, + { 81, -1, sizeof(::flyteidl::admin::DeleteTaskResponse)}, }; static ::google::protobuf::Message const * const file_default_instances[] = { @@ -399,20 +400,21 @@ const char descriptor_table_protodef_flyteidl_2fadmin_2fagent_2eproto[] = "\016GetTaskRequest\022\021\n\ttask_type\030\001 \001(\t\022\025\n\rre" "source_meta\030\002 \001(\014\"=\n\017GetTaskResponse\022*\n\010" "resource\030\001 \001(\0132\030.flyteidl.admin.Resource" - "\"\\\n\010Resource\022$\n\005state\030\001 \001(\0162\025.flyteidl.a" + "\"m\n\010Resource\022$\n\005state\030\001 \001(\0162\025.flyteidl.a" "dmin.State\022*\n\007outputs\030\002 \001(\0132\031.flyteidl.c" - "ore.LiteralMap\"=\n\021DeleteTaskRequest\022\021\n\tt" - "ask_type\030\001 \001(\t\022\025\n\rresource_meta\030\002 \001(\014\"\024\n" - "\022DeleteTaskResponse*^\n\005State\022\025\n\021RETRYABL" - "E_FAILURE\020\000\022\025\n\021PERMANENT_FAILURE\020\001\022\013\n\007PE" - "NDING\020\002\022\013\n\007RUNNING\020\003\022\r\n\tSUCCEEDED\020\004B=Z;g" - "ithub.com/flyteorg/flyte/flyteidl/gen/pb" - "-go/flyteidl/adminb\006proto3" + "ore.LiteralMap\022\017\n\007message\030\003 \001(\t\"=\n\021Delet" + "eTaskRequest\022\021\n\ttask_type\030\001 \001(\t\022\025\n\rresou" + "rce_meta\030\002 \001(\014\"\024\n\022DeleteTaskResponse*^\n\005" + "State\022\025\n\021RETRYABLE_FAILURE\020\000\022\025\n\021PERMANEN" + "T_FAILURE\020\001\022\013\n\007PENDING\020\002\022\013\n\007RUNNING\020\003\022\r\n" + "\tSUCCEEDED\020\004B=Z;github.com/flyteorg/flyt" + "e/flyteidl/gen/pb-go/flyteidl/adminb\006pro" + "to3" ; ::google::protobuf::internal::DescriptorTable descriptor_table_flyteidl_2fadmin_2fagent_2eproto = { false, InitDefaults_flyteidl_2fadmin_2fagent_2eproto, descriptor_table_protodef_flyteidl_2fadmin_2fagent_2eproto, - "flyteidl/admin/agent.proto", &assign_descriptors_table_flyteidl_2fadmin_2fagent_2eproto, 1426, + "flyteidl/admin/agent.proto", &assign_descriptors_table_flyteidl_2fadmin_2fagent_2eproto, 1443, }; void AddDescriptors_flyteidl_2fadmin_2fagent_2eproto() { @@ -2962,6 +2964,7 @@ void Resource::clear_outputs() { #if !defined(_MSC_VER) || _MSC_VER >= 1900 const int Resource::kStateFieldNumber; const int Resource::kOutputsFieldNumber; +const int Resource::kMessageFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 Resource::Resource() @@ -2973,6 +2976,10 @@ Resource::Resource(const Resource& from) : ::google::protobuf::Message(), _internal_metadata_(nullptr) { _internal_metadata_.MergeFrom(from._internal_metadata_); + message_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + if (from.message().size() > 0) { + message_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.message_); + } if (from.has_outputs()) { outputs_ = new ::flyteidl::core::LiteralMap(*from.outputs_); } else { @@ -2985,6 +2992,7 @@ Resource::Resource(const Resource& from) void Resource::SharedCtor() { ::google::protobuf::internal::InitSCC( &scc_info_Resource_flyteidl_2fadmin_2fagent_2eproto.base); + message_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); ::memset(&outputs_, 0, static_cast( reinterpret_cast(&state_) - reinterpret_cast(&outputs_)) + sizeof(state_)); @@ -2996,6 +3004,7 @@ Resource::~Resource() { } void Resource::SharedDtor() { + message_.DestroyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); if (this != internal_default_instance()) delete outputs_; } @@ -3014,6 +3023,7 @@ void Resource::Clear() { // Prevent compiler warnings about cached_has_bits being unused (void) cached_has_bits; + message_.ClearToEmptyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); if (GetArenaNoVirtual() == nullptr && outputs_ != nullptr) { delete outputs_; } @@ -3056,6 +3066,22 @@ const char* Resource::_InternalParse(const char* begin, const char* end, void* o {parser_till_end, object}, ptr - size, ptr)); break; } + // string message = 3; + case 3: { + if (static_cast<::google::protobuf::uint8>(tag) != 26) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + ctx->extra_parse_data().SetFieldName("flyteidl.admin.Resource.message"); + object = msg->mutable_message(); + if (size > end - ptr + ::google::protobuf::internal::ParseContext::kSlopBytes) { + parser_till_end = ::google::protobuf::internal::GreedyStringParserUTF8; + goto string_till_end; + } + GOOGLE_PROTOBUF_PARSER_ASSERT(::google::protobuf::internal::StringCheckUTF8(ptr, size, ctx)); + ::google::protobuf::internal::InlineGreedyStringParser(object, ptr, size, ctx); + ptr += size; + break; + } default: { handle_unusual: if ((tag & 7) == 4 || tag == 0) { @@ -3071,6 +3097,10 @@ const char* Resource::_InternalParse(const char* begin, const char* end, void* o } // switch } // while return ptr; +string_till_end: + static_cast<::std::string*>(object)->clear(); + static_cast<::std::string*>(object)->reserve(size); + goto len_delim_till_end; len_delim_till_end: return ctx->StoreAndTailCall(ptr, end, {_InternalParse, msg}, {parser_till_end, object}, size); @@ -3111,6 +3141,21 @@ bool Resource::MergePartialFromCodedStream( break; } + // string message = 3; + case 3: { + if (static_cast< ::google::protobuf::uint8>(tag) == (26 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadString( + input, this->mutable_message())); + DO_(::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->message().data(), static_cast(this->message().length()), + ::google::protobuf::internal::WireFormatLite::PARSE, + "flyteidl.admin.Resource.message")); + } else { + goto handle_unusual; + } + break; + } + default: { handle_unusual: if (tag == 0) { @@ -3150,6 +3195,16 @@ void Resource::SerializeWithCachedSizes( 2, HasBitSetters::outputs(this), output); } + // string message = 3; + if (this->message().size() > 0) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->message().data(), static_cast(this->message().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.admin.Resource.message"); + ::google::protobuf::internal::WireFormatLite::WriteStringMaybeAliased( + 3, this->message(), output); + } + if (_internal_metadata_.have_unknown_fields()) { ::google::protobuf::internal::WireFormat::SerializeUnknownFields( _internal_metadata_.unknown_fields(), output); @@ -3176,6 +3231,17 @@ ::google::protobuf::uint8* Resource::InternalSerializeWithCachedSizesToArray( 2, HasBitSetters::outputs(this), target); } + // string message = 3; + if (this->message().size() > 0) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->message().data(), static_cast(this->message().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.admin.Resource.message"); + target = + ::google::protobuf::internal::WireFormatLite::WriteStringToArray( + 3, this->message(), target); + } + if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); @@ -3197,6 +3263,13 @@ size_t Resource::ByteSizeLong() const { // Prevent compiler warnings about cached_has_bits being unused (void) cached_has_bits; + // string message = 3; + if (this->message().size() > 0) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::StringSize( + this->message()); + } + // .flyteidl.core.LiteralMap outputs = 2; if (this->has_outputs()) { total_size += 1 + @@ -3237,6 +3310,10 @@ void Resource::MergeFrom(const Resource& from) { ::google::protobuf::uint32 cached_has_bits = 0; (void) cached_has_bits; + if (from.message().size() > 0) { + + message_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.message_); + } if (from.has_outputs()) { mutable_outputs()->::flyteidl::core::LiteralMap::MergeFrom(from.outputs()); } @@ -3270,6 +3347,8 @@ void Resource::Swap(Resource* other) { void Resource::InternalSwap(Resource* other) { using std::swap; _internal_metadata_.Swap(&other->_internal_metadata_); + message_.Swap(&other->message_, &::google::protobuf::internal::GetEmptyStringAlreadyInited(), + GetArenaNoVirtual()); swap(outputs_, other->outputs_); swap(state_, other->state_); } diff --git a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.h b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.h index 5ee08079fc0..5d43236ad38 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.h +++ b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.h @@ -1015,6 +1015,20 @@ class Resource final : // accessors ------------------------------------------------------- + // string message = 3; + void clear_message(); + static const int kMessageFieldNumber = 3; + const ::std::string& message() const; + void set_message(const ::std::string& value); + #if LANG_CXX11 + void set_message(::std::string&& value); + #endif + void set_message(const char* value); + void set_message(const char* value, size_t size); + ::std::string* mutable_message(); + ::std::string* release_message(); + void set_allocated_message(::std::string* message); + // .flyteidl.core.LiteralMap outputs = 2; bool has_outputs() const; void clear_outputs(); @@ -1035,6 +1049,7 @@ class Resource final : class HasBitSetters; ::google::protobuf::internal::InternalMetadataWithArena _internal_metadata_; + ::google::protobuf::internal::ArenaStringPtr message_; ::flyteidl::core::LiteralMap* outputs_; int state_; mutable ::google::protobuf::internal::CachedSize _cached_size_; @@ -1985,6 +2000,59 @@ inline void Resource::set_allocated_outputs(::flyteidl::core::LiteralMap* output // @@protoc_insertion_point(field_set_allocated:flyteidl.admin.Resource.outputs) } +// string message = 3; +inline void Resource::clear_message() { + message_.ClearToEmptyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline const ::std::string& Resource::message() const { + // @@protoc_insertion_point(field_get:flyteidl.admin.Resource.message) + return message_.GetNoArena(); +} +inline void Resource::set_message(const ::std::string& value) { + + message_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), value); + // @@protoc_insertion_point(field_set:flyteidl.admin.Resource.message) +} +#if LANG_CXX11 +inline void Resource::set_message(::std::string&& value) { + + message_.SetNoArena( + &::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::move(value)); + // @@protoc_insertion_point(field_set_rvalue:flyteidl.admin.Resource.message) +} +#endif +inline void Resource::set_message(const char* value) { + GOOGLE_DCHECK(value != nullptr); + + message_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::string(value)); + // @@protoc_insertion_point(field_set_char:flyteidl.admin.Resource.message) +} +inline void Resource::set_message(const char* value, size_t size) { + + message_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), + ::std::string(reinterpret_cast(value), size)); + // @@protoc_insertion_point(field_set_pointer:flyteidl.admin.Resource.message) +} +inline ::std::string* Resource::mutable_message() { + + // @@protoc_insertion_point(field_mutable:flyteidl.admin.Resource.message) + return message_.MutableNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline ::std::string* Resource::release_message() { + // @@protoc_insertion_point(field_release:flyteidl.admin.Resource.message) + + return message_.ReleaseNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline void Resource::set_allocated_message(::std::string* message) { + if (message != nullptr) { + + } else { + + } + message_.SetAllocatedNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), message); + // @@protoc_insertion_point(field_set_allocated:flyteidl.admin.Resource.message) +} + // ------------------------------------------------------------------- // DeleteTaskRequest diff --git a/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.go b/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.go index 436d00d29a4..7511d46559b 100644 --- a/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.go +++ b/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.go @@ -349,10 +349,12 @@ type Resource struct { // The outputs of the execution. It's typically used by sql task. Agent service will create a // Structured dataset pointing to the query result table. // +optional - Outputs *core.LiteralMap `protobuf:"bytes,2,opt,name=outputs,proto3" json:"outputs,omitempty"` - XXX_NoUnkeyedLiteral struct{} `json:"-"` - XXX_unrecognized []byte `json:"-"` - XXX_sizecache int32 `json:"-"` + Outputs *core.LiteralMap `protobuf:"bytes,2,opt,name=outputs,proto3" json:"outputs,omitempty"` + // A descriptive message for the current state. e.g. waiting for cluster. + Message string `protobuf:"bytes,3,opt,name=message,proto3" json:"message,omitempty"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` } func (m *Resource) Reset() { *m = Resource{} } @@ -394,6 +396,13 @@ func (m *Resource) GetOutputs() *core.LiteralMap { return nil } +func (m *Resource) GetMessage() string { + if m != nil { + return m.Message + } + return "" +} + // A message used to delete a task. type DeleteTaskRequest struct { // A predefined yet extensible Task type identifier. @@ -494,51 +503,52 @@ func init() { func init() { proto.RegisterFile("flyteidl/admin/agent.proto", fileDescriptor_c434e52bb0028071) } var fileDescriptor_c434e52bb0028071 = []byte{ - // 726 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xa4, 0x55, 0xed, 0x6e, 0xe2, 0x46, - 0x14, 0x2d, 0x10, 0x08, 0x5c, 0xf2, 0x01, 0xd3, 0xa0, 0x3a, 0x24, 0xad, 0x22, 0xaa, 0x56, 0x51, - 0xab, 0x1a, 0x85, 0x54, 0x6d, 0xd2, 0xaa, 0xad, 0x48, 0x70, 0x11, 0x12, 0x41, 0xd1, 0x04, 0xaa, - 0xb6, 0xd2, 0x2e, 0x1a, 0xcc, 0x85, 0xb5, 0x30, 0x63, 0xaf, 0x67, 0x8c, 0xc2, 0xef, 0x7d, 0x89, - 0x7d, 0xdc, 0x95, 0xc7, 0x1f, 0x01, 0xc4, 0xae, 0x12, 0xed, 0x3f, 0xcf, 0x3d, 0xe7, 0x9e, 0x39, - 0x73, 0xee, 0xd8, 0x86, 0xea, 0xc4, 0x5e, 0x4a, 0xb4, 0xc6, 0x76, 0x9d, 0x8d, 0xe7, 0x16, 0xaf, - 0xb3, 0x29, 0x72, 0xa9, 0xbb, 0x9e, 0x23, 0x1d, 0x72, 0x10, 0x63, 0xba, 0xc2, 0xaa, 0xa7, 0x09, - 0xd7, 0x74, 0x3c, 0xac, 0xdb, 0x96, 0x44, 0x8f, 0xd9, 0x22, 0x64, 0x57, 0x8f, 0xd7, 0x51, 0xc9, - 0xc4, 0x2c, 0x86, 0xbe, 0x5e, 0x87, 0x2c, 0x2e, 0xd1, 0x9b, 0x30, 0x13, 0x23, 0xf8, 0x9b, 0x0d, - 0x78, 0x8c, 0x5c, 0x5a, 0x13, 0x0b, 0xbd, 0x10, 0xaf, 0xbd, 0xcf, 0x42, 0xa5, 0xcf, 0xc4, 0xcc, - 0x78, 0x44, 0xd3, 0x97, 0x96, 0xc3, 0xef, 0x50, 0xb2, 0x31, 0x93, 0x8c, 0x50, 0x28, 0x07, 0xfb, - 0x0c, 0x31, 0x46, 0x86, 0xd6, 0x58, 0x4b, 0x9d, 0xa5, 0xce, 0x8b, 0x8d, 0xef, 0xf5, 0xc4, 0x7d, - 0xa0, 0xaa, 0xaf, 0x09, 0x74, 0x92, 0x2d, 0xe8, 0xa1, 0x5c, 0x07, 0xc8, 0x29, 0x14, 0x38, 0x9b, - 0xa3, 0x70, 0x99, 0x89, 0x5a, 0xfa, 0x2c, 0x75, 0x5e, 0xa0, 0x4f, 0x05, 0xd2, 0x81, 0x9c, 0xcd, - 0x46, 0x68, 0x0b, 0x2d, 0x73, 0x96, 0x39, 0x2f, 0x36, 0x2e, 0xf4, 0xf5, 0x90, 0xf4, 0xad, 0x46, - 0xf5, 0xae, 0xea, 0x31, 0xb8, 0xf4, 0x96, 0x34, 0x12, 0x20, 0xff, 0x42, 0x91, 0x71, 0xee, 0x48, - 0x16, 0x30, 0x85, 0xb6, 0xa3, 0xf4, 0x7e, 0x79, 0x9e, 0x5e, 0xf3, 0xa9, 0x31, 0x14, 0x5d, 0x95, - 0x22, 0x3a, 0x7c, 0x39, 0xbb, 0x12, 0x43, 0x81, 0xde, 0xc2, 0x32, 0x71, 0xc8, 0x4c, 0xd3, 0xf1, - 0xb9, 0xd4, 0xb2, 0xea, 0x30, 0xe5, 0xd9, 0x95, 0x78, 0x08, 0x91, 0x66, 0x08, 0x10, 0x09, 0x15, - 0xe4, 0x0b, 0xcb, 0x73, 0xf8, 0x1c, 0xb9, 0x1c, 0x2e, 0x98, 0x67, 0xb1, 0x91, 0x8d, 0x42, 0xcb, - 0x29, 0x4f, 0x7f, 0x3d, 0xcf, 0x93, 0xf1, 0x24, 0xf1, 0x4f, 0xac, 0x10, 0x9a, 0x3b, 0xc2, 0x2d, - 0x50, 0xf5, 0x1a, 0x8a, 0x2b, 0xb1, 0x90, 0x12, 0x64, 0x66, 0xb8, 0x54, 0xd3, 0x2b, 0xd0, 0xe0, - 0x91, 0x1c, 0x41, 0x76, 0xc1, 0x6c, 0x3f, 0x9e, 0x42, 0xb8, 0xf8, 0x2d, 0x7d, 0x95, 0xaa, 0xfe, - 0x09, 0xa5, 0xcd, 0x04, 0x5e, 0xd4, 0xdf, 0x86, 0xe3, 0x8f, 0xba, 0x7d, 0x89, 0x50, 0xed, 0x5d, - 0x1a, 0xca, 0xb7, 0x1e, 0x32, 0x89, 0x41, 0x26, 0x14, 0xdf, 0xfa, 0x28, 0x24, 0xb9, 0x80, 0x9c, - 0xc5, 0x5d, 0x5f, 0x8a, 0xe8, 0x2e, 0x1e, 0x6f, 0xdc, 0xc5, 0x6e, 0xf8, 0xe6, 0xdc, 0x31, 0x97, - 0x46, 0x44, 0xf2, 0x2b, 0xe4, 0x25, 0xce, 0x5d, 0x9b, 0xc9, 0x70, 0x97, 0x62, 0xe3, 0x64, 0xcb, - 0x05, 0xee, 0x47, 0x14, 0x9a, 0x90, 0xc9, 0xb7, 0xb0, 0xef, 0xf8, 0xd2, 0xf5, 0xe5, 0xd0, 0xf5, - 0x70, 0x62, 0x3d, 0x6a, 0x19, 0xe5, 0x71, 0x2f, 0x2c, 0xde, 0xab, 0x1a, 0x79, 0x05, 0x5f, 0x6d, - 0xbc, 0x27, 0xf3, 0x68, 0x6a, 0xda, 0x8e, 0xda, 0xec, 0xbb, 0x67, 0x8d, 0x98, 0x56, 0xe4, 0xb6, - 0x72, 0xed, 0x1a, 0xc8, 0x6a, 0x08, 0xc2, 0x75, 0xb8, 0x50, 0xce, 0x3c, 0x14, 0x8e, 0xef, 0x99, - 0xa8, 0xb6, 0x53, 0x61, 0xec, 0xd1, 0xbd, 0xb8, 0x18, 0xb4, 0xd7, 0x28, 0x1c, 0xb4, 0x51, 0xae, - 0x86, 0x77, 0x02, 0x05, 0xe5, 0x55, 0x2e, 0x5d, 0x8c, 0x86, 0x90, 0x0f, 0x0a, 0xfd, 0xa5, 0xbb, - 0x45, 0x33, 0xbd, 0x45, 0xb3, 0x0d, 0x87, 0x89, 0x66, 0xe4, 0xe5, 0x67, 0xc8, 0xc7, 0x94, 0x68, - 0x26, 0xda, 0xe6, 0x89, 0x69, 0x84, 0xd3, 0x84, 0x59, 0xb3, 0x21, 0x1f, 0x57, 0xc9, 0x8f, 0x90, - 0x15, 0x32, 0x98, 0x4e, 0xd0, 0x7e, 0xd0, 0xa8, 0x6c, 0xb6, 0x3f, 0x04, 0x20, 0x0d, 0x39, 0xe4, - 0x12, 0x76, 0xc3, 0xfc, 0x45, 0x34, 0xcc, 0x4f, 0xdc, 0x80, 0x98, 0x59, 0x1b, 0x40, 0xb9, 0x85, - 0x36, 0xae, 0x5f, 0xa5, 0xcf, 0x4f, 0xe3, 0x08, 0xc8, 0xaa, 0x6c, 0x18, 0xc8, 0x0f, 0xaf, 0x21, - 0xab, 0x1c, 0x93, 0x0a, 0x94, 0xa9, 0xd1, 0xa7, 0xff, 0x35, 0x6f, 0xba, 0xc6, 0xf0, 0xef, 0x66, - 0xa7, 0x3b, 0xa0, 0x46, 0xe9, 0x8b, 0xa0, 0x7c, 0x6f, 0xd0, 0xbb, 0x66, 0xcf, 0xe8, 0xf5, 0x93, - 0x72, 0x8a, 0x14, 0x61, 0xf7, 0xde, 0xe8, 0xb5, 0x3a, 0xbd, 0x76, 0x29, 0x1d, 0x2c, 0xe8, 0xa0, - 0xd7, 0x0b, 0x16, 0x19, 0xb2, 0x0f, 0x85, 0x87, 0xc1, 0xed, 0xad, 0x61, 0xb4, 0x8c, 0x56, 0x69, - 0xe7, 0xe6, 0x8f, 0xff, 0x7f, 0x9f, 0x5a, 0xf2, 0x8d, 0x3f, 0xd2, 0x4d, 0x67, 0x5e, 0x57, 0x87, - 0x77, 0xbc, 0x69, 0xf8, 0x50, 0x4f, 0xbe, 0xf7, 0x53, 0xe4, 0x75, 0x77, 0xf4, 0xd3, 0xd4, 0xa9, - 0xaf, 0xff, 0x86, 0x46, 0x39, 0xf5, 0xe5, 0xbf, 0xfc, 0x10, 0x00, 0x00, 0xff, 0xff, 0x14, 0xef, - 0x1e, 0x8b, 0x9f, 0x06, 0x00, 0x00, + // 740 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xa4, 0x55, 0x7f, 0x6b, 0x2a, 0x47, + 0x14, 0xad, 0x1a, 0x8d, 0x5e, 0xf3, 0x43, 0xa7, 0x91, 0x6e, 0x4c, 0x5a, 0x82, 0xa5, 0x25, 0xb4, + 0x74, 0x25, 0xa6, 0xb4, 0x49, 0x4b, 0x5b, 0x4c, 0xdc, 0x8a, 0x60, 0x24, 0x4c, 0xb4, 0xb4, 0x85, + 0x56, 0xc6, 0xf5, 0xea, 0x5b, 0x5c, 0x67, 0xf7, 0xed, 0xcc, 0x4a, 0xfc, 0xfb, 0xc1, 0xfb, 0x0c, + 0xef, 0xe3, 0x3e, 0x76, 0xf6, 0x47, 0x54, 0x7c, 0x8f, 0x84, 0xf7, 0xdf, 0xce, 0x3d, 0xf7, 0x9e, + 0x39, 0x73, 0xce, 0x2c, 0x03, 0xd5, 0x89, 0xbd, 0x94, 0x68, 0x8d, 0xed, 0x3a, 0x1b, 0xcf, 0x2d, + 0x5e, 0x67, 0x53, 0xe4, 0x52, 0x77, 0x3d, 0x47, 0x3a, 0xe4, 0x20, 0xc6, 0x74, 0x85, 0x55, 0x4f, + 0x93, 0x5e, 0xd3, 0xf1, 0xb0, 0x6e, 0x5b, 0x12, 0x3d, 0x66, 0x8b, 0xb0, 0xbb, 0x7a, 0xbc, 0x8e, + 0x4a, 0x26, 0x66, 0x31, 0xf4, 0xe5, 0x3a, 0x64, 0x71, 0x89, 0xde, 0x84, 0x99, 0x18, 0xc1, 0x5f, + 0x6d, 0xc0, 0x63, 0xe4, 0xd2, 0x9a, 0x58, 0xe8, 0x85, 0x78, 0xed, 0x5d, 0x16, 0x2a, 0x7d, 0x26, + 0x66, 0xc6, 0x23, 0x9a, 0xbe, 0xb4, 0x1c, 0x7e, 0x87, 0x92, 0x8d, 0x99, 0x64, 0x84, 0x42, 0x39, + 0xd8, 0x67, 0x88, 0x31, 0x32, 0xb4, 0xc6, 0x5a, 0xea, 0x2c, 0x75, 0x5e, 0x6c, 0x7c, 0xab, 0x27, + 0xea, 0x03, 0x56, 0x7d, 0x8d, 0xa0, 0x93, 0x6c, 0x41, 0x0f, 0xe5, 0x3a, 0x40, 0x4e, 0xa1, 0xc0, + 0xd9, 0x1c, 0x85, 0xcb, 0x4c, 0xd4, 0xd2, 0x67, 0xa9, 0xf3, 0x02, 0x7d, 0x2a, 0x90, 0x0e, 0xe4, + 0x6c, 0x36, 0x42, 0x5b, 0x68, 0x99, 0xb3, 0xcc, 0x79, 0xb1, 0x71, 0xa1, 0xaf, 0x9b, 0xa4, 0x6f, + 0x15, 0xaa, 0x77, 0xd5, 0x8c, 0xc1, 0xa5, 0xb7, 0xa4, 0x11, 0x01, 0xf9, 0x1b, 0x8a, 0x8c, 0x73, + 0x47, 0xb2, 0xa0, 0x53, 0x68, 0x3b, 0x8a, 0xef, 0xa7, 0xe7, 0xf1, 0x35, 0x9f, 0x06, 0x43, 0xd2, + 0x55, 0x2a, 0xa2, 0xc3, 0xe7, 0xb3, 0x2b, 0x31, 0x14, 0xe8, 0x2d, 0x2c, 0x13, 0x87, 0xcc, 0x34, + 0x1d, 0x9f, 0x4b, 0x2d, 0xab, 0x0e, 0x53, 0x9e, 0x5d, 0x89, 0x87, 0x10, 0x69, 0x86, 0x00, 0x91, + 0x50, 0x41, 0xbe, 0xb0, 0x3c, 0x87, 0xcf, 0x91, 0xcb, 0xe1, 0x82, 0x79, 0x16, 0x1b, 0xd9, 0x28, + 0xb4, 0x9c, 0xd2, 0xf4, 0xc7, 0xf3, 0x34, 0x19, 0x4f, 0x14, 0x7f, 0xc5, 0x0c, 0xa1, 0xb8, 0x23, + 0xdc, 0x02, 0x55, 0xaf, 0xa1, 0xb8, 0x62, 0x0b, 0x29, 0x41, 0x66, 0x86, 0x4b, 0x95, 0x5e, 0x81, + 0x06, 0x9f, 0xe4, 0x08, 0xb2, 0x0b, 0x66, 0xfb, 0x71, 0x0a, 0xe1, 0xe2, 0x97, 0xf4, 0x55, 0xaa, + 0xfa, 0x3b, 0x94, 0x36, 0x1d, 0x78, 0xd1, 0x7c, 0x1b, 0x8e, 0x3f, 0xa8, 0xf6, 0x25, 0x44, 0xb5, + 0x37, 0x69, 0x28, 0xdf, 0x7a, 0xc8, 0x24, 0x06, 0x9e, 0x50, 0x7c, 0xed, 0xa3, 0x90, 0xe4, 0x02, + 0x72, 0x16, 0x77, 0x7d, 0x29, 0xa2, 0xbb, 0x78, 0xbc, 0x71, 0x17, 0xbb, 0xe1, 0x9f, 0x73, 0xc7, + 0x5c, 0x1a, 0x35, 0x92, 0x9f, 0x21, 0x2f, 0x71, 0xee, 0xda, 0x4c, 0x86, 0xbb, 0x14, 0x1b, 0x27, + 0x5b, 0x2e, 0x70, 0x3f, 0x6a, 0xa1, 0x49, 0x33, 0xf9, 0x1a, 0xf6, 0x1d, 0x5f, 0xba, 0xbe, 0x1c, + 0xba, 0x1e, 0x4e, 0xac, 0x47, 0x2d, 0xa3, 0x34, 0xee, 0x85, 0xc5, 0x7b, 0x55, 0x23, 0xff, 0xc1, + 0x17, 0x1b, 0xff, 0xc9, 0x3c, 0x4a, 0x4d, 0xdb, 0x51, 0x9b, 0x7d, 0xf3, 0xac, 0x88, 0x69, 0x45, + 0x6e, 0x2b, 0xd7, 0xae, 0x81, 0xac, 0x9a, 0x20, 0x5c, 0x87, 0x0b, 0xa5, 0xcc, 0x43, 0xe1, 0xf8, + 0x9e, 0x89, 0x6a, 0x3b, 0x65, 0xc6, 0x1e, 0xdd, 0x8b, 0x8b, 0xc1, 0x78, 0x8d, 0xc2, 0x41, 0x1b, + 0xe5, 0xaa, 0x79, 0x27, 0x50, 0x50, 0x5a, 0xe5, 0xd2, 0xc5, 0x28, 0x84, 0x7c, 0x50, 0xe8, 0x2f, + 0xdd, 0x2d, 0x9c, 0xe9, 0x2d, 0x9c, 0x6d, 0x38, 0x4c, 0x38, 0x23, 0x2d, 0x3f, 0x42, 0x3e, 0x6e, + 0x89, 0x32, 0xd1, 0x36, 0x4f, 0x4c, 0x23, 0x9c, 0x26, 0x9d, 0xb5, 0xb7, 0x29, 0xc8, 0xc7, 0x65, + 0xf2, 0x3d, 0x64, 0x85, 0x0c, 0xe2, 0x09, 0xe6, 0x0f, 0x1a, 0x95, 0xcd, 0xf9, 0x87, 0x00, 0xa4, + 0x61, 0x0f, 0xb9, 0x84, 0xdd, 0x30, 0x00, 0x11, 0xa5, 0xf9, 0x91, 0x2b, 0x10, 0x77, 0x12, 0x0d, + 0x76, 0xe7, 0x28, 0x04, 0x9b, 0x62, 0x14, 0x62, 0xbc, 0xac, 0x0d, 0xa0, 0xdc, 0x42, 0x1b, 0xd7, + 0x6f, 0xd9, 0xa7, 0x1b, 0x75, 0x04, 0x64, 0x95, 0x36, 0xf4, 0xea, 0xbb, 0xff, 0x21, 0xab, 0xce, + 0x42, 0x2a, 0x50, 0xa6, 0x46, 0x9f, 0xfe, 0xd3, 0xbc, 0xe9, 0x1a, 0xc3, 0x3f, 0x9b, 0x9d, 0xee, + 0x80, 0x1a, 0xa5, 0xcf, 0x82, 0xf2, 0xbd, 0x41, 0xef, 0x9a, 0x3d, 0xa3, 0xd7, 0x4f, 0xca, 0x29, + 0x52, 0x84, 0xdd, 0x7b, 0xa3, 0xd7, 0xea, 0xf4, 0xda, 0xa5, 0x74, 0xb0, 0xa0, 0x83, 0x5e, 0x2f, + 0x58, 0x64, 0xc8, 0x3e, 0x14, 0x1e, 0x06, 0xb7, 0xb7, 0x86, 0xd1, 0x32, 0x5a, 0xa5, 0x9d, 0x9b, + 0xdf, 0xfe, 0xfd, 0x75, 0x6a, 0xc9, 0x57, 0xfe, 0x48, 0x37, 0x9d, 0x79, 0x5d, 0xd9, 0xe2, 0x78, + 0xd3, 0xf0, 0xa3, 0x9e, 0x3c, 0x05, 0x53, 0xe4, 0x75, 0x77, 0xf4, 0xc3, 0xd4, 0xa9, 0xaf, 0xbf, + 0x50, 0xa3, 0x9c, 0x7a, 0x14, 0x2e, 0xdf, 0x07, 0x00, 0x00, 0xff, 0xff, 0xc9, 0x0a, 0x0e, 0xc8, + 0xba, 0x06, 0x00, 0x00, } diff --git a/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.validate.go b/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.validate.go index 0ba579cb0ba..7efab6b18ad 100644 --- a/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.validate.go +++ b/flyteidl/gen/pb-go/flyteidl/admin/agent.pb.validate.go @@ -454,6 +454,8 @@ func (m *Resource) Validate() error { } } + // no validation rules for Message + return nil } diff --git a/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json b/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json index 0a788c919c3..2d1d5ea07bd 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json +++ b/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json @@ -197,6 +197,10 @@ "outputs": { "$ref": "#/definitions/coreLiteralMap", "title": "The outputs of the execution. It's typically used by sql task. Agent service will create a\nStructured dataset pointing to the query result table.\n+optional" + }, + "message": { + "type": "string", + "description": "A descriptive message for the current state. e.g. waiting for cluster." } } }, diff --git a/flyteidl/gen/pb-java/flyteidl/admin/Agent.java b/flyteidl/gen/pb-java/flyteidl/admin/Agent.java index b0d9d2574e1..b2dfd16e61d 100644 --- a/flyteidl/gen/pb-java/flyteidl/admin/Agent.java +++ b/flyteidl/gen/pb-java/flyteidl/admin/Agent.java @@ -5460,6 +5460,24 @@ public interface ResourceOrBuilder extends * .flyteidl.core.LiteralMap outputs = 2; */ flyteidl.core.Literals.LiteralMapOrBuilder getOutputsOrBuilder(); + + /** + *
+     * A descriptive message for the current state. e.g. waiting for cluster.
+     * 
+ * + * string message = 3; + */ + java.lang.String getMessage(); + /** + *
+     * A descriptive message for the current state. e.g. waiting for cluster.
+     * 
+ * + * string message = 3; + */ + com.google.protobuf.ByteString + getMessageBytes(); } /** * Protobuf type {@code flyteidl.admin.Resource} @@ -5475,6 +5493,7 @@ private Resource(com.google.protobuf.GeneratedMessageV3.Builder builder) { } private Resource() { state_ = 0; + message_ = ""; } @java.lang.Override @@ -5520,6 +5539,12 @@ private Resource( break; } + case 26: { + java.lang.String s = input.readStringRequireUtf8(); + + message_ = s; + break; + } default: { if (!parseUnknownField( input, unknownFields, extensionRegistry, tag)) { @@ -5616,6 +5641,48 @@ public flyteidl.core.Literals.LiteralMapOrBuilder getOutputsOrBuilder() { return getOutputs(); } + public static final int MESSAGE_FIELD_NUMBER = 3; + private volatile java.lang.Object message_; + /** + *
+     * A descriptive message for the current state. e.g. waiting for cluster.
+     * 
+ * + * string message = 3; + */ + public java.lang.String getMessage() { + java.lang.Object ref = message_; + if (ref instanceof java.lang.String) { + return (java.lang.String) ref; + } else { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + message_ = s; + return s; + } + } + /** + *
+     * A descriptive message for the current state. e.g. waiting for cluster.
+     * 
+ * + * string message = 3; + */ + public com.google.protobuf.ByteString + getMessageBytes() { + java.lang.Object ref = message_; + if (ref instanceof java.lang.String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + message_ = b; + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + private byte memoizedIsInitialized = -1; @java.lang.Override public final boolean isInitialized() { @@ -5636,6 +5703,9 @@ public void writeTo(com.google.protobuf.CodedOutputStream output) if (outputs_ != null) { output.writeMessage(2, getOutputs()); } + if (!getMessageBytes().isEmpty()) { + com.google.protobuf.GeneratedMessageV3.writeString(output, 3, message_); + } unknownFields.writeTo(output); } @@ -5653,6 +5723,9 @@ public int getSerializedSize() { size += com.google.protobuf.CodedOutputStream .computeMessageSize(2, getOutputs()); } + if (!getMessageBytes().isEmpty()) { + size += com.google.protobuf.GeneratedMessageV3.computeStringSize(3, message_); + } size += unknownFields.getSerializedSize(); memoizedSize = size; return size; @@ -5674,6 +5747,8 @@ public boolean equals(final java.lang.Object obj) { if (!getOutputs() .equals(other.getOutputs())) return false; } + if (!getMessage() + .equals(other.getMessage())) return false; if (!unknownFields.equals(other.unknownFields)) return false; return true; } @@ -5691,6 +5766,8 @@ public int hashCode() { hash = (37 * hash) + OUTPUTS_FIELD_NUMBER; hash = (53 * hash) + getOutputs().hashCode(); } + hash = (37 * hash) + MESSAGE_FIELD_NUMBER; + hash = (53 * hash) + getMessage().hashCode(); hash = (29 * hash) + unknownFields.hashCode(); memoizedHashCode = hash; return hash; @@ -5832,6 +5909,8 @@ public Builder clear() { outputs_ = null; outputsBuilder_ = null; } + message_ = ""; + return this; } @@ -5864,6 +5943,7 @@ public flyteidl.admin.Agent.Resource buildPartial() { } else { result.outputs_ = outputsBuilder_.build(); } + result.message_ = message_; onBuilt(); return result; } @@ -5918,6 +5998,10 @@ public Builder mergeFrom(flyteidl.admin.Agent.Resource other) { if (other.hasOutputs()) { mergeOutputs(other.getOutputs()); } + if (!other.getMessage().isEmpty()) { + message_ = other.message_; + onChanged(); + } this.mergeUnknownFields(other.unknownFields); onChanged(); return this; @@ -6182,6 +6266,95 @@ public flyteidl.core.Literals.LiteralMapOrBuilder getOutputsOrBuilder() { } return outputsBuilder_; } + + private java.lang.Object message_ = ""; + /** + *
+       * A descriptive message for the current state. e.g. waiting for cluster.
+       * 
+ * + * string message = 3; + */ + public java.lang.String getMessage() { + java.lang.Object ref = message_; + if (!(ref instanceof java.lang.String)) { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + message_ = s; + return s; + } else { + return (java.lang.String) ref; + } + } + /** + *
+       * A descriptive message for the current state. e.g. waiting for cluster.
+       * 
+ * + * string message = 3; + */ + public com.google.protobuf.ByteString + getMessageBytes() { + java.lang.Object ref = message_; + if (ref instanceof String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + message_ = b; + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + /** + *
+       * A descriptive message for the current state. e.g. waiting for cluster.
+       * 
+ * + * string message = 3; + */ + public Builder setMessage( + java.lang.String value) { + if (value == null) { + throw new NullPointerException(); + } + + message_ = value; + onChanged(); + return this; + } + /** + *
+       * A descriptive message for the current state. e.g. waiting for cluster.
+       * 
+ * + * string message = 3; + */ + public Builder clearMessage() { + + message_ = getDefaultInstance().getMessage(); + onChanged(); + return this; + } + /** + *
+       * A descriptive message for the current state. e.g. waiting for cluster.
+       * 
+ * + * string message = 3; + */ + public Builder setMessageBytes( + com.google.protobuf.ByteString value) { + if (value == null) { + throw new NullPointerException(); + } + checkByteStringIsUtf8(value); + + message_ = value; + onChanged(); + return this; + } @java.lang.Override public final Builder setUnknownFields( final com.google.protobuf.UnknownFieldSet unknownFields) { @@ -7424,15 +7597,16 @@ public flyteidl.admin.Agent.DeleteTaskResponse getDefaultInstanceForType() { "\016GetTaskRequest\022\021\n\ttask_type\030\001 \001(\t\022\025\n\rre" + "source_meta\030\002 \001(\014\"=\n\017GetTaskResponse\022*\n\010" + "resource\030\001 \001(\0132\030.flyteidl.admin.Resource" + - "\"\\\n\010Resource\022$\n\005state\030\001 \001(\0162\025.flyteidl.a" + + "\"m\n\010Resource\022$\n\005state\030\001 \001(\0162\025.flyteidl.a" + "dmin.State\022*\n\007outputs\030\002 \001(\0132\031.flyteidl.c" + - "ore.LiteralMap\"=\n\021DeleteTaskRequest\022\021\n\tt" + - "ask_type\030\001 \001(\t\022\025\n\rresource_meta\030\002 \001(\014\"\024\n" + - "\022DeleteTaskResponse*^\n\005State\022\025\n\021RETRYABL" + - "E_FAILURE\020\000\022\025\n\021PERMANENT_FAILURE\020\001\022\013\n\007PE" + - "NDING\020\002\022\013\n\007RUNNING\020\003\022\r\n\tSUCCEEDED\020\004B=Z;g" + - "ithub.com/flyteorg/flyte/flyteidl/gen/pb" + - "-go/flyteidl/adminb\006proto3" + "ore.LiteralMap\022\017\n\007message\030\003 \001(\t\"=\n\021Delet" + + "eTaskRequest\022\021\n\ttask_type\030\001 \001(\t\022\025\n\rresou" + + "rce_meta\030\002 \001(\014\"\024\n\022DeleteTaskResponse*^\n\005" + + "State\022\025\n\021RETRYABLE_FAILURE\020\000\022\025\n\021PERMANEN" + + "T_FAILURE\020\001\022\013\n\007PENDING\020\002\022\013\n\007RUNNING\020\003\022\r\n" + + "\tSUCCEEDED\020\004B=Z;github.com/flyteorg/flyt" + + "e/flyteidl/gen/pb-go/flyteidl/adminb\006pro" + + "to3" }; com.google.protobuf.Descriptors.FileDescriptor.InternalDescriptorAssigner assigner = new com.google.protobuf.Descriptors.FileDescriptor. InternalDescriptorAssigner() { @@ -7503,7 +7677,7 @@ public com.google.protobuf.ExtensionRegistry assignDescriptors( internal_static_flyteidl_admin_Resource_fieldAccessorTable = new com.google.protobuf.GeneratedMessageV3.FieldAccessorTable( internal_static_flyteidl_admin_Resource_descriptor, - new java.lang.String[] { "State", "Outputs", }); + new java.lang.String[] { "State", "Outputs", "Message", }); internal_static_flyteidl_admin_DeleteTaskRequest_descriptor = getDescriptor().getMessageTypes().get(6); internal_static_flyteidl_admin_DeleteTaskRequest_fieldAccessorTable = new diff --git a/flyteidl/gen/pb-js/flyteidl.d.ts b/flyteidl/gen/pb-js/flyteidl.d.ts index 23d4dcceda0..4889385c65b 100644 --- a/flyteidl/gen/pb-js/flyteidl.d.ts +++ b/flyteidl/gen/pb-js/flyteidl.d.ts @@ -8050,6 +8050,9 @@ export namespace flyteidl { /** Resource outputs */ outputs?: (flyteidl.core.ILiteralMap|null); + + /** Resource message */ + message?: (string|null); } /** Represents a Resource. */ @@ -8067,6 +8070,9 @@ export namespace flyteidl { /** Resource outputs. */ public outputs?: (flyteidl.core.ILiteralMap|null); + /** Resource message. */ + public message: string; + /** * Creates a new Resource instance using the specified properties. * @param [properties] Properties to set diff --git a/flyteidl/gen/pb-js/flyteidl.js b/flyteidl/gen/pb-js/flyteidl.js index 592b6d58c17..46812b6db70 100644 --- a/flyteidl/gen/pb-js/flyteidl.js +++ b/flyteidl/gen/pb-js/flyteidl.js @@ -19640,6 +19640,7 @@ * @interface IResource * @property {flyteidl.admin.State|null} [state] Resource state * @property {flyteidl.core.ILiteralMap|null} [outputs] Resource outputs + * @property {string|null} [message] Resource message */ /** @@ -19673,6 +19674,14 @@ */ Resource.prototype.outputs = null; + /** + * Resource message. + * @member {string} message + * @memberof flyteidl.admin.Resource + * @instance + */ + Resource.prototype.message = ""; + /** * Creates a new Resource instance using the specified properties. * @function create @@ -19701,6 +19710,8 @@ writer.uint32(/* id 1, wireType 0 =*/8).int32(message.state); if (message.outputs != null && message.hasOwnProperty("outputs")) $root.flyteidl.core.LiteralMap.encode(message.outputs, writer.uint32(/* id 2, wireType 2 =*/18).fork()).ldelim(); + if (message.message != null && message.hasOwnProperty("message")) + writer.uint32(/* id 3, wireType 2 =*/26).string(message.message); return writer; }; @@ -19728,6 +19739,9 @@ case 2: message.outputs = $root.flyteidl.core.LiteralMap.decode(reader, reader.uint32()); break; + case 3: + message.message = reader.string(); + break; default: reader.skipType(tag & 7); break; @@ -19763,6 +19777,9 @@ if (error) return "outputs." + error; } + if (message.message != null && message.hasOwnProperty("message")) + if (!$util.isString(message.message)) + return "message: string expected"; return null; }; diff --git a/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.py b/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.py index 236728acf14..bb0aa9384af 100644 --- a/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.py +++ b/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.py @@ -17,7 +17,7 @@ from flyteidl.core import identifier_pb2 as flyteidl_dot_core_dot_identifier__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1a\x66lyteidl/admin/agent.proto\x12\x0e\x66lyteidl.admin\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x19\x66lyteidl/core/tasks.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1e\x66lyteidl/core/identifier.proto\"\x98\x05\n\x15TaskExecutionMetadata\x12R\n\x11task_execution_id\x18\x01 \x01(\x0b\x32&.flyteidl.core.TaskExecutionIdentifierR\x0ftaskExecutionId\x12\x1c\n\tnamespace\x18\x02 \x01(\tR\tnamespace\x12I\n\x06labels\x18\x03 \x03(\x0b\x32\x31.flyteidl.admin.TaskExecutionMetadata.LabelsEntryR\x06labels\x12X\n\x0b\x61nnotations\x18\x04 \x03(\x0b\x32\x36.flyteidl.admin.TaskExecutionMetadata.AnnotationsEntryR\x0b\x61nnotations\x12.\n\x13k8s_service_account\x18\x05 \x01(\tR\x11k8sServiceAccount\x12t\n\x15\x65nvironment_variables\x18\x06 \x03(\x0b\x32?.flyteidl.admin.TaskExecutionMetadata.EnvironmentVariablesEntryR\x14\x65nvironmentVariables\x1a\x39\n\x0bLabelsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1a>\n\x10\x41nnotationsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1aG\n\x19\x45nvironmentVariablesEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"\x83\x02\n\x11\x43reateTaskRequest\x12\x31\n\x06inputs\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.LiteralMapR\x06inputs\x12\x37\n\x08template\x18\x02 \x01(\x0b\x32\x1b.flyteidl.core.TaskTemplateR\x08template\x12#\n\routput_prefix\x18\x03 \x01(\tR\x0coutputPrefix\x12]\n\x17task_execution_metadata\x18\x04 \x01(\x0b\x32%.flyteidl.admin.TaskExecutionMetadataR\x15taskExecutionMetadata\"9\n\x12\x43reateTaskResponse\x12#\n\rresource_meta\x18\x01 \x01(\x0cR\x0cresourceMeta\"R\n\x0eGetTaskRequest\x12\x1b\n\ttask_type\x18\x01 \x01(\tR\x08taskType\x12#\n\rresource_meta\x18\x02 \x01(\x0cR\x0cresourceMeta\"G\n\x0fGetTaskResponse\x12\x34\n\x08resource\x18\x01 \x01(\x0b\x32\x18.flyteidl.admin.ResourceR\x08resource\"l\n\x08Resource\x12+\n\x05state\x18\x01 \x01(\x0e\x32\x15.flyteidl.admin.StateR\x05state\x12\x33\n\x07outputs\x18\x02 \x01(\x0b\x32\x19.flyteidl.core.LiteralMapR\x07outputs\"U\n\x11\x44\x65leteTaskRequest\x12\x1b\n\ttask_type\x18\x01 \x01(\tR\x08taskType\x12#\n\rresource_meta\x18\x02 \x01(\x0cR\x0cresourceMeta\"\x14\n\x12\x44\x65leteTaskResponse*^\n\x05State\x12\x15\n\x11RETRYABLE_FAILURE\x10\x00\x12\x15\n\x11PERMANENT_FAILURE\x10\x01\x12\x0b\n\x07PENDING\x10\x02\x12\x0b\n\x07RUNNING\x10\x03\x12\r\n\tSUCCEEDED\x10\x04\x42\xb6\x01\n\x12\x63om.flyteidl.adminB\nAgentProtoP\x01Z;github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/admin\xa2\x02\x03\x46\x41X\xaa\x02\x0e\x46lyteidl.Admin\xca\x02\x0e\x46lyteidl\\Admin\xe2\x02\x1a\x46lyteidl\\Admin\\GPBMetadata\xea\x02\x0f\x46lyteidl::Adminb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1a\x66lyteidl/admin/agent.proto\x12\x0e\x66lyteidl.admin\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x19\x66lyteidl/core/tasks.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1e\x66lyteidl/core/identifier.proto\"\x98\x05\n\x15TaskExecutionMetadata\x12R\n\x11task_execution_id\x18\x01 \x01(\x0b\x32&.flyteidl.core.TaskExecutionIdentifierR\x0ftaskExecutionId\x12\x1c\n\tnamespace\x18\x02 \x01(\tR\tnamespace\x12I\n\x06labels\x18\x03 \x03(\x0b\x32\x31.flyteidl.admin.TaskExecutionMetadata.LabelsEntryR\x06labels\x12X\n\x0b\x61nnotations\x18\x04 \x03(\x0b\x32\x36.flyteidl.admin.TaskExecutionMetadata.AnnotationsEntryR\x0b\x61nnotations\x12.\n\x13k8s_service_account\x18\x05 \x01(\tR\x11k8sServiceAccount\x12t\n\x15\x65nvironment_variables\x18\x06 \x03(\x0b\x32?.flyteidl.admin.TaskExecutionMetadata.EnvironmentVariablesEntryR\x14\x65nvironmentVariables\x1a\x39\n\x0bLabelsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1a>\n\x10\x41nnotationsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1aG\n\x19\x45nvironmentVariablesEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"\x83\x02\n\x11\x43reateTaskRequest\x12\x31\n\x06inputs\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.LiteralMapR\x06inputs\x12\x37\n\x08template\x18\x02 \x01(\x0b\x32\x1b.flyteidl.core.TaskTemplateR\x08template\x12#\n\routput_prefix\x18\x03 \x01(\tR\x0coutputPrefix\x12]\n\x17task_execution_metadata\x18\x04 \x01(\x0b\x32%.flyteidl.admin.TaskExecutionMetadataR\x15taskExecutionMetadata\"9\n\x12\x43reateTaskResponse\x12#\n\rresource_meta\x18\x01 \x01(\x0cR\x0cresourceMeta\"R\n\x0eGetTaskRequest\x12\x1b\n\ttask_type\x18\x01 \x01(\tR\x08taskType\x12#\n\rresource_meta\x18\x02 \x01(\x0cR\x0cresourceMeta\"G\n\x0fGetTaskResponse\x12\x34\n\x08resource\x18\x01 \x01(\x0b\x32\x18.flyteidl.admin.ResourceR\x08resource\"\x86\x01\n\x08Resource\x12+\n\x05state\x18\x01 \x01(\x0e\x32\x15.flyteidl.admin.StateR\x05state\x12\x33\n\x07outputs\x18\x02 \x01(\x0b\x32\x19.flyteidl.core.LiteralMapR\x07outputs\x12\x18\n\x07message\x18\x03 \x01(\tR\x07message\"U\n\x11\x44\x65leteTaskRequest\x12\x1b\n\ttask_type\x18\x01 \x01(\tR\x08taskType\x12#\n\rresource_meta\x18\x02 \x01(\x0cR\x0cresourceMeta\"\x14\n\x12\x44\x65leteTaskResponse*^\n\x05State\x12\x15\n\x11RETRYABLE_FAILURE\x10\x00\x12\x15\n\x11PERMANENT_FAILURE\x10\x01\x12\x0b\n\x07PENDING\x10\x02\x12\x0b\n\x07RUNNING\x10\x03\x12\r\n\tSUCCEEDED\x10\x04\x42\xb6\x01\n\x12\x63om.flyteidl.adminB\nAgentProtoP\x01Z;github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/admin\xa2\x02\x03\x46\x41X\xaa\x02\x0e\x46lyteidl.Admin\xca\x02\x0e\x46lyteidl\\Admin\xe2\x02\x1a\x46lyteidl\\Admin\\GPBMetadata\xea\x02\x0f\x46lyteidl::Adminb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -32,8 +32,8 @@ _TASKEXECUTIONMETADATA_ANNOTATIONSENTRY._serialized_options = b'8\001' _TASKEXECUTIONMETADATA_ENVIRONMENTVARIABLESENTRY._options = None _TASKEXECUTIONMETADATA_ENVIRONMENTVARIABLESENTRY._serialized_options = b'8\001' - _globals['_STATE']._serialized_start=1530 - _globals['_STATE']._serialized_end=1624 + _globals['_STATE']._serialized_start=1557 + _globals['_STATE']._serialized_end=1651 _globals['_TASKEXECUTIONMETADATA']._serialized_start=167 _globals['_TASKEXECUTIONMETADATA']._serialized_end=831 _globals['_TASKEXECUTIONMETADATA_LABELSENTRY']._serialized_start=637 @@ -50,10 +50,10 @@ _globals['_GETTASKREQUEST']._serialized_end=1236 _globals['_GETTASKRESPONSE']._serialized_start=1238 _globals['_GETTASKRESPONSE']._serialized_end=1309 - _globals['_RESOURCE']._serialized_start=1311 - _globals['_RESOURCE']._serialized_end=1419 - _globals['_DELETETASKREQUEST']._serialized_start=1421 - _globals['_DELETETASKREQUEST']._serialized_end=1506 - _globals['_DELETETASKRESPONSE']._serialized_start=1508 - _globals['_DELETETASKRESPONSE']._serialized_end=1528 + _globals['_RESOURCE']._serialized_start=1312 + _globals['_RESOURCE']._serialized_end=1446 + _globals['_DELETETASKREQUEST']._serialized_start=1448 + _globals['_DELETETASKREQUEST']._serialized_end=1533 + _globals['_DELETETASKRESPONSE']._serialized_start=1535 + _globals['_DELETETASKRESPONSE']._serialized_end=1555 # @@protoc_insertion_point(module_scope) diff --git a/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.pyi b/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.pyi index 830158d0b22..6ca7ba23a7f 100644 --- a/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.pyi +++ b/flyteidl/gen/pb_python/flyteidl/admin/agent_pb2.pyi @@ -93,12 +93,14 @@ class GetTaskResponse(_message.Message): def __init__(self, resource: _Optional[_Union[Resource, _Mapping]] = ...) -> None: ... class Resource(_message.Message): - __slots__ = ["state", "outputs"] + __slots__ = ["state", "outputs", "message"] STATE_FIELD_NUMBER: _ClassVar[int] OUTPUTS_FIELD_NUMBER: _ClassVar[int] + MESSAGE_FIELD_NUMBER: _ClassVar[int] state: State outputs: _literals_pb2.LiteralMap - def __init__(self, state: _Optional[_Union[State, str]] = ..., outputs: _Optional[_Union[_literals_pb2.LiteralMap, _Mapping]] = ...) -> None: ... + message: str + def __init__(self, state: _Optional[_Union[State, str]] = ..., outputs: _Optional[_Union[_literals_pb2.LiteralMap, _Mapping]] = ..., message: _Optional[str] = ...) -> None: ... class DeleteTaskRequest(_message.Message): __slots__ = ["task_type", "resource_meta"] diff --git a/flyteidl/gen/pb_rust/flyteidl.admin.rs b/flyteidl/gen/pb_rust/flyteidl.admin.rs index 5ac0ff4471b..759983a507e 100644 --- a/flyteidl/gen/pb_rust/flyteidl.admin.rs +++ b/flyteidl/gen/pb_rust/flyteidl.admin.rs @@ -78,6 +78,9 @@ pub struct Resource { /// +optional #[prost(message, optional, tag="2")] pub outputs: ::core::option::Option, + /// A descriptive message for the current state. e.g. waiting for cluster. + #[prost(string, tag="3")] + pub message: ::prost::alloc::string::String, } /// A message used to delete a task. #[allow(clippy::derive_partial_eq_without_eq)] diff --git a/flyteidl/protos/flyteidl/admin/agent.proto b/flyteidl/protos/flyteidl/admin/agent.proto index d85d4d9a9e7..a4dade0953c 100644 --- a/flyteidl/protos/flyteidl/admin/agent.proto +++ b/flyteidl/protos/flyteidl/admin/agent.proto @@ -73,6 +73,8 @@ message Resource { // Structured dataset pointing to the query result table. // +optional core.LiteralMap outputs = 2; + // A descriptive message for the current state. e.g. waiting for cluster. + string message = 3; } // A message used to delete a task. diff --git a/flyteplugins/go/tasks/pluginmachinery/core/phase.go b/flyteplugins/go/tasks/pluginmachinery/core/phase.go index 4f3836a8b10..3f8c2a0914a 100644 --- a/flyteplugins/go/tasks/pluginmachinery/core/phase.go +++ b/flyteplugins/go/tasks/pluginmachinery/core/phase.go @@ -226,7 +226,6 @@ func PhaseInfoQueuedWithTaskInfo(version uint32, reason string, info *TaskInfo) } func PhaseInfoInitializing(t time.Time, version uint32, reason string, info *TaskInfo) PhaseInfo { - pi := phaseInfo(PhaseInitializing, version, nil, info, false) pi.reason = reason return pi diff --git a/flyteplugins/go/tasks/plugins/webapi/agent/plugin.go b/flyteplugins/go/tasks/plugins/webapi/agent/plugin.go index a9b233a3bbe..c50d3617268 100644 --- a/flyteplugins/go/tasks/plugins/webapi/agent/plugin.go +++ b/flyteplugins/go/tasks/plugins/webapi/agent/plugin.go @@ -5,6 +5,7 @@ import ( "crypto/x509" "encoding/gob" "fmt" + "time" "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/admin" "github.com/flyteorg/flyte/flytestdlib/config" @@ -39,6 +40,7 @@ type Plugin struct { type ResourceWrapper struct { State admin.State Outputs *flyteIdl.LiteralMap + Message string } type ResourceMetaWrapper struct { @@ -141,6 +143,7 @@ func (p Plugin) Get(ctx context.Context, taskCtx webapi.GetContext) (latest weba return ResourceWrapper{ State: res.Resource.State, Outputs: res.Resource.Outputs, + Message: res.Resource.Message, }, nil } @@ -171,6 +174,8 @@ func (p Plugin) Status(ctx context.Context, taskCtx webapi.StatusContext) (phase taskInfo := &core.TaskInfo{} switch resource.State { + case admin.State_PENDING: + return core.PhaseInfoInitializing(time.Now(), core.DefaultPhaseVersion, resource.Message, taskInfo), nil case admin.State_RUNNING: return core.PhaseInfoRunning(core.DefaultPhaseVersion, taskInfo), nil case admin.State_PERMANENT_FAILURE: diff --git a/flyteplugins/go/tasks/plugins/webapi/agent/plugin_test.go b/flyteplugins/go/tasks/plugins/webapi/agent/plugin_test.go index 7a4ea350b68..3a8e759908b 100644 --- a/flyteplugins/go/tasks/plugins/webapi/agent/plugin_test.go +++ b/flyteplugins/go/tasks/plugins/webapi/agent/plugin_test.go @@ -5,12 +5,14 @@ import ( "testing" "time" + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/admin" "github.com/flyteorg/flyte/flytestdlib/config" "google.golang.org/grpc" pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" pluginCoreMocks "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" + webapiPlugin "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/webapi/mocks" "github.com/flyteorg/flyte/flytestdlib/promutils" "github.com/stretchr/testify/assert" ) @@ -99,4 +101,70 @@ func TestPlugin(t *testing.T) { ctx, _ = getFinalContext(context.TODO(), "CreateTask", &Agent{Endpoint: "localhost:8080", Timeouts: map[string]config.Duration{"CreateTask": {Duration: 1 * time.Millisecond}}}) assert.NotEqual(t, context.TODO(), ctx) }) + + t.Run("test PENDING Status", func(t *testing.T) { + taskContext := new(webapiPlugin.StatusContext) + taskContext.On("Resource").Return(ResourceWrapper{ + State: admin.State_PENDING, + Outputs: nil, + Message: "Waiting for cluster", + }) + + phase, err := plugin.Status(context.Background(), taskContext) + assert.NoError(t, err) + assert.Equal(t, pluginsCore.PhaseInitializing, phase.Phase()) + assert.Equal(t, "Waiting for cluster", phase.Reason()) + }) + + t.Run("test RUNNING Status", func(t *testing.T) { + taskContext := new(webapiPlugin.StatusContext) + taskContext.On("Resource").Return(ResourceWrapper{ + State: admin.State_RUNNING, + Outputs: nil, + Message: "Job is running", + }) + + phase, err := plugin.Status(context.Background(), taskContext) + assert.NoError(t, err) + assert.Equal(t, pluginsCore.PhaseRunning, phase.Phase()) + }) + + t.Run("test PERMANENT_FAILURE Status", func(t *testing.T) { + taskContext := new(webapiPlugin.StatusContext) + taskContext.On("Resource").Return(ResourceWrapper{ + State: admin.State_PERMANENT_FAILURE, + Outputs: nil, + Message: "", + }) + + phase, err := plugin.Status(context.Background(), taskContext) + assert.NoError(t, err) + assert.Equal(t, pluginsCore.PhasePermanentFailure, phase.Phase()) + }) + + t.Run("test RETRYABLE_FAILURE Status", func(t *testing.T) { + taskContext := new(webapiPlugin.StatusContext) + taskContext.On("Resource").Return(ResourceWrapper{ + State: admin.State_RETRYABLE_FAILURE, + Outputs: nil, + Message: "", + }) + + phase, err := plugin.Status(context.Background(), taskContext) + assert.NoError(t, err) + assert.Equal(t, pluginsCore.PhaseRetryableFailure, phase.Phase()) + }) + + t.Run("test UNDEFINED Status", func(t *testing.T) { + taskContext := new(webapiPlugin.StatusContext) + taskContext.On("Resource").Return(ResourceWrapper{ + State: 5, + Outputs: nil, + Message: "", + }) + + phase, err := plugin.Status(context.Background(), taskContext) + assert.Error(t, err) + assert.Equal(t, pluginsCore.PhaseUndefined, phase.Phase()) + }) } From 9e8a9094038cade392a6388218a74d31f479c970 Mon Sep 17 00:00:00 2001 From: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> Date: Tue, 10 Oct 2023 14:25:15 -0700 Subject: [PATCH 03/18] Add gh workflow to publish flyteidl to buf (#4178) * Add gh workflow to publish flyteidl to buf Signed-off-by: Eduardo Apolinario * Put in the right place Signed-off-by: Eduardo Apolinario * Do not push to buf on pull requests Signed-off-by: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> * Only push to buf if flyteidl changes Signed-off-by: Eduardo Apolinario --------- Signed-off-by: Eduardo Apolinario Signed-off-by: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> Co-authored-by: Eduardo Apolinario --- .github/workflows/flyteidl-buf-publish.yml | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) create mode 100644 .github/workflows/flyteidl-buf-publish.yml diff --git a/.github/workflows/flyteidl-buf-publish.yml b/.github/workflows/flyteidl-buf-publish.yml new file mode 100644 index 00000000000..3a6915db6f3 --- /dev/null +++ b/.github/workflows/flyteidl-buf-publish.yml @@ -0,0 +1,18 @@ +name: Publish flyteidl Buf Package + +on: + push: + branches: + - master + paths: + - 'flyteidl/**' +jobs: + buf: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: bufbuild/buf-setup-action@v1 + - uses: bufbuild/buf-push-action@v1 + with: + buf_token: ${{ secrets.BUF_TOKEN }} + input: 'flyteidl/protos' From 27bafe01b0e59b1e553fa3ca4ab7b04ada33c0e1 Mon Sep 17 00:00:00 2001 From: Akrati Verma <128223364+blindaks@users.noreply.github.com> Date: Wed, 11 Oct 2023 10:50:35 +0530 Subject: [PATCH 04/18] Update README.md (#4197) * Update README.md Signed-off-by: Akrati Verma <128223364+blindaks@users.noreply.github.com> * Update README.md Signed-off-by: Akrati Verma <128223364+blindaks@users.noreply.github.com> --------- Signed-off-by: Akrati Verma <128223364+blindaks@users.noreply.github.com> --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 8dd663230f4..ba0a5537e0c 100644 --- a/README.md +++ b/README.md @@ -31,8 +31,8 @@ Flyte is an **open-source orchestrator** that facilitates building production-gr OpenSSF Best Practices Flyte Helm Chart - - Flyte Twitter + + Flyte Slack

From 26228bd5c71b445aff5ba2d5375eb0193a1fa1a5 Mon Sep 17 00:00:00 2001 From: Future-Outlier Date: Thu, 12 Oct 2023 03:30:57 +0800 Subject: [PATCH 05/18] Support Evaluator in Kubeflow TensorFlow Training Operator (#4168) --------- Signed-off-by: Future Outlier Co-authored-by: Future Outlier --- .../plugins/kubeflow/tensorflow.pb.cc | 95 +++++- .../flyteidl/plugins/kubeflow/tensorflow.pb.h | 61 ++++ .../pb-cpp/flyteidl/plugins/tensorflow.pb.cc | 66 ++++- .../pb-cpp/flyteidl/plugins/tensorflow.pb.h | 21 ++ .../plugins/kubeflow/tensorflow.pb.go | 66 +++-- .../kubeflow/tensorflow.pb.validate.go | 10 + .../pb-go/flyteidl/plugins/tensorflow.pb.go | 46 +-- .../plugins/tensorflow.pb.validate.go | 2 + .../pb-java/flyteidl/plugins/Tensorflow.java | 128 +++++++- .../flyteidl/plugins/kubeflow/Tensorflow.java | 276 +++++++++++++++++- .../plugins/kubeflow/tensorflow_pb2.py | 8 +- .../plugins/kubeflow/tensorflow_pb2.pyi | 6 +- .../flyteidl/plugins/tensorflow_pb2.py | 4 +- .../flyteidl/plugins/tensorflow_pb2.pyi | 6 +- .../gen/pb_rust/flyteidl.plugins.kubeflow.rs | 3 + flyteidl/gen/pb_rust/flyteidl.plugins.rs | 7 +- .../plugins/kubeflow/tensorflow.proto | 3 + .../protos/flyteidl/plugins/tensorflow.proto | 6 +- .../k8s/kfoperators/common/common_operator.go | 14 +- .../common/common_operator_test.go | 8 +- .../tasks/plugins/k8s/kfoperators/mpi/mpi.go | 2 +- .../plugins/k8s/kfoperators/mpi/mpi_test.go | 2 +- .../k8s/kfoperators/pytorch/pytorch.go | 2 +- .../k8s/kfoperators/pytorch/pytorch_test.go | 4 +- .../k8s/kfoperators/tensorflow/tensorflow.go | 25 +- .../kfoperators/tensorflow/tensorflow_test.go | 87 ++++-- 26 files changed, 820 insertions(+), 138 deletions(-) diff --git a/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.cc index 303cb8003ed..187cc048177 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.cc @@ -83,6 +83,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fplugins_2fkubeflow_2ften PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingTask, ps_replicas_), PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingTask, chief_replicas_), PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingTask, run_policy_), + PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingTask, evaluator_replicas_), ~0u, // no _has_bits_ PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec, _internal_metadata_), ~0u, // no _extensions_ @@ -95,7 +96,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fplugins_2fkubeflow_2ften }; static const ::google::protobuf::internal::MigrationSchema schemas[] PROTOBUF_SECTION_VARIABLE(protodesc_cold) = { { 0, -1, sizeof(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingTask)}, - { 9, -1, sizeof(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec)}, + { 10, -1, sizeof(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec)}, }; static ::google::protobuf::Message const * const file_default_instances[] = { @@ -113,7 +114,7 @@ const char descriptor_table_protodef_flyteidl_2fplugins_2fkubeflow_2ftensorflow_ "\n*flyteidl/plugins/kubeflow/tensorflow.p" "roto\022\031flyteidl.plugins.kubeflow\032\031flyteid" "l/core/tasks.proto\032&flyteidl/plugins/kub" - "eflow/common.proto\"\362\002\n!DistributedTensor" + "eflow/common.proto\"\323\003\n!DistributedTensor" "flowTrainingTask\022\\\n\017worker_replicas\030\001 \001(" "\0132C.flyteidl.plugins.kubeflow.Distribute" "dTensorflowTrainingReplicaSpec\022X\n\013ps_rep" @@ -122,19 +123,21 @@ const char descriptor_table_protodef_flyteidl_2fplugins_2fkubeflow_2ftensorflow_ "\022[\n\016chief_replicas\030\003 \001(\0132C.flyteidl.plug" "ins.kubeflow.DistributedTensorflowTraini" "ngReplicaSpec\0228\n\nrun_policy\030\004 \001(\0132$.flyt" - "eidl.plugins.kubeflow.RunPolicy\"\272\001\n(Dist" - "ributedTensorflowTrainingReplicaSpec\022\020\n\010" - "replicas\030\001 \001(\005\022\r\n\005image\030\002 \001(\t\022+\n\tresourc" - "es\030\003 \001(\0132\030.flyteidl.core.Resources\022@\n\016re" - "start_policy\030\004 \001(\0162(.flyteidl.plugins.ku" - "beflow.RestartPolicyB\?Z=github.com/flyte" - "org/flyte/flyteidl/gen/pb-go/flyteidl/pl" - "uginsb\006proto3" + "eidl.plugins.kubeflow.RunPolicy\022_\n\022evalu" + "ator_replicas\030\005 \001(\0132C.flyteidl.plugins.k" + "ubeflow.DistributedTensorflowTrainingRep" + "licaSpec\"\272\001\n(DistributedTensorflowTraini" + "ngReplicaSpec\022\020\n\010replicas\030\001 \001(\005\022\r\n\005image" + "\030\002 \001(\t\022+\n\tresources\030\003 \001(\0132\030.flyteidl.cor" + "e.Resources\022@\n\016restart_policy\030\004 \001(\0162(.fl" + "yteidl.plugins.kubeflow.RestartPolicyB\?Z" + "=github.com/flyteorg/flyte/flyteidl/gen/" + "pb-go/flyteidl/pluginsb\006proto3" ; ::google::protobuf::internal::DescriptorTable descriptor_table_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto = { false, InitDefaults_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto, descriptor_table_protodef_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto, - "flyteidl/plugins/kubeflow/tensorflow.proto", &assign_descriptors_table_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto, 773, + "flyteidl/plugins/kubeflow/tensorflow.proto", &assign_descriptors_table_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto, 870, }; void AddDescriptors_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto() { @@ -163,6 +166,8 @@ void DistributedTensorflowTrainingTask::InitAsDefaultInstance() { ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec::internal_default_instance()); ::flyteidl::plugins::kubeflow::_DistributedTensorflowTrainingTask_default_instance_._instance.get_mutable()->run_policy_ = const_cast< ::flyteidl::plugins::kubeflow::RunPolicy*>( ::flyteidl::plugins::kubeflow::RunPolicy::internal_default_instance()); + ::flyteidl::plugins::kubeflow::_DistributedTensorflowTrainingTask_default_instance_._instance.get_mutable()->evaluator_replicas_ = const_cast< ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec*>( + ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec::internal_default_instance()); } class DistributedTensorflowTrainingTask::HasBitSetters { public: @@ -170,6 +175,7 @@ class DistributedTensorflowTrainingTask::HasBitSetters { static const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& ps_replicas(const DistributedTensorflowTrainingTask* msg); static const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& chief_replicas(const DistributedTensorflowTrainingTask* msg); static const ::flyteidl::plugins::kubeflow::RunPolicy& run_policy(const DistributedTensorflowTrainingTask* msg); + static const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& evaluator_replicas(const DistributedTensorflowTrainingTask* msg); }; const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& @@ -188,6 +194,10 @@ const ::flyteidl::plugins::kubeflow::RunPolicy& DistributedTensorflowTrainingTask::HasBitSetters::run_policy(const DistributedTensorflowTrainingTask* msg) { return *msg->run_policy_; } +const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& +DistributedTensorflowTrainingTask::HasBitSetters::evaluator_replicas(const DistributedTensorflowTrainingTask* msg) { + return *msg->evaluator_replicas_; +} void DistributedTensorflowTrainingTask::clear_run_policy() { if (GetArenaNoVirtual() == nullptr && run_policy_ != nullptr) { delete run_policy_; @@ -199,6 +209,7 @@ const int DistributedTensorflowTrainingTask::kWorkerReplicasFieldNumber; const int DistributedTensorflowTrainingTask::kPsReplicasFieldNumber; const int DistributedTensorflowTrainingTask::kChiefReplicasFieldNumber; const int DistributedTensorflowTrainingTask::kRunPolicyFieldNumber; +const int DistributedTensorflowTrainingTask::kEvaluatorReplicasFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 DistributedTensorflowTrainingTask::DistributedTensorflowTrainingTask() @@ -230,6 +241,11 @@ DistributedTensorflowTrainingTask::DistributedTensorflowTrainingTask(const Distr } else { run_policy_ = nullptr; } + if (from.has_evaluator_replicas()) { + evaluator_replicas_ = new ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec(*from.evaluator_replicas_); + } else { + evaluator_replicas_ = nullptr; + } // @@protoc_insertion_point(copy_constructor:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask) } @@ -237,8 +253,8 @@ void DistributedTensorflowTrainingTask::SharedCtor() { ::google::protobuf::internal::InitSCC( &scc_info_DistributedTensorflowTrainingTask_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto.base); ::memset(&worker_replicas_, 0, static_cast( - reinterpret_cast(&run_policy_) - - reinterpret_cast(&worker_replicas_)) + sizeof(run_policy_)); + reinterpret_cast(&evaluator_replicas_) - + reinterpret_cast(&worker_replicas_)) + sizeof(evaluator_replicas_)); } DistributedTensorflowTrainingTask::~DistributedTensorflowTrainingTask() { @@ -251,6 +267,7 @@ void DistributedTensorflowTrainingTask::SharedDtor() { if (this != internal_default_instance()) delete ps_replicas_; if (this != internal_default_instance()) delete chief_replicas_; if (this != internal_default_instance()) delete run_policy_; + if (this != internal_default_instance()) delete evaluator_replicas_; } void DistributedTensorflowTrainingTask::SetCachedSize(int size) const { @@ -284,6 +301,10 @@ void DistributedTensorflowTrainingTask::Clear() { delete run_policy_; } run_policy_ = nullptr; + if (GetArenaNoVirtual() == nullptr && evaluator_replicas_ != nullptr) { + delete evaluator_replicas_; + } + evaluator_replicas_ = nullptr; _internal_metadata_.Clear(); } @@ -352,6 +373,19 @@ const char* DistributedTensorflowTrainingTask::_InternalParse(const char* begin, {parser_till_end, object}, ptr - size, ptr)); break; } + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + case 5: { + if (static_cast<::google::protobuf::uint8>(tag) != 42) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec::_InternalParse; + object = msg->mutable_evaluator_replicas(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); + break; + } default: { handle_unusual: if ((tag & 7) == 4 || tag == 0) { @@ -426,6 +460,17 @@ bool DistributedTensorflowTrainingTask::MergePartialFromCodedStream( break; } + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + case 5: { + if (static_cast< ::google::protobuf::uint8>(tag) == (42 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( + input, mutable_evaluator_replicas())); + } else { + goto handle_unusual; + } + break; + } + default: { handle_unusual: if (tag == 0) { @@ -477,6 +522,12 @@ void DistributedTensorflowTrainingTask::SerializeWithCachedSizes( 4, HasBitSetters::run_policy(this), output); } + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + if (this->has_evaluator_replicas()) { + ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( + 5, HasBitSetters::evaluator_replicas(this), output); + } + if (_internal_metadata_.have_unknown_fields()) { ::google::protobuf::internal::WireFormat::SerializeUnknownFields( _internal_metadata_.unknown_fields(), output); @@ -518,6 +569,13 @@ ::google::protobuf::uint8* DistributedTensorflowTrainingTask::InternalSerializeW 4, HasBitSetters::run_policy(this), target); } + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + if (this->has_evaluator_replicas()) { + target = ::google::protobuf::internal::WireFormatLite:: + InternalWriteMessageToArray( + 5, HasBitSetters::evaluator_replicas(this), target); + } + if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); @@ -567,6 +625,13 @@ size_t DistributedTensorflowTrainingTask::ByteSizeLong() const { *run_policy_); } + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + if (this->has_evaluator_replicas()) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::MessageSize( + *evaluator_replicas_); + } + int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); SetCachedSize(cached_size); return total_size; @@ -606,6 +671,9 @@ void DistributedTensorflowTrainingTask::MergeFrom(const DistributedTensorflowTra if (from.has_run_policy()) { mutable_run_policy()->::flyteidl::plugins::kubeflow::RunPolicy::MergeFrom(from.run_policy()); } + if (from.has_evaluator_replicas()) { + mutable_evaluator_replicas()->::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec::MergeFrom(from.evaluator_replicas()); + } } void DistributedTensorflowTrainingTask::CopyFrom(const ::google::protobuf::Message& from) { @@ -637,6 +705,7 @@ void DistributedTensorflowTrainingTask::InternalSwap(DistributedTensorflowTraini swap(ps_replicas_, other->ps_replicas_); swap(chief_replicas_, other->chief_replicas_); swap(run_policy_, other->run_policy_); + swap(evaluator_replicas_, other->evaluator_replicas_); } ::google::protobuf::Metadata DistributedTensorflowTrainingTask::GetMetadata() const { diff --git a/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.h b/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.h index 9839ca48179..4a100b0233f 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.h +++ b/flyteidl/gen/pb-cpp/flyteidl/plugins/kubeflow/tensorflow.pb.h @@ -205,6 +205,15 @@ class DistributedTensorflowTrainingTask final : ::flyteidl::plugins::kubeflow::RunPolicy* mutable_run_policy(); void set_allocated_run_policy(::flyteidl::plugins::kubeflow::RunPolicy* run_policy); + // .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + bool has_evaluator_replicas() const; + void clear_evaluator_replicas(); + static const int kEvaluatorReplicasFieldNumber = 5; + const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& evaluator_replicas() const; + ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* release_evaluator_replicas(); + ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* mutable_evaluator_replicas(); + void set_allocated_evaluator_replicas(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* evaluator_replicas); + // @@protoc_insertion_point(class_scope:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask) private: class HasBitSetters; @@ -214,6 +223,7 @@ class DistributedTensorflowTrainingTask final : ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* ps_replicas_; ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* chief_replicas_; ::flyteidl::plugins::kubeflow::RunPolicy* run_policy_; + ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* evaluator_replicas_; mutable ::google::protobuf::internal::CachedSize _cached_size_; friend struct ::TableStruct_flyteidl_2fplugins_2fkubeflow_2ftensorflow_2eproto; }; @@ -570,6 +580,57 @@ inline void DistributedTensorflowTrainingTask::set_allocated_run_policy(::flytei // @@protoc_insertion_point(field_set_allocated:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask.run_policy) } +// .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; +inline bool DistributedTensorflowTrainingTask::has_evaluator_replicas() const { + return this != internal_default_instance() && evaluator_replicas_ != nullptr; +} +inline void DistributedTensorflowTrainingTask::clear_evaluator_replicas() { + if (GetArenaNoVirtual() == nullptr && evaluator_replicas_ != nullptr) { + delete evaluator_replicas_; + } + evaluator_replicas_ = nullptr; +} +inline const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec& DistributedTensorflowTrainingTask::evaluator_replicas() const { + const ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* p = evaluator_replicas_; + // @@protoc_insertion_point(field_get:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask.evaluator_replicas) + return p != nullptr ? *p : *reinterpret_cast( + &::flyteidl::plugins::kubeflow::_DistributedTensorflowTrainingReplicaSpec_default_instance_); +} +inline ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* DistributedTensorflowTrainingTask::release_evaluator_replicas() { + // @@protoc_insertion_point(field_release:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask.evaluator_replicas) + + ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* temp = evaluator_replicas_; + evaluator_replicas_ = nullptr; + return temp; +} +inline ::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* DistributedTensorflowTrainingTask::mutable_evaluator_replicas() { + + if (evaluator_replicas_ == nullptr) { + auto* p = CreateMaybeMessage<::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec>(GetArenaNoVirtual()); + evaluator_replicas_ = p; + } + // @@protoc_insertion_point(field_mutable:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask.evaluator_replicas) + return evaluator_replicas_; +} +inline void DistributedTensorflowTrainingTask::set_allocated_evaluator_replicas(::flyteidl::plugins::kubeflow::DistributedTensorflowTrainingReplicaSpec* evaluator_replicas) { + ::google::protobuf::Arena* message_arena = GetArenaNoVirtual(); + if (message_arena == nullptr) { + delete evaluator_replicas_; + } + if (evaluator_replicas) { + ::google::protobuf::Arena* submessage_arena = nullptr; + if (message_arena != submessage_arena) { + evaluator_replicas = ::google::protobuf::internal::GetOwnedMessage( + message_arena, evaluator_replicas, submessage_arena); + } + + } else { + + } + evaluator_replicas_ = evaluator_replicas; + // @@protoc_insertion_point(field_set_allocated:flyteidl.plugins.kubeflow.DistributedTensorflowTrainingTask.evaluator_replicas) +} + // ------------------------------------------------------------------- // DistributedTensorflowTrainingReplicaSpec diff --git a/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.cc index 7d01fad6e21..23d343be1ab 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.cc @@ -55,6 +55,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fplugins_2ftensorflow_2ep PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::DistributedTensorflowTrainingTask, workers_), PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::DistributedTensorflowTrainingTask, ps_replicas_), PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::DistributedTensorflowTrainingTask, chief_replicas_), + PROTOBUF_FIELD_OFFSET(::flyteidl::plugins::DistributedTensorflowTrainingTask, evaluator_replicas_), }; static const ::google::protobuf::internal::MigrationSchema schemas[] PROTOBUF_SECTION_VARIABLE(protodesc_cold) = { { 0, -1, sizeof(::flyteidl::plugins::DistributedTensorflowTrainingTask)}, @@ -72,16 +73,17 @@ ::google::protobuf::internal::AssignDescriptorsTable assign_descriptors_table_fl const char descriptor_table_protodef_flyteidl_2fplugins_2ftensorflow_2eproto[] = "\n!flyteidl/plugins/tensorflow.proto\022\020fly" - "teidl.plugins\"a\n!DistributedTensorflowTr" + "teidl.plugins\"}\n!DistributedTensorflowTr" "ainingTask\022\017\n\007workers\030\001 \001(\005\022\023\n\013ps_replic" - "as\030\002 \001(\005\022\026\n\016chief_replicas\030\003 \001(\005B\?Z=gith" - "ub.com/flyteorg/flyte/flyteidl/gen/pb-go" - "/flyteidl/pluginsb\006proto3" + "as\030\002 \001(\005\022\026\n\016chief_replicas\030\003 \001(\005\022\032\n\022eval" + "uator_replicas\030\004 \001(\005B\?Z=github.com/flyte" + "org/flyte/flyteidl/gen/pb-go/flyteidl/pl" + "uginsb\006proto3" ; ::google::protobuf::internal::DescriptorTable descriptor_table_flyteidl_2fplugins_2ftensorflow_2eproto = { false, InitDefaults_flyteidl_2fplugins_2ftensorflow_2eproto, descriptor_table_protodef_flyteidl_2fplugins_2ftensorflow_2eproto, - "flyteidl/plugins/tensorflow.proto", &assign_descriptors_table_flyteidl_2fplugins_2ftensorflow_2eproto, 225, + "flyteidl/plugins/tensorflow.proto", &assign_descriptors_table_flyteidl_2fplugins_2ftensorflow_2eproto, 253, }; void AddDescriptors_flyteidl_2fplugins_2ftensorflow_2eproto() { @@ -108,6 +110,7 @@ class DistributedTensorflowTrainingTask::HasBitSetters { const int DistributedTensorflowTrainingTask::kWorkersFieldNumber; const int DistributedTensorflowTrainingTask::kPsReplicasFieldNumber; const int DistributedTensorflowTrainingTask::kChiefReplicasFieldNumber; +const int DistributedTensorflowTrainingTask::kEvaluatorReplicasFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 DistributedTensorflowTrainingTask::DistributedTensorflowTrainingTask() @@ -120,15 +123,15 @@ DistributedTensorflowTrainingTask::DistributedTensorflowTrainingTask(const Distr _internal_metadata_(nullptr) { _internal_metadata_.MergeFrom(from._internal_metadata_); ::memcpy(&workers_, &from.workers_, - static_cast(reinterpret_cast(&chief_replicas_) - - reinterpret_cast(&workers_)) + sizeof(chief_replicas_)); + static_cast(reinterpret_cast(&evaluator_replicas_) - + reinterpret_cast(&workers_)) + sizeof(evaluator_replicas_)); // @@protoc_insertion_point(copy_constructor:flyteidl.plugins.DistributedTensorflowTrainingTask) } void DistributedTensorflowTrainingTask::SharedCtor() { ::memset(&workers_, 0, static_cast( - reinterpret_cast(&chief_replicas_) - - reinterpret_cast(&workers_)) + sizeof(chief_replicas_)); + reinterpret_cast(&evaluator_replicas_) - + reinterpret_cast(&workers_)) + sizeof(evaluator_replicas_)); } DistributedTensorflowTrainingTask::~DistributedTensorflowTrainingTask() { @@ -155,8 +158,8 @@ void DistributedTensorflowTrainingTask::Clear() { (void) cached_has_bits; ::memset(&workers_, 0, static_cast( - reinterpret_cast(&chief_replicas_) - - reinterpret_cast(&workers_)) + sizeof(chief_replicas_)); + reinterpret_cast(&evaluator_replicas_) - + reinterpret_cast(&workers_)) + sizeof(evaluator_replicas_)); _internal_metadata_.Clear(); } @@ -194,6 +197,13 @@ const char* DistributedTensorflowTrainingTask::_InternalParse(const char* begin, GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); break; } + // int32 evaluator_replicas = 4; + case 4: { + if (static_cast<::google::protobuf::uint8>(tag) != 32) goto handle_unusual; + msg->set_evaluator_replicas(::google::protobuf::internal::ReadVarint(&ptr)); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + break; + } default: { handle_unusual: if ((tag & 7) == 4 || tag == 0) { @@ -260,6 +270,19 @@ bool DistributedTensorflowTrainingTask::MergePartialFromCodedStream( break; } + // int32 evaluator_replicas = 4; + case 4: { + if (static_cast< ::google::protobuf::uint8>(tag) == (32 & 0xFF)) { + + DO_((::google::protobuf::internal::WireFormatLite::ReadPrimitive< + ::google::protobuf::int32, ::google::protobuf::internal::WireFormatLite::TYPE_INT32>( + input, &evaluator_replicas_))); + } else { + goto handle_unusual; + } + break; + } + default: { handle_unusual: if (tag == 0) { @@ -302,6 +325,11 @@ void DistributedTensorflowTrainingTask::SerializeWithCachedSizes( ::google::protobuf::internal::WireFormatLite::WriteInt32(3, this->chief_replicas(), output); } + // int32 evaluator_replicas = 4; + if (this->evaluator_replicas() != 0) { + ::google::protobuf::internal::WireFormatLite::WriteInt32(4, this->evaluator_replicas(), output); + } + if (_internal_metadata_.have_unknown_fields()) { ::google::protobuf::internal::WireFormat::SerializeUnknownFields( _internal_metadata_.unknown_fields(), output); @@ -330,6 +358,11 @@ ::google::protobuf::uint8* DistributedTensorflowTrainingTask::InternalSerializeW target = ::google::protobuf::internal::WireFormatLite::WriteInt32ToArray(3, this->chief_replicas(), target); } + // int32 evaluator_replicas = 4; + if (this->evaluator_replicas() != 0) { + target = ::google::protobuf::internal::WireFormatLite::WriteInt32ToArray(4, this->evaluator_replicas(), target); + } + if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); @@ -372,6 +405,13 @@ size_t DistributedTensorflowTrainingTask::ByteSizeLong() const { this->chief_replicas()); } + // int32 evaluator_replicas = 4; + if (this->evaluator_replicas() != 0) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::Int32Size( + this->evaluator_replicas()); + } + int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); SetCachedSize(cached_size); return total_size; @@ -408,6 +448,9 @@ void DistributedTensorflowTrainingTask::MergeFrom(const DistributedTensorflowTra if (from.chief_replicas() != 0) { set_chief_replicas(from.chief_replicas()); } + if (from.evaluator_replicas() != 0) { + set_evaluator_replicas(from.evaluator_replicas()); + } } void DistributedTensorflowTrainingTask::CopyFrom(const ::google::protobuf::Message& from) { @@ -438,6 +481,7 @@ void DistributedTensorflowTrainingTask::InternalSwap(DistributedTensorflowTraini swap(workers_, other->workers_); swap(ps_replicas_, other->ps_replicas_); swap(chief_replicas_, other->chief_replicas_); + swap(evaluator_replicas_, other->evaluator_replicas_); } ::google::protobuf::Metadata DistributedTensorflowTrainingTask::GetMetadata() const { diff --git a/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.h b/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.h index 613ed31d80f..4150592a60f 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.h +++ b/flyteidl/gen/pb-cpp/flyteidl/plugins/tensorflow.pb.h @@ -178,6 +178,12 @@ class DistributedTensorflowTrainingTask final : ::google::protobuf::int32 chief_replicas() const; void set_chief_replicas(::google::protobuf::int32 value); + // int32 evaluator_replicas = 4; + void clear_evaluator_replicas(); + static const int kEvaluatorReplicasFieldNumber = 4; + ::google::protobuf::int32 evaluator_replicas() const; + void set_evaluator_replicas(::google::protobuf::int32 value); + // @@protoc_insertion_point(class_scope:flyteidl.plugins.DistributedTensorflowTrainingTask) private: class HasBitSetters; @@ -186,6 +192,7 @@ class DistributedTensorflowTrainingTask final : ::google::protobuf::int32 workers_; ::google::protobuf::int32 ps_replicas_; ::google::protobuf::int32 chief_replicas_; + ::google::protobuf::int32 evaluator_replicas_; mutable ::google::protobuf::internal::CachedSize _cached_size_; friend struct ::TableStruct_flyteidl_2fplugins_2ftensorflow_2eproto; }; @@ -242,6 +249,20 @@ inline void DistributedTensorflowTrainingTask::set_chief_replicas(::google::prot // @@protoc_insertion_point(field_set:flyteidl.plugins.DistributedTensorflowTrainingTask.chief_replicas) } +// int32 evaluator_replicas = 4; +inline void DistributedTensorflowTrainingTask::clear_evaluator_replicas() { + evaluator_replicas_ = 0; +} +inline ::google::protobuf::int32 DistributedTensorflowTrainingTask::evaluator_replicas() const { + // @@protoc_insertion_point(field_get:flyteidl.plugins.DistributedTensorflowTrainingTask.evaluator_replicas) + return evaluator_replicas_; +} +inline void DistributedTensorflowTrainingTask::set_evaluator_replicas(::google::protobuf::int32 value) { + + evaluator_replicas_ = value; + // @@protoc_insertion_point(field_set:flyteidl.plugins.DistributedTensorflowTrainingTask.evaluator_replicas) +} + #ifdef __GNUC__ #pragma GCC diagnostic pop #endif // __GNUC__ diff --git a/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.go b/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.go index ccb4eff2ac4..04243dec6e5 100644 --- a/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.go +++ b/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.go @@ -32,10 +32,12 @@ type DistributedTensorflowTrainingTask struct { // RunPolicy encapsulates various runtime policies of the distributed training // job, for example how to clean up resources and how long the job can stay // active. - RunPolicy *RunPolicy `protobuf:"bytes,4,opt,name=run_policy,json=runPolicy,proto3" json:"run_policy,omitempty"` - XXX_NoUnkeyedLiteral struct{} `json:"-"` - XXX_unrecognized []byte `json:"-"` - XXX_sizecache int32 `json:"-"` + RunPolicy *RunPolicy `protobuf:"bytes,4,opt,name=run_policy,json=runPolicy,proto3" json:"run_policy,omitempty"` + // Evaluator replicas spec + EvaluatorReplicas *DistributedTensorflowTrainingReplicaSpec `protobuf:"bytes,5,opt,name=evaluator_replicas,json=evaluatorReplicas,proto3" json:"evaluator_replicas,omitempty"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` } func (m *DistributedTensorflowTrainingTask) Reset() { *m = DistributedTensorflowTrainingTask{} } @@ -91,6 +93,13 @@ func (m *DistributedTensorflowTrainingTask) GetRunPolicy() *RunPolicy { return nil } +func (m *DistributedTensorflowTrainingTask) GetEvaluatorReplicas() *DistributedTensorflowTrainingReplicaSpec { + if m != nil { + return m.EvaluatorReplicas + } + return nil +} + type DistributedTensorflowTrainingReplicaSpec struct { // Number of replicas Replicas int32 `protobuf:"varint,1,opt,name=replicas,proto3" json:"replicas,omitempty"` @@ -170,28 +179,29 @@ func init() { } var fileDescriptor_93de2bd764ddf01a = []byte{ - // 358 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xac, 0x92, 0x41, 0x4b, 0xc3, 0x30, - 0x14, 0xc7, 0x99, 0x73, 0xe2, 0x32, 0x56, 0xa1, 0x78, 0x98, 0x3b, 0xe9, 0x10, 0x19, 0x82, 0x0d, - 0x4c, 0xf0, 0x26, 0x82, 0xf3, 0xae, 0xc4, 0x9d, 0xbc, 0x8c, 0x36, 0x7b, 0xeb, 0x62, 0xdb, 0x24, - 0xbc, 0xa4, 0x8c, 0x7d, 0x23, 0xbf, 0x98, 0xdf, 0x43, 0xd6, 0xac, 0xed, 0x14, 0x36, 0x3c, 0xec, - 0xf6, 0x5e, 0xf3, 0xcf, 0xff, 0xf7, 0x5e, 0xfa, 0x27, 0xb7, 0xf3, 0x74, 0x65, 0x41, 0xcc, 0x52, - 0xaa, 0xd3, 0x3c, 0x16, 0xd2, 0xd0, 0x24, 0x8f, 0x60, 0x9e, 0xaa, 0x25, 0xb5, 0x20, 0x8d, 0xc2, - 0x75, 0x19, 0x68, 0x54, 0x56, 0xf9, 0x17, 0xa5, 0x36, 0xd8, 0x68, 0x83, 0x52, 0xdb, 0xaf, 0x8e, - 0x28, 0x57, 0x08, 0xd4, 0x86, 0x26, 0x31, 0xee, 0x56, 0xff, 0x66, 0x37, 0x81, 0xab, 0x2c, 0x53, - 0xd2, 0xe9, 0x06, 0x5f, 0x4d, 0x72, 0xf5, 0x22, 0x8c, 0x45, 0x11, 0xe5, 0x16, 0x66, 0x93, 0x8a, - 0x3e, 0xc1, 0x50, 0x48, 0x21, 0xe3, 0x49, 0x68, 0x12, 0x3f, 0x25, 0x67, 0x4b, 0x85, 0x09, 0xe0, - 0x14, 0x41, 0xa7, 0x82, 0x87, 0xa6, 0xd7, 0xb8, 0x6c, 0x0c, 0x3b, 0xa3, 0x71, 0xb0, 0x73, 0xba, - 0x60, 0xaf, 0x2d, 0x73, 0x3e, 0xef, 0x1a, 0x38, 0xf3, 0x9c, 0xf7, 0xe6, 0x93, 0xf1, 0x67, 0xa4, - 0xa3, 0x4d, 0x4d, 0x3a, 0x3a, 0x1c, 0x89, 0x68, 0x53, 0x51, 0x3e, 0x89, 0xc7, 0x17, 0x02, 0xe6, - 0x35, 0xa8, 0x79, 0x38, 0x50, 0xb7, 0xb0, 0xae, 0x58, 0x63, 0x42, 0x30, 0x97, 0x53, 0xad, 0x52, - 0xc1, 0x57, 0xbd, 0xe3, 0x82, 0x73, 0xbd, 0x87, 0xc3, 0x72, 0xf9, 0x56, 0x68, 0x59, 0x1b, 0xcb, - 0x72, 0xf0, 0xdd, 0x20, 0xc3, 0xff, 0x0e, 0xe0, 0xf7, 0xc9, 0xe9, 0xaf, 0x5f, 0xd5, 0x62, 0x55, - 0xef, 0x9f, 0x93, 0x96, 0xc8, 0xc2, 0x18, 0x8a, 0x97, 0x6d, 0x33, 0xd7, 0xf8, 0x0f, 0xa4, 0x8d, - 0x60, 0x54, 0x8e, 0x1c, 0xca, 0xa7, 0xe8, 0xd5, 0x23, 0xae, 0x03, 0x16, 0xb0, 0xf2, 0x9c, 0xd5, - 0x52, 0xff, 0x95, 0x78, 0x08, 0xc6, 0x86, 0x68, 0xb7, 0xf7, 0xf3, 0x46, 0xc3, 0x7d, 0xfb, 0xb9, - 0x0b, 0x9b, 0x1d, 0xbb, 0xb8, 0xdd, 0x3e, 0x3f, 0x7d, 0x3c, 0xc6, 0xc2, 0x2e, 0xf2, 0x28, 0xe0, - 0x2a, 0xa3, 0x85, 0x89, 0xc2, 0xd8, 0x15, 0xb4, 0x8a, 0x75, 0x0c, 0x92, 0xea, 0xe8, 0x2e, 0x56, - 0xf4, 0x6f, 0xd2, 0xa3, 0x93, 0x22, 0xda, 0xf7, 0x3f, 0x01, 0x00, 0x00, 0xff, 0xff, 0x15, 0x1b, - 0x83, 0x24, 0x66, 0x03, 0x00, 0x00, + // 382 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xac, 0x93, 0xc1, 0x6a, 0xe3, 0x30, + 0x10, 0x86, 0xc9, 0x6e, 0xb2, 0x6c, 0x14, 0xe2, 0x65, 0xc5, 0x1e, 0xb2, 0x39, 0xed, 0x86, 0x65, + 0x09, 0x85, 0x5a, 0x90, 0x42, 0x6f, 0xa5, 0xd0, 0xf4, 0xde, 0xa2, 0xe6, 0xd4, 0x4b, 0x90, 0x15, + 0xc5, 0x51, 0x2d, 0x4b, 0x62, 0x24, 0x35, 0xe4, 0x5d, 0xfa, 0x7a, 0x7d, 0x8f, 0x12, 0x3b, 0xb6, + 0xd3, 0x42, 0x42, 0x0f, 0xb9, 0xcd, 0x58, 0xff, 0xfc, 0xdf, 0x78, 0xa4, 0x41, 0x67, 0x4b, 0xb5, + 0xf1, 0x42, 0x2e, 0x14, 0xb1, 0x2a, 0xa4, 0x52, 0x3b, 0x92, 0x85, 0x44, 0x2c, 0x95, 0x59, 0x13, + 0x2f, 0xb4, 0x33, 0xb0, 0x0d, 0x63, 0x0b, 0xc6, 0x1b, 0xfc, 0xbb, 0xd2, 0xc6, 0x3b, 0x6d, 0x5c, + 0x69, 0x87, 0xf5, 0x11, 0xe1, 0x06, 0x04, 0xf1, 0xcc, 0x65, 0xae, 0xac, 0x1a, 0xfe, 0x3f, 0x4c, + 0xe0, 0x26, 0xcf, 0x8d, 0x2e, 0x75, 0xa3, 0x97, 0x36, 0xfa, 0x7b, 0x2b, 0x9d, 0x07, 0x99, 0x04, + 0x2f, 0x16, 0xb3, 0x9a, 0x3e, 0x03, 0x26, 0xb5, 0xd4, 0xe9, 0x8c, 0xb9, 0x0c, 0x2b, 0xf4, 0x63, + 0x6d, 0x20, 0x13, 0x30, 0x07, 0x61, 0x95, 0xe4, 0xcc, 0x0d, 0x5a, 0x7f, 0x5a, 0xe3, 0xde, 0x64, + 0x1a, 0x1f, 0xec, 0x2e, 0x3e, 0x6a, 0x4b, 0x4b, 0x9f, 0x07, 0x2b, 0x38, 0x8d, 0x4a, 0xef, 0xdd, + 0x27, 0x87, 0x17, 0xa8, 0x67, 0x5d, 0x43, 0xfa, 0x72, 0x3a, 0x12, 0xb2, 0xae, 0xa6, 0x3c, 0xa1, + 0x88, 0xaf, 0xa4, 0x58, 0x36, 0xa0, 0xaf, 0xa7, 0x03, 0xf5, 0x0b, 0xeb, 0x9a, 0x35, 0x45, 0x08, + 0x82, 0x9e, 0x5b, 0xa3, 0x24, 0xdf, 0x0c, 0xda, 0x05, 0xe7, 0xdf, 0x11, 0x0e, 0x0d, 0xfa, 0xbe, + 0xd0, 0xd2, 0x2e, 0x54, 0x21, 0x06, 0x84, 0xc5, 0x33, 0x53, 0x81, 0x79, 0xb3, 0x77, 0x0f, 0x9d, + 0xd3, 0x35, 0xfd, 0xb3, 0xb6, 0xaf, 0x1a, 0x1f, 0xbd, 0xb6, 0xd0, 0xf8, 0xb3, 0xf5, 0x78, 0x88, + 0xbe, 0xbf, 0x7b, 0x1e, 0x1d, 0x5a, 0xe7, 0xf8, 0x17, 0xea, 0xc8, 0x9c, 0xa5, 0xa2, 0xb8, 0xcd, + 0x2e, 0x2d, 0x13, 0x7c, 0x89, 0xba, 0x20, 0x9c, 0x09, 0xc0, 0x45, 0x35, 0xfe, 0x41, 0xf3, 0x27, + 0xdb, 0x47, 0x1d, 0xd3, 0xea, 0x9c, 0x36, 0x52, 0x7c, 0x87, 0x22, 0x10, 0xce, 0x33, 0xf0, 0xfb, + 0x33, 0x8d, 0x26, 0xe3, 0x63, 0x33, 0x2d, 0x0b, 0x76, 0x73, 0xed, 0xc3, 0x7e, 0x7a, 0x73, 0xfd, + 0x78, 0x95, 0x4a, 0xbf, 0x0a, 0x49, 0xcc, 0x4d, 0x4e, 0x0a, 0x13, 0x03, 0x69, 0x19, 0x90, 0x7a, + 0x95, 0x52, 0xa1, 0x89, 0x4d, 0xce, 0x53, 0x43, 0x3e, 0x6e, 0x57, 0xf2, 0xad, 0x58, 0xa7, 0x8b, + 0xb7, 0x00, 0x00, 0x00, 0xff, 0xff, 0x74, 0xcf, 0x60, 0xf4, 0xda, 0x03, 0x00, 0x00, } diff --git a/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.validate.go b/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.validate.go index 098b4dc7cf4..397b3a813b4 100644 --- a/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.validate.go +++ b/flyteidl/gen/pb-go/flyteidl/plugins/kubeflow/tensorflow.pb.validate.go @@ -84,6 +84,16 @@ func (m *DistributedTensorflowTrainingTask) Validate() error { } } + if v, ok := interface{}(m.GetEvaluatorReplicas()).(interface{ Validate() error }); ok { + if err := v.Validate(); err != nil { + return DistributedTensorflowTrainingTaskValidationError{ + field: "EvaluatorReplicas", + reason: "embedded message failed validation", + cause: err, + } + } + } + return nil } diff --git a/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.go b/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.go index d9f1006792e..a07ff3feebb 100644 --- a/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.go +++ b/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.go @@ -22,11 +22,15 @@ const _ = proto.ProtoPackageIsVersion3 // please upgrade the proto package // Custom proto for plugin that enables distributed training using https://github.com/kubeflow/tf-operator type DistributedTensorflowTrainingTask struct { - // number of worker, ps, chief replicas spawned in the cluster for this job + // number of worker replicas spawned in the cluster for this job Workers int32 `protobuf:"varint,1,opt,name=workers,proto3" json:"workers,omitempty"` // PS -> Parameter server - PsReplicas int32 `protobuf:"varint,2,opt,name=ps_replicas,json=psReplicas,proto3" json:"ps_replicas,omitempty"` - ChiefReplicas int32 `protobuf:"varint,3,opt,name=chief_replicas,json=chiefReplicas,proto3" json:"chief_replicas,omitempty"` + // number of ps replicas spawned in the cluster for this job + PsReplicas int32 `protobuf:"varint,2,opt,name=ps_replicas,json=psReplicas,proto3" json:"ps_replicas,omitempty"` + // number of chief replicas spawned in the cluster for this job + ChiefReplicas int32 `protobuf:"varint,3,opt,name=chief_replicas,json=chiefReplicas,proto3" json:"chief_replicas,omitempty"` + // number of evaluator replicas spawned in the cluster for this job + EvaluatorReplicas int32 `protobuf:"varint,4,opt,name=evaluator_replicas,json=evaluatorReplicas,proto3" json:"evaluator_replicas,omitempty"` XXX_NoUnkeyedLiteral struct{} `json:"-"` XXX_unrecognized []byte `json:"-"` XXX_sizecache int32 `json:"-"` @@ -78,6 +82,13 @@ func (m *DistributedTensorflowTrainingTask) GetChiefReplicas() int32 { return 0 } +func (m *DistributedTensorflowTrainingTask) GetEvaluatorReplicas() int32 { + if m != nil { + return m.EvaluatorReplicas + } + return 0 +} + func init() { proto.RegisterType((*DistributedTensorflowTrainingTask)(nil), "flyteidl.plugins.DistributedTensorflowTrainingTask") } @@ -85,18 +96,19 @@ func init() { func init() { proto.RegisterFile("flyteidl/plugins/tensorflow.proto", fileDescriptor_8da02783614e1bcc) } var fileDescriptor_8da02783614e1bcc = []byte{ - // 203 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x5c, 0x8f, 0xc1, 0x4a, 0xc4, 0x30, - 0x10, 0x86, 0xa9, 0xa2, 0x42, 0x44, 0x91, 0x9c, 0x7a, 0xd3, 0x0a, 0x82, 0x17, 0x9b, 0x83, 0x67, - 0x11, 0xc4, 0x27, 0x28, 0x3d, 0x79, 0x91, 0xa6, 0x4d, 0xd3, 0xa1, 0xd9, 0x4c, 0x98, 0x49, 0x29, - 0xfb, 0x00, 0xfb, 0xde, 0x0b, 0xd9, 0xb6, 0x0b, 0x7b, 0x9b, 0xf9, 0xe7, 0x1b, 0xf8, 0x3f, 0x51, - 0xf4, 0x6e, 0x1f, 0x0d, 0x74, 0x4e, 0x05, 0x37, 0x59, 0xf0, 0xac, 0xa2, 0xf1, 0x8c, 0xd4, 0x3b, - 0x9c, 0xcb, 0x40, 0x18, 0x51, 0x3e, 0xad, 0x48, 0xb9, 0x20, 0xaf, 0x87, 0x4c, 0x14, 0xbf, 0xc0, - 0x91, 0x40, 0x4f, 0xd1, 0x74, 0xf5, 0xf6, 0x51, 0x53, 0x03, 0x1e, 0xbc, 0xad, 0x1b, 0x1e, 0x65, - 0x2e, 0xee, 0x66, 0xa4, 0xd1, 0x10, 0xe7, 0xd9, 0x4b, 0xf6, 0x7e, 0x53, 0xad, 0xab, 0x7c, 0x16, - 0xf7, 0x81, 0xff, 0xc9, 0x04, 0x07, 0x6d, 0xc3, 0xf9, 0x55, 0xba, 0x8a, 0xc0, 0xd5, 0x92, 0xc8, - 0x37, 0xf1, 0xd8, 0x0e, 0x60, 0xfa, 0x33, 0x73, 0x9d, 0x98, 0x87, 0x94, 0xae, 0xd8, 0xcf, 0xf7, - 0xdf, 0x97, 0x85, 0x38, 0x4c, 0xba, 0x6c, 0x71, 0xa7, 0x52, 0x4d, 0x24, 0x7b, 0x1a, 0xd4, 0x26, - 0x66, 0x8d, 0x57, 0x41, 0x7f, 0x58, 0x54, 0x97, 0xae, 0xfa, 0x36, 0x19, 0x7e, 0x1e, 0x03, 0x00, - 0x00, 0xff, 0xff, 0x55, 0x40, 0x42, 0x06, 0x06, 0x01, 0x00, 0x00, + // 220 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x5c, 0xd0, 0xbf, 0x4a, 0x04, 0x31, + 0x10, 0x06, 0x70, 0xd6, 0xbf, 0x10, 0x51, 0x34, 0xd5, 0x76, 0x7a, 0x82, 0x60, 0x73, 0x9b, 0xc2, + 0x5a, 0x04, 0xf1, 0x09, 0x8e, 0xad, 0x6c, 0x24, 0xbb, 0x37, 0x9b, 0x1b, 0x2e, 0x66, 0xc2, 0x4c, + 0xe2, 0xe1, 0x7b, 0xf9, 0x80, 0x42, 0xbc, 0xcd, 0x81, 0x5d, 0xf2, 0x7d, 0xbf, 0x14, 0xf9, 0xd4, + 0x62, 0xf2, 0xdf, 0x09, 0x70, 0xed, 0x4d, 0xf4, 0xd9, 0x61, 0x10, 0x93, 0x20, 0x08, 0xf1, 0xe4, + 0x69, 0xd7, 0x45, 0xa6, 0x44, 0xfa, 0x7a, 0x26, 0xdd, 0x9e, 0xdc, 0xff, 0x34, 0x6a, 0xf1, 0x86, + 0x92, 0x18, 0x87, 0x9c, 0x60, 0xdd, 0xd7, 0x17, 0x3d, 0x5b, 0x0c, 0x18, 0x5c, 0x6f, 0x65, 0xab, + 0x5b, 0x75, 0xbe, 0x23, 0xde, 0x02, 0x4b, 0xdb, 0xdc, 0x35, 0x8f, 0xa7, 0xab, 0xf9, 0xaa, 0x6f, + 0xd5, 0x45, 0x94, 0x0f, 0x86, 0xe8, 0x71, 0xb4, 0xd2, 0x1e, 0x95, 0x56, 0x45, 0x59, 0xed, 0x13, + 0xfd, 0xa0, 0xae, 0xc6, 0x0d, 0xc2, 0x74, 0x30, 0xc7, 0xc5, 0x5c, 0x96, 0xb4, 0xb2, 0xa5, 0xd2, + 0xf0, 0x65, 0x7d, 0xb6, 0x89, 0xf8, 0x40, 0x4f, 0x0a, 0xbd, 0xa9, 0xcd, 0xcc, 0x5f, 0x5f, 0xde, + 0x9f, 0x1d, 0xa6, 0x4d, 0x1e, 0xba, 0x91, 0x3e, 0x4d, 0xf9, 0x15, 0xb1, 0xfb, 0x3b, 0x98, 0xba, + 0x83, 0x83, 0x60, 0xe2, 0xb0, 0x74, 0x64, 0xfe, 0x4f, 0x33, 0x9c, 0x95, 0x41, 0x9e, 0x7e, 0x03, + 0x00, 0x00, 0xff, 0xff, 0xa6, 0x22, 0x34, 0xcf, 0x35, 0x01, 0x00, 0x00, } diff --git a/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.validate.go b/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.validate.go index ed7a8eeb80e..00db969ce69 100644 --- a/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.validate.go +++ b/flyteidl/gen/pb-go/flyteidl/plugins/tensorflow.pb.validate.go @@ -50,6 +50,8 @@ func (m *DistributedTensorflowTrainingTask) Validate() error { // no validation rules for ChiefReplicas + // no validation rules for EvaluatorReplicas + return nil } diff --git a/flyteidl/gen/pb-java/flyteidl/plugins/Tensorflow.java b/flyteidl/gen/pb-java/flyteidl/plugins/Tensorflow.java index 14ab5db6bdd..2353f8b78be 100644 --- a/flyteidl/gen/pb-java/flyteidl/plugins/Tensorflow.java +++ b/flyteidl/gen/pb-java/flyteidl/plugins/Tensorflow.java @@ -20,7 +20,7 @@ public interface DistributedTensorflowTrainingTaskOrBuilder extends /** *
-     * number of worker, ps, chief replicas spawned in the cluster for this job
+     * number of worker replicas spawned in the cluster for this job
      * 
* * int32 workers = 1; @@ -30,6 +30,7 @@ public interface DistributedTensorflowTrainingTaskOrBuilder extends /** *
      * PS -> Parameter server
+     * number of ps replicas spawned in the cluster for this job
      * 
* * int32 ps_replicas = 2; @@ -37,9 +38,22 @@ public interface DistributedTensorflowTrainingTaskOrBuilder extends int getPsReplicas(); /** + *
+     * number of chief replicas spawned in the cluster for this job
+     * 
+ * * int32 chief_replicas = 3; */ int getChiefReplicas(); + + /** + *
+     * number of evaluator replicas spawned in the cluster for this job
+     * 
+ * + * int32 evaluator_replicas = 4; + */ + int getEvaluatorReplicas(); } /** *
@@ -99,6 +113,11 @@ private DistributedTensorflowTrainingTask(
               chiefReplicas_ = input.readInt32();
               break;
             }
+            case 32: {
+
+              evaluatorReplicas_ = input.readInt32();
+              break;
+            }
             default: {
               if (!parseUnknownField(
                   input, unknownFields, extensionRegistry, tag)) {
@@ -135,7 +154,7 @@ private DistributedTensorflowTrainingTask(
     private int workers_;
     /**
      * 
-     * number of worker, ps, chief replicas spawned in the cluster for this job
+     * number of worker replicas spawned in the cluster for this job
      * 
* * int32 workers = 1; @@ -149,6 +168,7 @@ public int getWorkers() { /** *
      * PS -> Parameter server
+     * number of ps replicas spawned in the cluster for this job
      * 
* * int32 ps_replicas = 2; @@ -160,12 +180,29 @@ public int getPsReplicas() { public static final int CHIEF_REPLICAS_FIELD_NUMBER = 3; private int chiefReplicas_; /** + *
+     * number of chief replicas spawned in the cluster for this job
+     * 
+ * * int32 chief_replicas = 3; */ public int getChiefReplicas() { return chiefReplicas_; } + public static final int EVALUATOR_REPLICAS_FIELD_NUMBER = 4; + private int evaluatorReplicas_; + /** + *
+     * number of evaluator replicas spawned in the cluster for this job
+     * 
+ * + * int32 evaluator_replicas = 4; + */ + public int getEvaluatorReplicas() { + return evaluatorReplicas_; + } + private byte memoizedIsInitialized = -1; @java.lang.Override public final boolean isInitialized() { @@ -189,6 +226,9 @@ public void writeTo(com.google.protobuf.CodedOutputStream output) if (chiefReplicas_ != 0) { output.writeInt32(3, chiefReplicas_); } + if (evaluatorReplicas_ != 0) { + output.writeInt32(4, evaluatorReplicas_); + } unknownFields.writeTo(output); } @@ -210,6 +250,10 @@ public int getSerializedSize() { size += com.google.protobuf.CodedOutputStream .computeInt32Size(3, chiefReplicas_); } + if (evaluatorReplicas_ != 0) { + size += com.google.protobuf.CodedOutputStream + .computeInt32Size(4, evaluatorReplicas_); + } size += unknownFields.getSerializedSize(); memoizedSize = size; return size; @@ -231,6 +275,8 @@ public boolean equals(final java.lang.Object obj) { != other.getPsReplicas()) return false; if (getChiefReplicas() != other.getChiefReplicas()) return false; + if (getEvaluatorReplicas() + != other.getEvaluatorReplicas()) return false; if (!unknownFields.equals(other.unknownFields)) return false; return true; } @@ -248,6 +294,8 @@ public int hashCode() { hash = (53 * hash) + getPsReplicas(); hash = (37 * hash) + CHIEF_REPLICAS_FIELD_NUMBER; hash = (53 * hash) + getChiefReplicas(); + hash = (37 * hash) + EVALUATOR_REPLICAS_FIELD_NUMBER; + hash = (53 * hash) + getEvaluatorReplicas(); hash = (29 * hash) + unknownFields.hashCode(); memoizedHashCode = hash; return hash; @@ -391,6 +439,8 @@ public Builder clear() { chiefReplicas_ = 0; + evaluatorReplicas_ = 0; + return this; } @@ -420,6 +470,7 @@ public flyteidl.plugins.Tensorflow.DistributedTensorflowTrainingTask buildPartia result.workers_ = workers_; result.psReplicas_ = psReplicas_; result.chiefReplicas_ = chiefReplicas_; + result.evaluatorReplicas_ = evaluatorReplicas_; onBuilt(); return result; } @@ -477,6 +528,9 @@ public Builder mergeFrom(flyteidl.plugins.Tensorflow.DistributedTensorflowTraini if (other.getChiefReplicas() != 0) { setChiefReplicas(other.getChiefReplicas()); } + if (other.getEvaluatorReplicas() != 0) { + setEvaluatorReplicas(other.getEvaluatorReplicas()); + } this.mergeUnknownFields(other.unknownFields); onChanged(); return this; @@ -509,7 +563,7 @@ public Builder mergeFrom( private int workers_ ; /** *
-       * number of worker, ps, chief replicas spawned in the cluster for this job
+       * number of worker replicas spawned in the cluster for this job
        * 
* * int32 workers = 1; @@ -519,7 +573,7 @@ public int getWorkers() { } /** *
-       * number of worker, ps, chief replicas spawned in the cluster for this job
+       * number of worker replicas spawned in the cluster for this job
        * 
* * int32 workers = 1; @@ -532,7 +586,7 @@ public Builder setWorkers(int value) { } /** *
-       * number of worker, ps, chief replicas spawned in the cluster for this job
+       * number of worker replicas spawned in the cluster for this job
        * 
* * int32 workers = 1; @@ -548,6 +602,7 @@ public Builder clearWorkers() { /** *
        * PS -> Parameter server
+       * number of ps replicas spawned in the cluster for this job
        * 
* * int32 ps_replicas = 2; @@ -558,6 +613,7 @@ public int getPsReplicas() { /** *
        * PS -> Parameter server
+       * number of ps replicas spawned in the cluster for this job
        * 
* * int32 ps_replicas = 2; @@ -571,6 +627,7 @@ public Builder setPsReplicas(int value) { /** *
        * PS -> Parameter server
+       * number of ps replicas spawned in the cluster for this job
        * 
* * int32 ps_replicas = 2; @@ -584,12 +641,20 @@ public Builder clearPsReplicas() { private int chiefReplicas_ ; /** + *
+       * number of chief replicas spawned in the cluster for this job
+       * 
+ * * int32 chief_replicas = 3; */ public int getChiefReplicas() { return chiefReplicas_; } /** + *
+       * number of chief replicas spawned in the cluster for this job
+       * 
+ * * int32 chief_replicas = 3; */ public Builder setChiefReplicas(int value) { @@ -599,6 +664,10 @@ public Builder setChiefReplicas(int value) { return this; } /** + *
+       * number of chief replicas spawned in the cluster for this job
+       * 
+ * * int32 chief_replicas = 3; */ public Builder clearChiefReplicas() { @@ -607,6 +676,44 @@ public Builder clearChiefReplicas() { onChanged(); return this; } + + private int evaluatorReplicas_ ; + /** + *
+       * number of evaluator replicas spawned in the cluster for this job
+       * 
+ * + * int32 evaluator_replicas = 4; + */ + public int getEvaluatorReplicas() { + return evaluatorReplicas_; + } + /** + *
+       * number of evaluator replicas spawned in the cluster for this job
+       * 
+ * + * int32 evaluator_replicas = 4; + */ + public Builder setEvaluatorReplicas(int value) { + + evaluatorReplicas_ = value; + onChanged(); + return this; + } + /** + *
+       * number of evaluator replicas spawned in the cluster for this job
+       * 
+ * + * int32 evaluator_replicas = 4; + */ + public Builder clearEvaluatorReplicas() { + + evaluatorReplicas_ = 0; + onChanged(); + return this; + } @java.lang.Override public final Builder setUnknownFields( final com.google.protobuf.UnknownFieldSet unknownFields) { @@ -675,11 +782,12 @@ public flyteidl.plugins.Tensorflow.DistributedTensorflowTrainingTask getDefaultI static { java.lang.String[] descriptorData = { "\n!flyteidl/plugins/tensorflow.proto\022\020fly" + - "teidl.plugins\"a\n!DistributedTensorflowTr" + + "teidl.plugins\"}\n!DistributedTensorflowTr" + "ainingTask\022\017\n\007workers\030\001 \001(\005\022\023\n\013ps_replic" + - "as\030\002 \001(\005\022\026\n\016chief_replicas\030\003 \001(\005B?Z=gith" + - "ub.com/flyteorg/flyte/flyteidl/gen/pb-go" + - "/flyteidl/pluginsb\006proto3" + "as\030\002 \001(\005\022\026\n\016chief_replicas\030\003 \001(\005\022\032\n\022eval" + + "uator_replicas\030\004 \001(\005B?Z=github.com/flyte" + + "org/flyte/flyteidl/gen/pb-go/flyteidl/pl" + + "uginsb\006proto3" }; com.google.protobuf.Descriptors.FileDescriptor.InternalDescriptorAssigner assigner = new com.google.protobuf.Descriptors.FileDescriptor. InternalDescriptorAssigner() { @@ -698,7 +806,7 @@ public com.google.protobuf.ExtensionRegistry assignDescriptors( internal_static_flyteidl_plugins_DistributedTensorflowTrainingTask_fieldAccessorTable = new com.google.protobuf.GeneratedMessageV3.FieldAccessorTable( internal_static_flyteidl_plugins_DistributedTensorflowTrainingTask_descriptor, - new java.lang.String[] { "Workers", "PsReplicas", "ChiefReplicas", }); + new java.lang.String[] { "Workers", "PsReplicas", "ChiefReplicas", "EvaluatorReplicas", }); } // @@protoc_insertion_point(outer_class_scope) diff --git a/flyteidl/gen/pb-java/flyteidl/plugins/kubeflow/Tensorflow.java b/flyteidl/gen/pb-java/flyteidl/plugins/kubeflow/Tensorflow.java index ec763705c8a..324f3e5d01d 100644 --- a/flyteidl/gen/pb-java/flyteidl/plugins/kubeflow/Tensorflow.java +++ b/flyteidl/gen/pb-java/flyteidl/plugins/kubeflow/Tensorflow.java @@ -123,6 +123,31 @@ public interface DistributedTensorflowTrainingTaskOrBuilder extends * .flyteidl.plugins.kubeflow.RunPolicy run_policy = 4; */ flyteidl.plugins.kubeflow.Common.RunPolicyOrBuilder getRunPolicyOrBuilder(); + + /** + *
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + boolean hasEvaluatorReplicas(); + /** + *
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec getEvaluatorReplicas(); + /** + *
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder getEvaluatorReplicasOrBuilder(); } /** *
@@ -219,6 +244,19 @@ private DistributedTensorflowTrainingTask(
 
               break;
             }
+            case 42: {
+              flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder subBuilder = null;
+              if (evaluatorReplicas_ != null) {
+                subBuilder = evaluatorReplicas_.toBuilder();
+              }
+              evaluatorReplicas_ = input.readMessage(flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.parser(), extensionRegistry);
+              if (subBuilder != null) {
+                subBuilder.mergeFrom(evaluatorReplicas_);
+                evaluatorReplicas_ = subBuilder.buildPartial();
+              }
+
+              break;
+            }
             default: {
               if (!parseUnknownField(
                   input, unknownFields, extensionRegistry, tag)) {
@@ -389,6 +427,39 @@ public flyteidl.plugins.kubeflow.Common.RunPolicyOrBuilder getRunPolicyOrBuilder
       return getRunPolicy();
     }
 
+    public static final int EVALUATOR_REPLICAS_FIELD_NUMBER = 5;
+    private flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec evaluatorReplicas_;
+    /**
+     * 
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public boolean hasEvaluatorReplicas() { + return evaluatorReplicas_ != null; + } + /** + *
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec getEvaluatorReplicas() { + return evaluatorReplicas_ == null ? flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.getDefaultInstance() : evaluatorReplicas_; + } + /** + *
+     * Evaluator replicas spec
+     * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder getEvaluatorReplicasOrBuilder() { + return getEvaluatorReplicas(); + } + private byte memoizedIsInitialized = -1; @java.lang.Override public final boolean isInitialized() { @@ -415,6 +486,9 @@ public void writeTo(com.google.protobuf.CodedOutputStream output) if (runPolicy_ != null) { output.writeMessage(4, getRunPolicy()); } + if (evaluatorReplicas_ != null) { + output.writeMessage(5, getEvaluatorReplicas()); + } unknownFields.writeTo(output); } @@ -440,6 +514,10 @@ public int getSerializedSize() { size += com.google.protobuf.CodedOutputStream .computeMessageSize(4, getRunPolicy()); } + if (evaluatorReplicas_ != null) { + size += com.google.protobuf.CodedOutputStream + .computeMessageSize(5, getEvaluatorReplicas()); + } size += unknownFields.getSerializedSize(); memoizedSize = size; return size; @@ -475,6 +553,11 @@ public boolean equals(final java.lang.Object obj) { if (!getRunPolicy() .equals(other.getRunPolicy())) return false; } + if (hasEvaluatorReplicas() != other.hasEvaluatorReplicas()) return false; + if (hasEvaluatorReplicas()) { + if (!getEvaluatorReplicas() + .equals(other.getEvaluatorReplicas())) return false; + } if (!unknownFields.equals(other.unknownFields)) return false; return true; } @@ -502,6 +585,10 @@ public int hashCode() { hash = (37 * hash) + RUN_POLICY_FIELD_NUMBER; hash = (53 * hash) + getRunPolicy().hashCode(); } + if (hasEvaluatorReplicas()) { + hash = (37 * hash) + EVALUATOR_REPLICAS_FIELD_NUMBER; + hash = (53 * hash) + getEvaluatorReplicas().hashCode(); + } hash = (29 * hash) + unknownFields.hashCode(); memoizedHashCode = hash; return hash; @@ -663,6 +750,12 @@ public Builder clear() { runPolicy_ = null; runPolicyBuilder_ = null; } + if (evaluatorReplicasBuilder_ == null) { + evaluatorReplicas_ = null; + } else { + evaluatorReplicas_ = null; + evaluatorReplicasBuilder_ = null; + } return this; } @@ -709,6 +802,11 @@ public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingTask bu } else { result.runPolicy_ = runPolicyBuilder_.build(); } + if (evaluatorReplicasBuilder_ == null) { + result.evaluatorReplicas_ = evaluatorReplicas_; + } else { + result.evaluatorReplicas_ = evaluatorReplicasBuilder_.build(); + } onBuilt(); return result; } @@ -769,6 +867,9 @@ public Builder mergeFrom(flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorf if (other.hasRunPolicy()) { mergeRunPolicy(other.getRunPolicy()); } + if (other.hasEvaluatorReplicas()) { + mergeEvaluatorReplicas(other.getEvaluatorReplicas()); + } this.mergeUnknownFields(other.unknownFields); onChanged(); return this; @@ -1427,6 +1528,159 @@ public flyteidl.plugins.kubeflow.Common.RunPolicyOrBuilder getRunPolicyOrBuilder } return runPolicyBuilder_; } + + private flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec evaluatorReplicas_; + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder> evaluatorReplicasBuilder_; + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public boolean hasEvaluatorReplicas() { + return evaluatorReplicasBuilder_ != null || evaluatorReplicas_ != null; + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec getEvaluatorReplicas() { + if (evaluatorReplicasBuilder_ == null) { + return evaluatorReplicas_ == null ? flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.getDefaultInstance() : evaluatorReplicas_; + } else { + return evaluatorReplicasBuilder_.getMessage(); + } + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public Builder setEvaluatorReplicas(flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec value) { + if (evaluatorReplicasBuilder_ == null) { + if (value == null) { + throw new NullPointerException(); + } + evaluatorReplicas_ = value; + onChanged(); + } else { + evaluatorReplicasBuilder_.setMessage(value); + } + + return this; + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public Builder setEvaluatorReplicas( + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder builderForValue) { + if (evaluatorReplicasBuilder_ == null) { + evaluatorReplicas_ = builderForValue.build(); + onChanged(); + } else { + evaluatorReplicasBuilder_.setMessage(builderForValue.build()); + } + + return this; + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public Builder mergeEvaluatorReplicas(flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec value) { + if (evaluatorReplicasBuilder_ == null) { + if (evaluatorReplicas_ != null) { + evaluatorReplicas_ = + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.newBuilder(evaluatorReplicas_).mergeFrom(value).buildPartial(); + } else { + evaluatorReplicas_ = value; + } + onChanged(); + } else { + evaluatorReplicasBuilder_.mergeFrom(value); + } + + return this; + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public Builder clearEvaluatorReplicas() { + if (evaluatorReplicasBuilder_ == null) { + evaluatorReplicas_ = null; + onChanged(); + } else { + evaluatorReplicas_ = null; + evaluatorReplicasBuilder_ = null; + } + + return this; + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder getEvaluatorReplicasBuilder() { + + onChanged(); + return getEvaluatorReplicasFieldBuilder().getBuilder(); + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder getEvaluatorReplicasOrBuilder() { + if (evaluatorReplicasBuilder_ != null) { + return evaluatorReplicasBuilder_.getMessageOrBuilder(); + } else { + return evaluatorReplicas_ == null ? + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.getDefaultInstance() : evaluatorReplicas_; + } + } + /** + *
+       * Evaluator replicas spec
+       * 
+ * + * .flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; + */ + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder> + getEvaluatorReplicasFieldBuilder() { + if (evaluatorReplicasBuilder_ == null) { + evaluatorReplicasBuilder_ = new com.google.protobuf.SingleFieldBuilderV3< + flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpec.Builder, flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplicaSpecOrBuilder>( + getEvaluatorReplicas(), + getParentForChildren(), + isClean()); + evaluatorReplicas_ = null; + } + return evaluatorReplicasBuilder_; + } @java.lang.Override public final Builder setUnknownFields( final com.google.protobuf.UnknownFieldSet unknownFields) { @@ -2553,7 +2807,7 @@ public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplica "\n*flyteidl/plugins/kubeflow/tensorflow.p" + "roto\022\031flyteidl.plugins.kubeflow\032\031flyteid" + "l/core/tasks.proto\032&flyteidl/plugins/kub" + - "eflow/common.proto\"\362\002\n!DistributedTensor" + + "eflow/common.proto\"\323\003\n!DistributedTensor" + "flowTrainingTask\022\\\n\017worker_replicas\030\001 \001(" + "\0132C.flyteidl.plugins.kubeflow.Distribute" + "dTensorflowTrainingReplicaSpec\022X\n\013ps_rep" + @@ -2562,14 +2816,16 @@ public flyteidl.plugins.kubeflow.Tensorflow.DistributedTensorflowTrainingReplica "\022[\n\016chief_replicas\030\003 \001(\0132C.flyteidl.plug" + "ins.kubeflow.DistributedTensorflowTraini" + "ngReplicaSpec\0228\n\nrun_policy\030\004 \001(\0132$.flyt" + - "eidl.plugins.kubeflow.RunPolicy\"\272\001\n(Dist" + - "ributedTensorflowTrainingReplicaSpec\022\020\n\010" + - "replicas\030\001 \001(\005\022\r\n\005image\030\002 \001(\t\022+\n\tresourc" + - "es\030\003 \001(\0132\030.flyteidl.core.Resources\022@\n\016re" + - "start_policy\030\004 \001(\0162(.flyteidl.plugins.ku" + - "beflow.RestartPolicyB?Z=github.com/flyte" + - "org/flyte/flyteidl/gen/pb-go/flyteidl/pl" + - "uginsb\006proto3" + "eidl.plugins.kubeflow.RunPolicy\022_\n\022evalu" + + "ator_replicas\030\005 \001(\0132C.flyteidl.plugins.k" + + "ubeflow.DistributedTensorflowTrainingRep" + + "licaSpec\"\272\001\n(DistributedTensorflowTraini" + + "ngReplicaSpec\022\020\n\010replicas\030\001 \001(\005\022\r\n\005image" + + "\030\002 \001(\t\022+\n\tresources\030\003 \001(\0132\030.flyteidl.cor" + + "e.Resources\022@\n\016restart_policy\030\004 \001(\0162(.fl" + + "yteidl.plugins.kubeflow.RestartPolicyB?Z" + + "=github.com/flyteorg/flyte/flyteidl/gen/" + + "pb-go/flyteidl/pluginsb\006proto3" }; com.google.protobuf.Descriptors.FileDescriptor.InternalDescriptorAssigner assigner = new com.google.protobuf.Descriptors.FileDescriptor. InternalDescriptorAssigner() { @@ -2590,7 +2846,7 @@ public com.google.protobuf.ExtensionRegistry assignDescriptors( internal_static_flyteidl_plugins_kubeflow_DistributedTensorflowTrainingTask_fieldAccessorTable = new com.google.protobuf.GeneratedMessageV3.FieldAccessorTable( internal_static_flyteidl_plugins_kubeflow_DistributedTensorflowTrainingTask_descriptor, - new java.lang.String[] { "WorkerReplicas", "PsReplicas", "ChiefReplicas", "RunPolicy", }); + new java.lang.String[] { "WorkerReplicas", "PsReplicas", "ChiefReplicas", "RunPolicy", "EvaluatorReplicas", }); internal_static_flyteidl_plugins_kubeflow_DistributedTensorflowTrainingReplicaSpec_descriptor = getDescriptor().getMessageTypes().get(1); internal_static_flyteidl_plugins_kubeflow_DistributedTensorflowTrainingReplicaSpec_fieldAccessorTable = new diff --git a/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.py b/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.py index 15f0d965581..f0c086f9e7d 100644 --- a/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.py +++ b/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.py @@ -15,7 +15,7 @@ from flyteidl.plugins.kubeflow import common_pb2 as flyteidl_dot_plugins_dot_kubeflow_dot_common__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n*flyteidl/plugins/kubeflow/tensorflow.proto\x12\x19\x66lyteidl.plugins.kubeflow\x1a\x19\x66lyteidl/core/tasks.proto\x1a&flyteidl/plugins/kubeflow/common.proto\"\xa8\x03\n!DistributedTensorflowTrainingTask\x12l\n\x0fworker_replicas\x18\x01 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\x0eworkerReplicas\x12\x64\n\x0bps_replicas\x18\x02 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\npsReplicas\x12j\n\x0e\x63hief_replicas\x18\x03 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\rchiefReplicas\x12\x43\n\nrun_policy\x18\x04 \x01(\x0b\x32$.flyteidl.plugins.kubeflow.RunPolicyR\trunPolicy\"\xe5\x01\n(DistributedTensorflowTrainingReplicaSpec\x12\x1a\n\x08replicas\x18\x01 \x01(\x05R\x08replicas\x12\x14\n\x05image\x18\x02 \x01(\tR\x05image\x12\x36\n\tresources\x18\x03 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresources\x12O\n\x0erestart_policy\x18\x04 \x01(\x0e\x32(.flyteidl.plugins.kubeflow.RestartPolicyR\rrestartPolicyB\xf5\x01\n\x1d\x63om.flyteidl.plugins.kubeflowB\x0fTensorflowProtoP\x01Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\xa2\x02\x03\x46PK\xaa\x02\x19\x46lyteidl.Plugins.Kubeflow\xca\x02\x19\x46lyteidl\\Plugins\\Kubeflow\xe2\x02%Flyteidl\\Plugins\\Kubeflow\\GPBMetadata\xea\x02\x1b\x46lyteidl::Plugins::Kubeflowb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n*flyteidl/plugins/kubeflow/tensorflow.proto\x12\x19\x66lyteidl.plugins.kubeflow\x1a\x19\x66lyteidl/core/tasks.proto\x1a&flyteidl/plugins/kubeflow/common.proto\"\x9c\x04\n!DistributedTensorflowTrainingTask\x12l\n\x0fworker_replicas\x18\x01 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\x0eworkerReplicas\x12\x64\n\x0bps_replicas\x18\x02 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\npsReplicas\x12j\n\x0e\x63hief_replicas\x18\x03 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\rchiefReplicas\x12\x43\n\nrun_policy\x18\x04 \x01(\x0b\x32$.flyteidl.plugins.kubeflow.RunPolicyR\trunPolicy\x12r\n\x12\x65valuator_replicas\x18\x05 \x01(\x0b\x32\x43.flyteidl.plugins.kubeflow.DistributedTensorflowTrainingReplicaSpecR\x11\x65valuatorReplicas\"\xe5\x01\n(DistributedTensorflowTrainingReplicaSpec\x12\x1a\n\x08replicas\x18\x01 \x01(\x05R\x08replicas\x12\x14\n\x05image\x18\x02 \x01(\tR\x05image\x12\x36\n\tresources\x18\x03 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresources\x12O\n\x0erestart_policy\x18\x04 \x01(\x0e\x32(.flyteidl.plugins.kubeflow.RestartPolicyR\rrestartPolicyB\xf5\x01\n\x1d\x63om.flyteidl.plugins.kubeflowB\x0fTensorflowProtoP\x01Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\xa2\x02\x03\x46PK\xaa\x02\x19\x46lyteidl.Plugins.Kubeflow\xca\x02\x19\x46lyteidl\\Plugins\\Kubeflow\xe2\x02%Flyteidl\\Plugins\\Kubeflow\\GPBMetadata\xea\x02\x1b\x46lyteidl::Plugins::Kubeflowb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -25,7 +25,7 @@ DESCRIPTOR._options = None DESCRIPTOR._serialized_options = b'\n\035com.flyteidl.plugins.kubeflowB\017TensorflowProtoP\001Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\242\002\003FPK\252\002\031Flyteidl.Plugins.Kubeflow\312\002\031Flyteidl\\Plugins\\Kubeflow\342\002%Flyteidl\\Plugins\\Kubeflow\\GPBMetadata\352\002\033Flyteidl::Plugins::Kubeflow' _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_start=141 - _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_end=565 - _globals['_DISTRIBUTEDTENSORFLOWTRAININGREPLICASPEC']._serialized_start=568 - _globals['_DISTRIBUTEDTENSORFLOWTRAININGREPLICASPEC']._serialized_end=797 + _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_end=681 + _globals['_DISTRIBUTEDTENSORFLOWTRAININGREPLICASPEC']._serialized_start=684 + _globals['_DISTRIBUTEDTENSORFLOWTRAININGREPLICASPEC']._serialized_end=913 # @@protoc_insertion_point(module_scope) diff --git a/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.pyi b/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.pyi index e08a1ff9837..4a999f70e8b 100644 --- a/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.pyi +++ b/flyteidl/gen/pb_python/flyteidl/plugins/kubeflow/tensorflow_pb2.pyi @@ -7,16 +7,18 @@ from typing import ClassVar as _ClassVar, Mapping as _Mapping, Optional as _Opti DESCRIPTOR: _descriptor.FileDescriptor class DistributedTensorflowTrainingTask(_message.Message): - __slots__ = ["worker_replicas", "ps_replicas", "chief_replicas", "run_policy"] + __slots__ = ["worker_replicas", "ps_replicas", "chief_replicas", "run_policy", "evaluator_replicas"] WORKER_REPLICAS_FIELD_NUMBER: _ClassVar[int] PS_REPLICAS_FIELD_NUMBER: _ClassVar[int] CHIEF_REPLICAS_FIELD_NUMBER: _ClassVar[int] RUN_POLICY_FIELD_NUMBER: _ClassVar[int] + EVALUATOR_REPLICAS_FIELD_NUMBER: _ClassVar[int] worker_replicas: DistributedTensorflowTrainingReplicaSpec ps_replicas: DistributedTensorflowTrainingReplicaSpec chief_replicas: DistributedTensorflowTrainingReplicaSpec run_policy: _common_pb2.RunPolicy - def __init__(self, worker_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., ps_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., chief_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., run_policy: _Optional[_Union[_common_pb2.RunPolicy, _Mapping]] = ...) -> None: ... + evaluator_replicas: DistributedTensorflowTrainingReplicaSpec + def __init__(self, worker_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., ps_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., chief_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ..., run_policy: _Optional[_Union[_common_pb2.RunPolicy, _Mapping]] = ..., evaluator_replicas: _Optional[_Union[DistributedTensorflowTrainingReplicaSpec, _Mapping]] = ...) -> None: ... class DistributedTensorflowTrainingReplicaSpec(_message.Message): __slots__ = ["replicas", "image", "resources", "restart_policy"] diff --git a/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.py b/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.py index 187526d6d5b..ceed4231bbd 100644 --- a/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.py +++ b/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.py @@ -13,7 +13,7 @@ -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n!flyteidl/plugins/tensorflow.proto\x12\x10\x66lyteidl.plugins\"\x85\x01\n!DistributedTensorflowTrainingTask\x12\x18\n\x07workers\x18\x01 \x01(\x05R\x07workers\x12\x1f\n\x0bps_replicas\x18\x02 \x01(\x05R\npsReplicas\x12%\n\x0e\x63hief_replicas\x18\x03 \x01(\x05R\rchiefReplicasB\xc7\x01\n\x14\x63om.flyteidl.pluginsB\x0fTensorflowProtoP\x01Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\xa2\x02\x03\x46PX\xaa\x02\x10\x46lyteidl.Plugins\xca\x02\x10\x46lyteidl\\Plugins\xe2\x02\x1c\x46lyteidl\\Plugins\\GPBMetadata\xea\x02\x11\x46lyteidl::Pluginsb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n!flyteidl/plugins/tensorflow.proto\x12\x10\x66lyteidl.plugins\"\xb4\x01\n!DistributedTensorflowTrainingTask\x12\x18\n\x07workers\x18\x01 \x01(\x05R\x07workers\x12\x1f\n\x0bps_replicas\x18\x02 \x01(\x05R\npsReplicas\x12%\n\x0e\x63hief_replicas\x18\x03 \x01(\x05R\rchiefReplicas\x12-\n\x12\x65valuator_replicas\x18\x04 \x01(\x05R\x11\x65valuatorReplicasB\xc7\x01\n\x14\x63om.flyteidl.pluginsB\x0fTensorflowProtoP\x01Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\xa2\x02\x03\x46PX\xaa\x02\x10\x46lyteidl.Plugins\xca\x02\x10\x46lyteidl\\Plugins\xe2\x02\x1c\x46lyteidl\\Plugins\\GPBMetadata\xea\x02\x11\x46lyteidl::Pluginsb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -23,5 +23,5 @@ DESCRIPTOR._options = None DESCRIPTOR._serialized_options = b'\n\024com.flyteidl.pluginsB\017TensorflowProtoP\001Z=github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins\242\002\003FPX\252\002\020Flyteidl.Plugins\312\002\020Flyteidl\\Plugins\342\002\034Flyteidl\\Plugins\\GPBMetadata\352\002\021Flyteidl::Plugins' _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_start=56 - _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_end=189 + _globals['_DISTRIBUTEDTENSORFLOWTRAININGTASK']._serialized_end=236 # @@protoc_insertion_point(module_scope) diff --git a/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.pyi b/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.pyi index d3dc028af35..81e2bc30b90 100644 --- a/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.pyi +++ b/flyteidl/gen/pb_python/flyteidl/plugins/tensorflow_pb2.pyi @@ -5,11 +5,13 @@ from typing import ClassVar as _ClassVar, Optional as _Optional DESCRIPTOR: _descriptor.FileDescriptor class DistributedTensorflowTrainingTask(_message.Message): - __slots__ = ["workers", "ps_replicas", "chief_replicas"] + __slots__ = ["workers", "ps_replicas", "chief_replicas", "evaluator_replicas"] WORKERS_FIELD_NUMBER: _ClassVar[int] PS_REPLICAS_FIELD_NUMBER: _ClassVar[int] CHIEF_REPLICAS_FIELD_NUMBER: _ClassVar[int] + EVALUATOR_REPLICAS_FIELD_NUMBER: _ClassVar[int] workers: int ps_replicas: int chief_replicas: int - def __init__(self, workers: _Optional[int] = ..., ps_replicas: _Optional[int] = ..., chief_replicas: _Optional[int] = ...) -> None: ... + evaluator_replicas: int + def __init__(self, workers: _Optional[int] = ..., ps_replicas: _Optional[int] = ..., chief_replicas: _Optional[int] = ..., evaluator_replicas: _Optional[int] = ...) -> None: ... diff --git a/flyteidl/gen/pb_rust/flyteidl.plugins.kubeflow.rs b/flyteidl/gen/pb_rust/flyteidl.plugins.kubeflow.rs index 59c1f681a0a..96d46653daf 100644 --- a/flyteidl/gen/pb_rust/flyteidl.plugins.kubeflow.rs +++ b/flyteidl/gen/pb_rust/flyteidl.plugins.kubeflow.rs @@ -182,6 +182,9 @@ pub struct DistributedTensorflowTrainingTask { /// active. #[prost(message, optional, tag="4")] pub run_policy: ::core::option::Option, + /// Evaluator replicas spec + #[prost(message, optional, tag="5")] + pub evaluator_replicas: ::core::option::Option, } #[allow(clippy::derive_partial_eq_without_eq)] #[derive(Clone, PartialEq, ::prost::Message)] diff --git a/flyteidl/gen/pb_rust/flyteidl.plugins.rs b/flyteidl/gen/pb_rust/flyteidl.plugins.rs index 5c7873b5d2f..11e4ad05af2 100644 --- a/flyteidl/gen/pb_rust/flyteidl.plugins.rs +++ b/flyteidl/gen/pb_rust/flyteidl.plugins.rs @@ -290,14 +290,19 @@ pub struct SparkJob { #[allow(clippy::derive_partial_eq_without_eq)] #[derive(Clone, PartialEq, ::prost::Message)] pub struct DistributedTensorflowTrainingTask { - /// number of worker, ps, chief replicas spawned in the cluster for this job + /// number of worker replicas spawned in the cluster for this job #[prost(int32, tag="1")] pub workers: i32, /// PS -> Parameter server + /// number of ps replicas spawned in the cluster for this job #[prost(int32, tag="2")] pub ps_replicas: i32, + /// number of chief replicas spawned in the cluster for this job #[prost(int32, tag="3")] pub chief_replicas: i32, + /// number of evaluator replicas spawned in the cluster for this job + #[prost(int32, tag="4")] + pub evaluator_replicas: i32, } /// Represents an Execution that was launched and could be waited on. #[allow(clippy::derive_partial_eq_without_eq)] diff --git a/flyteidl/protos/flyteidl/plugins/kubeflow/tensorflow.proto b/flyteidl/protos/flyteidl/plugins/kubeflow/tensorflow.proto index 4cf31535484..789666b989e 100644 --- a/flyteidl/protos/flyteidl/plugins/kubeflow/tensorflow.proto +++ b/flyteidl/protos/flyteidl/plugins/kubeflow/tensorflow.proto @@ -22,6 +22,9 @@ message DistributedTensorflowTrainingTask { // job, for example how to clean up resources and how long the job can stay // active. RunPolicy run_policy = 4; + + // Evaluator replicas spec + DistributedTensorflowTrainingReplicaSpec evaluator_replicas = 5; } message DistributedTensorflowTrainingReplicaSpec { diff --git a/flyteidl/protos/flyteidl/plugins/tensorflow.proto b/flyteidl/protos/flyteidl/plugins/tensorflow.proto index e768ad65adf..e494a6cc324 100644 --- a/flyteidl/protos/flyteidl/plugins/tensorflow.proto +++ b/flyteidl/protos/flyteidl/plugins/tensorflow.proto @@ -6,9 +6,13 @@ option go_package = "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugi // Custom proto for plugin that enables distributed training using https://github.com/kubeflow/tf-operator message DistributedTensorflowTrainingTask { - // number of worker, ps, chief replicas spawned in the cluster for this job + // number of worker replicas spawned in the cluster for this job int32 workers = 1; // PS -> Parameter server + // number of ps replicas spawned in the cluster for this job int32 ps_replicas = 2; + // number of chief replicas spawned in the cluster for this job int32 chief_replicas = 3; + // number of evaluator replicas spawned in the cluster for this job + int32 evaluator_replicas = 4; } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator.go index 4e38d791c49..f6a9787dbbf 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator.go @@ -93,7 +93,7 @@ func GetMPIPhaseInfo(currentCondition commonOp.JobCondition, occurredAt time.Tim // GetLogs will return the logs for kubeflow job func GetLogs(pluginContext k8s.PluginContext, taskType string, objectMeta meta_v1.ObjectMeta, hasMaster bool, - workersCount int32, psReplicasCount int32, chiefReplicasCount int32) ([]*core.TaskLog, error) { + workersCount int32, psReplicasCount int32, chiefReplicasCount int32, evaluatorReplicasCount int32) ([]*core.TaskLog, error) { name := objectMeta.Name namespace := objectMeta.Namespace @@ -181,6 +181,18 @@ func GetLogs(pluginContext k8s.PluginContext, taskType string, objectMeta meta_v } taskLogs = append(taskLogs, chiefReplicaLog.TaskLogs...) } + // get evaluator log, and the max number of evaluator is 1 + if evaluatorReplicasCount != 0 { + evaluatorReplicasCount, err := logPlugin.GetTaskLogs(tasklog.Input{ + PodName: name + fmt.Sprintf("-evaluatorReplica-%d", 0), + Namespace: namespace, + TaskExecutionIdentifier: &taskExecID, + }) + if err != nil { + return nil, err + } + taskLogs = append(taskLogs, evaluatorReplicasCount.TaskLogs...) + } return taskLogs, nil } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator_test.go index 4f5d70dc5c3..6c083b68988 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/common/common_operator_test.go @@ -167,7 +167,7 @@ func TestGetLogs(t *testing.T) { Name: "test", Namespace: "mpi-namespace", } - jobLogs, err := GetLogs(taskCtx, MPITaskType, mpiJobObjectMeta, false, workers, launcher, 0) + jobLogs, err := GetLogs(taskCtx, MPITaskType, mpiJobObjectMeta, false, workers, launcher, 0, 0) assert.NoError(t, err) assert.Equal(t, 1, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-0/pod?namespace=mpi-namespace", "mpi-namespace", "test"), jobLogs[0].Uri) @@ -176,7 +176,7 @@ func TestGetLogs(t *testing.T) { Name: "test", Namespace: "pytorch-namespace", } - jobLogs, err = GetLogs(taskCtx, PytorchTaskType, pytorchJobObjectMeta, true, workers, launcher, 0) + jobLogs, err = GetLogs(taskCtx, PytorchTaskType, pytorchJobObjectMeta, true, workers, launcher, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-master-0/pod?namespace=pytorch-namespace", "pytorch-namespace", "test"), jobLogs[0].Uri) @@ -186,7 +186,7 @@ func TestGetLogs(t *testing.T) { Name: "test", Namespace: "tensorflow-namespace", } - jobLogs, err = GetLogs(taskCtx, TensorflowTaskType, tensorflowJobObjectMeta, false, workers, launcher, 1) + jobLogs, err = GetLogs(taskCtx, TensorflowTaskType, tensorflowJobObjectMeta, false, workers, launcher, 1, 0) assert.NoError(t, err) assert.Equal(t, 3, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-0/pod?namespace=tensorflow-namespace", "tensorflow-namespace", "test"), jobLogs[0].Uri) @@ -209,7 +209,7 @@ func TestGetLogsTemplateUri(t *testing.T) { Time: time.Date(2022, time.January, 1, 12, 0, 0, 0, time.UTC), }, } - jobLogs, err := GetLogs(taskCtx, PytorchTaskType, pytorchJobObjectMeta, true, 1, 0, 0) + jobLogs, err := GetLogs(taskCtx, PytorchTaskType, pytorchJobObjectMeta, true, 1, 0, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) assert.Equal(t, fmt.Sprintf("https://console.cloud.google.com/logs/query;query=resource.labels.pod_name=%s-master-0×tamp>%s", "test", "2022-01-01T12:00:00Z"), jobLogs[0].Uri) diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi.go index 492dd322353..25e45ad7274 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi.go @@ -210,7 +210,7 @@ func (mpiOperatorResourceHandler) GetTaskPhase(_ context.Context, pluginContext numLauncherReplicas = app.Spec.MPIReplicaSpecs[kubeflowv1.MPIJobReplicaTypeLauncher].Replicas taskLogs, err := common.GetLogs(pluginContext, common.MPITaskType, app.ObjectMeta, false, - *numWorkers, *numLauncherReplicas, 0) + *numWorkers, *numLauncherReplicas, 0, 0) if err != nil { return pluginsCore.PhaseInfoUndefined, err } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go index 96d8efdcb89..b7c66b08f60 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go @@ -410,7 +410,7 @@ func TestGetLogs(t *testing.T) { mpiResourceHandler := mpiOperatorResourceHandler{} mpiJob := dummyMPIJobResource(mpiResourceHandler, workers, launcher, slots, mpiOp.JobRunning) taskCtx := dummyMPITaskContext(dummyMPITaskTemplate("", dummyMPICustomObj(workers, launcher, slots))) - jobLogs, err := common.GetLogs(taskCtx, common.MPITaskType, mpiJob.ObjectMeta, false, workers, launcher, 0) + jobLogs, err := common.GetLogs(taskCtx, common.MPITaskType, mpiJob.ObjectMeta, false, workers, launcher, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-0/pod?namespace=mpi-namespace", jobNamespace, jobName), jobLogs[0].Uri) diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch.go index 07f3df0ef5d..2461c7bc187 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch.go @@ -231,7 +231,7 @@ func (pytorchOperatorResourceHandler) GetTaskPhase(_ context.Context, pluginCont workersCount := app.Spec.PyTorchReplicaSpecs[kubeflowv1.PyTorchJobReplicaTypeWorker].Replicas - taskLogs, err := common.GetLogs(pluginContext, common.PytorchTaskType, app.ObjectMeta, hasMaster, *workersCount, 0, 0) + taskLogs, err := common.GetLogs(pluginContext, common.PytorchTaskType, app.ObjectMeta, hasMaster, *workersCount, 0, 0, 0) if err != nil { return pluginsCore.PhaseInfoUndefined, err } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go index a1e37f51249..9407294971d 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go @@ -455,7 +455,7 @@ func TestGetLogs(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} pytorchJob := dummyPytorchJobResource(pytorchResourceHandler, workers, commonOp.JobRunning) taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers))) - jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0) + jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0, 0) assert.NoError(t, err) assert.Equal(t, 3, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-master-0/pod?namespace=pytorch-namespace", jobNamespace, jobName), jobLogs[0].Uri) @@ -475,7 +475,7 @@ func TestGetLogsElastic(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} pytorchJob := dummyPytorchJobResource(pytorchResourceHandler, workers, commonOp.JobRunning) taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers))) - jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0) + jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-0/pod?namespace=pytorch-namespace", jobNamespace, jobName), jobLogs[0].Uri) diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow.go index 578bd2a0d6c..1c4c9658198 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow.go @@ -80,6 +80,11 @@ func (tensorflowOperatorResourceHandler) BuildResource(ctx context.Context, task PodSpec: podSpec.DeepCopy(), RestartPolicy: commonOp.RestartPolicyNever, }, + kubeflowv1.TFJobReplicaTypeEval: { + ReplicaNum: int32(0), + PodSpec: podSpec.DeepCopy(), + RestartPolicy: commonOp.RestartPolicyNever, + }, } runPolicy := commonOp.RunPolicy{} @@ -94,6 +99,7 @@ func (tensorflowOperatorResourceHandler) BuildResource(ctx context.Context, task replicaSpecMap[kubeflowv1.TFJobReplicaTypeChief].ReplicaNum = tensorflowTaskExtraArgs.GetChiefReplicas() replicaSpecMap[kubeflowv1.TFJobReplicaTypeWorker].ReplicaNum = tensorflowTaskExtraArgs.GetWorkers() replicaSpecMap[kubeflowv1.TFJobReplicaTypePS].ReplicaNum = tensorflowTaskExtraArgs.GetPsReplicas() + replicaSpecMap[kubeflowv1.TFJobReplicaTypeEval].ReplicaNum = tensorflowTaskExtraArgs.GetEvaluatorReplicas() } else if taskTemplate.TaskTypeVersion == 1 { kfTensorflowTaskExtraArgs := kfplugins.DistributedTensorflowTrainingTask{} @@ -151,6 +157,22 @@ func (tensorflowOperatorResourceHandler) BuildResource(ctx context.Context, task replicaSpecMap[kubeflowv1.TFJobReplicaTypePS].ReplicaNum = psReplicaSpec.GetReplicas() } + evaluatorReplicaSpec := kfTensorflowTaskExtraArgs.GetEvaluatorReplicas() + if evaluatorReplicaSpec != nil { + err := common.OverrideContainerSpec( + replicaSpecMap[kubeflowv1.TFJobReplicaTypeEval].PodSpec, + kubeflowv1.TFJobDefaultContainerName, + evaluatorReplicaSpec.GetImage(), + evaluatorReplicaSpec.GetResources(), + nil, + ) + if err != nil { + return nil, err + } + replicaSpecMap[kubeflowv1.TFJobReplicaTypeEval].RestartPolicy = common.ParseRestartPolicy(evaluatorReplicaSpec.GetRestartPolicy()) + replicaSpecMap[kubeflowv1.TFJobReplicaTypeEval].ReplicaNum = evaluatorReplicaSpec.GetReplicas() + } + if kfTensorflowTaskExtraArgs.GetRunPolicy() != nil { runPolicy = common.ParseRunPolicy(*kfTensorflowTaskExtraArgs.GetRunPolicy()) } @@ -207,9 +229,10 @@ func (tensorflowOperatorResourceHandler) GetTaskPhase(_ context.Context, pluginC workersCount := app.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeWorker].Replicas psReplicasCount := app.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypePS].Replicas chiefCount := app.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeChief].Replicas + evaluatorReplicasCount := app.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeEval].Replicas taskLogs, err := common.GetLogs(pluginContext, common.TensorflowTaskType, app.ObjectMeta, false, - *workersCount, *psReplicasCount, *chiefCount) + *workersCount, *psReplicasCount, *chiefCount, *evaluatorReplicasCount) if err != nil { return pluginsCore.PhaseInfoUndefined, err } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go index 7ec1983bea1..764be381811 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go @@ -70,11 +70,12 @@ var ( jobNamespace = "tensorflow-namespace" ) -func dummyTensorFlowCustomObj(workers int32, psReplicas int32, chiefReplicas int32) *plugins.DistributedTensorflowTrainingTask { +func dummyTensorFlowCustomObj(workers int32, psReplicas int32, chiefReplicas int32, evaluatorReplicas int32) *plugins.DistributedTensorflowTrainingTask { return &plugins.DistributedTensorflowTrainingTask{ - Workers: workers, - PsReplicas: psReplicas, - ChiefReplicas: chiefReplicas, + Workers: workers, + PsReplicas: psReplicas, + ChiefReplicas: chiefReplicas, + EvaluatorReplicas: evaluatorReplicas, } } @@ -175,7 +176,7 @@ func dummyTensorFlowTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.Tas } func dummyTensorFlowJobResource(tensorflowResourceHandler tensorflowOperatorResourceHandler, - workers int32, psReplicas int32, chiefReplicas int32, conditionType commonOp.JobConditionType) *kubeflowv1.TFJob { + workers int32, psReplicas int32, chiefReplicas int32, evaluatorReplicas int32, conditionType commonOp.JobConditionType) *kubeflowv1.TFJob { var jobConditions []commonOp.JobCondition now := time.Now() @@ -274,7 +275,7 @@ func dummyTensorFlowJobResource(tensorflowResourceHandler tensorflowOperatorReso } } - tfObj := dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas) + tfObj := dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas, evaluatorReplicas) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) if err != nil { @@ -300,7 +301,7 @@ func dummyTensorFlowJobResource(tensorflowResourceHandler tensorflowOperatorReso func TestBuildResourceTensorFlow(t *testing.T) { tensorflowResourceHandler := tensorflowOperatorResourceHandler{} - tfObj := dummyTensorFlowCustomObj(100, 50, 1) + tfObj := dummyTensorFlowCustomObj(100, 50, 1, 1) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) @@ -312,6 +313,7 @@ func TestBuildResourceTensorFlow(t *testing.T) { assert.Equal(t, int32(100), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeWorker].Replicas) assert.Equal(t, int32(50), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypePS].Replicas) assert.Equal(t, int32(1), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeChief].Replicas) + assert.Equal(t, int32(1), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeEval].Replicas) // verify TaskExecutionMetadata labels and annotations are copied to the TensorFlowJob for k, v := range dummyAnnotations { @@ -346,10 +348,10 @@ func TestGetTaskPhase(t *testing.T) { ctx := context.TODO() dummyTensorFlowJobResourceCreator := func(conditionType commonOp.JobConditionType) *kubeflowv1.TFJob { - return dummyTensorFlowJobResource(tensorflowResourceHandler, 2, 1, 1, conditionType) + return dummyTensorFlowJobResource(tensorflowResourceHandler, 2, 1, 1, 1, conditionType) } - taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(2, 1, 1))) + taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(2, 1, 1, 1))) taskPhase, err := tensorflowResourceHandler.GetTaskPhase(ctx, taskCtx, dummyTensorFlowJobResourceCreator(commonOp.JobCreated)) assert.NoError(t, err) assert.Equal(t, pluginsCore.PhaseQueued, taskPhase.Phase()) @@ -390,18 +392,20 @@ func TestGetLogs(t *testing.T) { workers := int32(2) psReplicas := int32(1) chiefReplicas := int32(1) + evaluatorReplicas := int32(1) tensorflowResourceHandler := tensorflowOperatorResourceHandler{} - tensorFlowJob := dummyTensorFlowJobResource(tensorflowResourceHandler, workers, psReplicas, chiefReplicas, commonOp.JobRunning) - taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas))) + tensorFlowJob := dummyTensorFlowJobResource(tensorflowResourceHandler, workers, psReplicas, chiefReplicas, evaluatorReplicas, commonOp.JobRunning) + taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas, evaluatorReplicas))) jobLogs, err := common.GetLogs(taskCtx, common.TensorflowTaskType, tensorFlowJob.ObjectMeta, false, - workers, psReplicas, chiefReplicas) + workers, psReplicas, chiefReplicas, evaluatorReplicas) assert.NoError(t, err) - assert.Equal(t, 4, len(jobLogs)) + assert.Equal(t, 5, len(jobLogs)) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-0/pod?namespace=tensorflow-namespace", jobNamespace, jobName), jobLogs[0].Uri) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-worker-1/pod?namespace=tensorflow-namespace", jobNamespace, jobName), jobLogs[1].Uri) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-psReplica-0/pod?namespace=tensorflow-namespace", jobNamespace, jobName), jobLogs[2].Uri) assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-chiefReplica-0/pod?namespace=tensorflow-namespace", jobNamespace, jobName), jobLogs[3].Uri) + assert.Equal(t, fmt.Sprintf("k8s.com/#!/log/%s/%s-evaluatorReplica-0/pod?namespace=tensorflow-namespace", jobNamespace, jobName), jobLogs[4].Uri) } func TestGetProperties(t *testing.T) { @@ -412,26 +416,31 @@ func TestGetProperties(t *testing.T) { func TestReplicaCounts(t *testing.T) { for _, test := range []struct { - name string - chiefReplicaCount int32 - psReplicaCount int32 - workerReplicaCount int32 - expectError bool - contains []commonOp.ReplicaType - notContains []commonOp.ReplicaType + name string + chiefReplicaCount int32 + psReplicaCount int32 + workerReplicaCount int32 + evaluatorReplicaCount int32 + expectError bool + contains []commonOp.ReplicaType + notContains []commonOp.ReplicaType }{ - {"NoWorkers", 1, 1, 0, true, nil, nil}, - {"SingleChief", 1, 0, 1, false, + {"NoWorkers", 1, 1, 0, 1, true, nil, nil}, + {"SingleChief", 1, 0, 1, 0, false, []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypeChief, kubeflowv1.TFJobReplicaTypeWorker}, - []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypePS}}, - {"SinglePS", 0, 1, 1, false, + []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypePS, kubeflowv1.TFJobReplicaTypeEval}}, + {"SinglePS", 0, 1, 1, 0, false, []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypePS, kubeflowv1.TFJobReplicaTypeWorker}, - []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypeChief}}, + []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypeChief, kubeflowv1.TFJobReplicaTypeEval}}, + {"AllContains", 1, 1, 1, 1, false, + []commonOp.ReplicaType{kubeflowv1.TFJobReplicaTypePS, kubeflowv1.TFJobReplicaTypeWorker, kubeflowv1.TFJobReplicaTypeChief, kubeflowv1.TFJobReplicaTypeEval}, + nil, + }, } { t.Run(test.name, func(t *testing.T) { tensorflowResourceHandler := tensorflowOperatorResourceHandler{} - tfObj := dummyTensorFlowCustomObj(test.workerReplicaCount, test.psReplicaCount, test.chiefReplicaCount) + tfObj := dummyTensorFlowCustomObj(test.workerReplicaCount, test.psReplicaCount, test.chiefReplicaCount, test.evaluatorReplicaCount) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) @@ -499,6 +508,21 @@ func TestBuildResourceTensorFlowV1(t *testing.T) { }, }, }, + EvaluatorReplicas: &kfplugins.DistributedTensorflowTrainingReplicaSpec{ + Replicas: 1, + Image: testImage, + Resources: &core.Resources{ + Requests: []*core.Resources_ResourceEntry{ + {Name: core.Resources_CPU, Value: "250m"}, + {Name: core.Resources_MEMORY, Value: "1Gi"}, + }, + Limits: []*core.Resources_ResourceEntry{ + {Name: core.Resources_CPU, Value: "500m"}, + {Name: core.Resources_MEMORY, Value: "2Gi"}, + }, + }, + RestartPolicy: kfplugins.RestartPolicy_RESTART_POLICY_ALWAYS, + }, RunPolicy: &kfplugins.RunPolicy{ CleanPodPolicy: kfplugins.CleanPodPolicy_CLEANPOD_POLICY_ALL, ActiveDeadlineSeconds: int32(100), @@ -534,6 +558,16 @@ func TestBuildResourceTensorFlowV1(t *testing.T) { corev1.ResourceCPU: resource.MustParse("500m"), }, }, + kubeflowv1.TFJobReplicaTypeEval: { + Requests: corev1.ResourceList{ + corev1.ResourceCPU: resource.MustParse("250m"), + corev1.ResourceMemory: resource.MustParse("1Gi"), + }, + Limits: corev1.ResourceList{ + corev1.ResourceCPU: resource.MustParse("500m"), + corev1.ResourceMemory: resource.MustParse("2Gi"), + }, + }, } tensorflowResourceHandler := tensorflowOperatorResourceHandler{} @@ -550,6 +584,7 @@ func TestBuildResourceTensorFlowV1(t *testing.T) { assert.Equal(t, int32(100), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeWorker].Replicas) assert.Equal(t, int32(50), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypePS].Replicas) assert.Equal(t, int32(1), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeChief].Replicas) + assert.Equal(t, int32(1), *tensorflowJob.Spec.TFReplicaSpecs[kubeflowv1.TFJobReplicaTypeEval].Replicas) for replicaType, replicaSpec := range tensorflowJob.Spec.TFReplicaSpecs { var hasContainerWithDefaultTensorFlowName = false From b9f6e8c87eb9808191117c0b5baef8964b81d187 Mon Sep 17 00:00:00 2001 From: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> Date: Wed, 11 Oct 2023 15:20:02 -0700 Subject: [PATCH 06/18] [Monorepo] Enable proxy-authorization in admin client (#4189) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Add proxyCommand to client config Signed-off-by: Fabio Grätz * Add proxy auth unary interceptor Signed-off-by: Fabio Grätz * Use proxy auth in http client for oauth Signed-off-by: Fabio Grätz * Cache tokens obtained from external commands Signed-off-by: Fabio Grätz * Make tests pass Signed-off-by: Fabio Grätz * Add tests for proxy auth interceptor Signed-off-by: Fabio Grätz * Make work without 2nd token cache but instead with 2nd credentials future Signed-off-by: Fabio Grätz * Adapt existing tests to not using a 2nd token cache but a 2nd credentials future Signed-off-by: Fabio Grätz * Adapt new tests to not using a 2nd token cache but a 2nd credentials future Signed-off-by: Fabio Grätz * Fix number of opts in NewAdminConnection Signed-off-by: Fabio Grätz * Don't overwrite original error in NewProxyAuthInterceptor Signed-off-by: Fabio Grätz * Improve error message when failing to create http client for oauth Signed-off-by: Fabio Grätz * Actually don't return any error from setHTTPClientContext at all as before Signed-off-by: Fabio Grätz * Don't require github.com/golang-jwt/jwt anymore Signed-off-by: Fabio Grätz * Make tests pass again Signed-off-by: Fabio Grätz * Lint Signed-off-by: Fabio Grätz * make -C flyteidl generate Signed-off-by: Eduardo Apolinario * Fix flytepropeller's getAdminClient Signed-off-by: Fabio Grätz --------- Signed-off-by: Fabio Grätz Signed-off-by: Eduardo Apolinario Co-authored-by: Fabio Grätz Co-authored-by: Eduardo Apolinario --- flyteidl/clients/go/admin/auth_interceptor.go | 85 ++++++++++-- .../clients/go/admin/auth_interceptor_test.go | 127 +++++++++++++++++- flyteidl/clients/go/admin/client.go | 19 ++- flyteidl/clients/go/admin/config.go | 2 + flyteidl/clients/go/admin/config_flags.go | 1 + .../clients/go/admin/config_flags_test.go | 14 ++ .../go/admin/pkce/auth_flow_orchestrator.go | 10 +- .../go/admin/pkce/handle_app_call_back.go | 5 +- .../admin/pkce/handle_app_call_back_test.go | 2 +- flytepropeller/pkg/controller/controller.go | 2 +- 10 files changed, 242 insertions(+), 25 deletions(-) diff --git a/flyteidl/clients/go/admin/auth_interceptor.go b/flyteidl/clients/go/admin/auth_interceptor.go index daa91968bc1..ef94d857567 100644 --- a/flyteidl/clients/go/admin/auth_interceptor.go +++ b/flyteidl/clients/go/admin/auth_interceptor.go @@ -2,6 +2,7 @@ package admin import ( "context" + "errors" "fmt" "net/http" @@ -16,10 +17,12 @@ import ( "google.golang.org/grpc" ) +const ProxyAuthorizationHeader = "proxy-authorization" + // MaterializeCredentials will attempt to build a TokenSource given the anonymously available information exposed by the server. // Once established, it'll invoke PerRPCCredentialsFuture.Store() on perRPCCredentials to populate it with the appropriate values. -func MaterializeCredentials(ctx context.Context, cfg *Config, tokenCache cache.TokenCache, perRPCCredentials *PerRPCCredentialsFuture) error { - authMetadataClient, err := InitializeAuthMetadataClient(ctx, cfg) +func MaterializeCredentials(ctx context.Context, cfg *Config, tokenCache cache.TokenCache, perRPCCredentials *PerRPCCredentialsFuture, proxyCredentialsFuture *PerRPCCredentialsFuture) error { + authMetadataClient, err := InitializeAuthMetadataClient(ctx, cfg, proxyCredentialsFuture) if err != nil { return fmt.Errorf("failed to initialized Auth Metadata Client. Error: %w", err) } @@ -48,19 +51,70 @@ func MaterializeCredentials(ctx context.Context, cfg *Config, tokenCache cache.T return nil } +func GetProxyTokenSource(ctx context.Context, cfg *Config) (oauth2.TokenSource, error) { + tokenSourceProvider, err := NewExternalTokenSourceProvider(cfg.ProxyCommand) + if err != nil { + return nil, fmt.Errorf("failed to initialized proxy authorization token source provider. Err: %w", err) + } + proxyTokenSource, err := tokenSourceProvider.GetTokenSource(ctx) + if err != nil { + return nil, err + } + return proxyTokenSource, nil +} + +func MaterializeProxyAuthCredentials(ctx context.Context, cfg *Config, proxyCredentialsFuture *PerRPCCredentialsFuture) error { + proxyTokenSource, err := GetProxyTokenSource(ctx, cfg) + if err != nil { + return err + } + + wrappedTokenSource := NewCustomHeaderTokenSource(proxyTokenSource, cfg.UseInsecureConnection, ProxyAuthorizationHeader) + proxyCredentialsFuture.Store(wrappedTokenSource) + + return nil +} + func shouldAttemptToAuthenticate(errorCode codes.Code) bool { return errorCode == codes.Unauthenticated } +type proxyAuthTransport struct { + transport http.RoundTripper + proxyCredentialsFuture *PerRPCCredentialsFuture +} + +func (c *proxyAuthTransport) RoundTrip(req *http.Request) (*http.Response, error) { + // check if the proxy credentials future is initialized + if !c.proxyCredentialsFuture.IsInitialized() { + return nil, errors.New("proxy credentials future is not initialized") + } + + metadata, err := c.proxyCredentialsFuture.GetRequestMetadata(context.Background(), "") + if err != nil { + return nil, err + } + token := metadata[ProxyAuthorizationHeader] + req.Header.Add(ProxyAuthorizationHeader, token) + return c.transport.RoundTrip(req) +} + // Set up http client used in oauth2 -func setHTTPClientContext(ctx context.Context, cfg *Config) context.Context { +func setHTTPClientContext(ctx context.Context, cfg *Config, proxyCredentialsFuture *PerRPCCredentialsFuture) context.Context { httpClient := &http.Client{} + transport := &http.Transport{} if len(cfg.HTTPProxyURL.String()) > 0 { // create a transport that uses the proxy - transport := &http.Transport{ - Proxy: http.ProxyURL(&cfg.HTTPProxyURL.URL), + transport.Proxy = http.ProxyURL(&cfg.HTTPProxyURL.URL) + } + + if cfg.ProxyCommand != nil { + httpClient.Transport = &proxyAuthTransport{ + transport: transport, + proxyCredentialsFuture: proxyCredentialsFuture, } + } else { httpClient.Transport = transport } @@ -77,9 +131,9 @@ func setHTTPClientContext(ctx context.Context, cfg *Config) context.Context { // more. It'll fail hard if it couldn't do so (i.e. it will no longer attempt to send an unauthenticated request). Once // a token source has been created, it'll invoke the grpc pipeline again, this time the grpc.PerRPCCredentials should // be able to find and acquire a valid AccessToken to annotate the request with. -func NewAuthInterceptor(cfg *Config, tokenCache cache.TokenCache, credentialsFuture *PerRPCCredentialsFuture) grpc.UnaryClientInterceptor { +func NewAuthInterceptor(cfg *Config, tokenCache cache.TokenCache, credentialsFuture *PerRPCCredentialsFuture, proxyCredentialsFuture *PerRPCCredentialsFuture) grpc.UnaryClientInterceptor { return func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error { - ctx = setHTTPClientContext(ctx, cfg) + ctx = setHTTPClientContext(ctx, cfg, proxyCredentialsFuture) err := invoker(ctx, method, req, reply, cc, opts...) if err != nil { @@ -89,7 +143,7 @@ func NewAuthInterceptor(cfg *Config, tokenCache cache.TokenCache, credentialsFut // If the error we receive from executing the request expects if shouldAttemptToAuthenticate(st.Code()) { logger.Debugf(ctx, "Request failed due to [%v]. Attempting to establish an authenticated connection and trying again.", st.Code()) - newErr := MaterializeCredentials(ctx, cfg, tokenCache, credentialsFuture) + newErr := MaterializeCredentials(ctx, cfg, tokenCache, credentialsFuture, proxyCredentialsFuture) if newErr != nil { return fmt.Errorf("authentication error! Original Error: %v, Auth Error: %w", err, newErr) } @@ -102,3 +156,18 @@ func NewAuthInterceptor(cfg *Config, tokenCache cache.TokenCache, credentialsFut return err } } + +func NewProxyAuthInterceptor(cfg *Config, proxyCredentialsFuture *PerRPCCredentialsFuture) grpc.UnaryClientInterceptor { + return func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error { + + err := invoker(ctx, method, req, reply, cc, opts...) + if err != nil { + newErr := MaterializeProxyAuthCredentials(ctx, cfg, proxyCredentialsFuture) + if newErr != nil { + return fmt.Errorf("proxy authorization error! Original Error: %v, Proxy Auth Error: %w", err, newErr) + } + return invoker(ctx, method, req, reply, cc, opts...) + } + return err + } +} diff --git a/flyteidl/clients/go/admin/auth_interceptor_test.go b/flyteidl/clients/go/admin/auth_interceptor_test.go index fccd64769f9..ce99c992708 100644 --- a/flyteidl/clients/go/admin/auth_interceptor_test.go +++ b/flyteidl/clients/go/admin/auth_interceptor_test.go @@ -15,6 +15,7 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/mock" + "golang.org/x/oauth2" "google.golang.org/grpc" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" @@ -114,7 +115,8 @@ func newAuthMetadataServer(t testing.TB, port int, impl service.AuthMetadataServ func Test_newAuthInterceptor(t *testing.T) { t.Run("Other Error", func(t *testing.T) { f := NewPerRPCCredentialsFuture() - interceptor := NewAuthInterceptor(&Config{}, &mocks.TokenCache{}, f) + p := NewPerRPCCredentialsFuture() + interceptor := NewAuthInterceptor(&Config{}, &mocks.TokenCache{}, f, p) otherError := func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, opts ...grpc.CallOption) error { return status.New(codes.Canceled, "").Err() } @@ -146,11 +148,12 @@ func Test_newAuthInterceptor(t *testing.T) { assert.NoError(t, err) f := NewPerRPCCredentialsFuture() + p := NewPerRPCCredentialsFuture() interceptor := NewAuthInterceptor(&Config{ Endpoint: config.URL{URL: *u}, UseInsecureConnection: true, AuthType: AuthTypeClientSecret, - }, &mocks.TokenCache{}, f) + }, &mocks.TokenCache{}, f, p) unauthenticated := func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, opts ...grpc.CallOption) error { return status.New(codes.Unauthenticated, "").Err() } @@ -177,11 +180,13 @@ func Test_newAuthInterceptor(t *testing.T) { assert.NoError(t, err) f := NewPerRPCCredentialsFuture() + p := NewPerRPCCredentialsFuture() + interceptor := NewAuthInterceptor(&Config{ Endpoint: config.URL{URL: *u}, UseInsecureConnection: true, AuthType: AuthTypeClientSecret, - }, &mocks.TokenCache{}, f) + }, &mocks.TokenCache{}, f, p) authenticated := func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, opts ...grpc.CallOption) error { return nil } @@ -216,11 +221,13 @@ func Test_newAuthInterceptor(t *testing.T) { assert.NoError(t, err) f := NewPerRPCCredentialsFuture() + p := NewPerRPCCredentialsFuture() + interceptor := NewAuthInterceptor(&Config{ Endpoint: config.URL{URL: *u}, UseInsecureConnection: true, AuthType: AuthTypeClientSecret, - }, &mocks.TokenCache{}, f) + }, &mocks.TokenCache{}, f, p) unauthenticated := func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, opts ...grpc.CallOption) error { return status.New(codes.Aborted, "").Err() } @@ -246,6 +253,8 @@ func TestMaterializeCredentials(t *testing.T) { assert.NoError(t, err) f := NewPerRPCCredentialsFuture() + p := NewPerRPCCredentialsFuture() + err = MaterializeCredentials(ctx, &Config{ Endpoint: config.URL{URL: *u}, UseInsecureConnection: true, @@ -254,7 +263,7 @@ func TestMaterializeCredentials(t *testing.T) { Scopes: []string{"all"}, Audience: "http://localhost:30081", AuthorizationHeader: "authorization", - }, &mocks.TokenCache{}, f) + }, &mocks.TokenCache{}, f, p) assert.NoError(t, err) }) t.Run("Failed to fetch client metadata", func(t *testing.T) { @@ -271,13 +280,119 @@ func TestMaterializeCredentials(t *testing.T) { assert.NoError(t, err) f := NewPerRPCCredentialsFuture() + p := NewPerRPCCredentialsFuture() + err = MaterializeCredentials(ctx, &Config{ Endpoint: config.URL{URL: *u}, UseInsecureConnection: true, AuthType: AuthTypeClientSecret, TokenURL: fmt.Sprintf("http://localhost:%d/api/v1/token", port), Scopes: []string{"all"}, - }, &mocks.TokenCache{}, f) + }, &mocks.TokenCache{}, f, p) assert.EqualError(t, err, "failed to fetch client metadata. Error: rpc error: code = Unknown desc = expected err") }) } + +func TestNewProxyAuthInterceptor(t *testing.T) { + cfg := &Config{ + ProxyCommand: []string{"echo", "test-token"}, + } + + p := NewPerRPCCredentialsFuture() + + interceptor := NewProxyAuthInterceptor(cfg, p) + + ctx := context.Background() + method := "/test.method" + req := "request" + reply := "reply" + cc := new(grpc.ClientConn) + + errorInvoker := func(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, opts ...grpc.CallOption) error { + return errors.New("test error") + } + + // Call should return an error and trigger the interceptor to materialize proxy auth credentials + err := interceptor(ctx, method, req, reply, cc, errorInvoker) + assert.Error(t, err) + + // Check if proxyCredentialsFuture contains a proxy auth header token + creds, err := p.Get().GetRequestMetadata(ctx, "") + assert.True(t, p.IsInitialized()) + assert.NoError(t, err) + assert.Equal(t, "Bearer test-token", creds[ProxyAuthorizationHeader]) +} + +type testRoundTripper struct { + RoundTripFunc func(req *http.Request) (*http.Response, error) +} + +func (t *testRoundTripper) RoundTrip(req *http.Request) (*http.Response, error) { + return t.RoundTripFunc(req) +} + +func TestSetHTTPClientContext(t *testing.T) { + ctx := context.Background() + + t.Run("no proxy command and no proxy url", func(t *testing.T) { + cfg := &Config{} + + newCtx := setHTTPClientContext(ctx, cfg, nil) + + httpClient, ok := newCtx.Value(oauth2.HTTPClient).(*http.Client) + assert.True(t, ok) + + transport, ok := httpClient.Transport.(*http.Transport) + assert.True(t, ok) + assert.Nil(t, transport.Proxy) + }) + + t.Run("proxy url", func(t *testing.T) { + cfg := &Config{ + HTTPProxyURL: config. + URL{URL: url.URL{ + Scheme: "http", + Host: "localhost:8080", + }}, + } + newCtx := setHTTPClientContext(ctx, cfg, nil) + + httpClient, ok := newCtx.Value(oauth2.HTTPClient).(*http.Client) + assert.True(t, ok) + + transport, ok := httpClient.Transport.(*http.Transport) + assert.True(t, ok) + assert.NotNil(t, transport.Proxy) + }) + + t.Run("proxy command adds proxy-authorization header", func(t *testing.T) { + cfg := &Config{ + ProxyCommand: []string{"echo", "test-token-http-client"}, + } + + p := NewPerRPCCredentialsFuture() + err := MaterializeProxyAuthCredentials(ctx, cfg, p) + assert.NoError(t, err) + + newCtx := setHTTPClientContext(ctx, cfg, p) + + httpClient, ok := newCtx.Value(oauth2.HTTPClient).(*http.Client) + assert.True(t, ok) + + pat, ok := httpClient.Transport.(*proxyAuthTransport) + assert.True(t, ok) + + testRoundTripper := &testRoundTripper{ + RoundTripFunc: func(req *http.Request) (*http.Response, error) { + // Check if the ProxyAuthorizationHeader is correctly set + assert.Equal(t, "Bearer test-token-http-client", req.Header.Get(ProxyAuthorizationHeader)) + return &http.Response{StatusCode: http.StatusOK}, nil + }, + } + pat.transport = testRoundTripper + + req, _ := http.NewRequest("GET", "http://example.com", nil) + _, err = httpClient.Do(req) + assert.NoError(t, err) + }) +} diff --git a/flyteidl/clients/go/admin/client.go b/flyteidl/clients/go/admin/client.go index ffba612a096..7555f64045a 100644 --- a/flyteidl/clients/go/admin/client.go +++ b/flyteidl/clients/go/admin/client.go @@ -110,9 +110,9 @@ func getAuthenticationDialOption(ctx context.Context, cfg *Config, tokenSourcePr } // InitializeAuthMetadataClient creates a new anonymously Auth Metadata Service client. -func InitializeAuthMetadataClient(ctx context.Context, cfg *Config) (client service.AuthMetadataServiceClient, err error) { +func InitializeAuthMetadataClient(ctx context.Context, cfg *Config, proxyCredentialsFuture *PerRPCCredentialsFuture) (client service.AuthMetadataServiceClient, err error) { // Create an unauthenticated connection to fetch AuthMetadata - authMetadataConnection, err := NewAdminConnection(ctx, cfg) + authMetadataConnection, err := NewAdminConnection(ctx, cfg, proxyCredentialsFuture) if err != nil { return nil, fmt.Errorf("failed to initialized admin connection. Error: %w", err) } @@ -120,11 +120,11 @@ func InitializeAuthMetadataClient(ctx context.Context, cfg *Config) (client serv return service.NewAuthMetadataServiceClient(authMetadataConnection), nil } -func NewAdminConnection(ctx context.Context, cfg *Config, opts ...grpc.DialOption) (*grpc.ClientConn, error) { +func NewAdminConnection(ctx context.Context, cfg *Config, proxyCredentialsFuture *PerRPCCredentialsFuture, opts ...grpc.DialOption) (*grpc.ClientConn, error) { if opts == nil { // Initialize opts list to the potential number of options we will add. Initialization optimizes memory // allocation. - opts = make([]grpc.DialOption, 0, 5) + opts = make([]grpc.DialOption, 0, 7) } if cfg.UseInsecureConnection { @@ -153,6 +153,11 @@ func NewAdminConnection(ctx context.Context, cfg *Config, opts ...grpc.DialOptio opts = append(opts, GetAdditionalAdminClientConfigOptions(cfg)...) + if cfg.ProxyCommand != nil { + opts = append(opts, grpc.WithChainUnaryInterceptor(NewProxyAuthInterceptor(cfg, proxyCredentialsFuture))) + opts = append(opts, grpc.WithPerRPCCredentials(proxyCredentialsFuture)) + } + return grpc.Dial(cfg.Endpoint.String(), opts...) } @@ -172,15 +177,17 @@ func InitializeAdminClient(ctx context.Context, cfg *Config, opts ...grpc.DialOp // for the process. Note that if called with different cfg/dialoptions, it will not refresh the connection. func initializeClients(ctx context.Context, cfg *Config, tokenCache cache.TokenCache, opts ...grpc.DialOption) (*Clientset, error) { credentialsFuture := NewPerRPCCredentialsFuture() + proxyCredentialsFuture := NewPerRPCCredentialsFuture() + opts = append(opts, - grpc.WithChainUnaryInterceptor(NewAuthInterceptor(cfg, tokenCache, credentialsFuture)), + grpc.WithChainUnaryInterceptor(NewAuthInterceptor(cfg, tokenCache, credentialsFuture, proxyCredentialsFuture)), grpc.WithPerRPCCredentials(credentialsFuture)) if cfg.DefaultServiceConfig != "" { opts = append(opts, grpc.WithDefaultServiceConfig(cfg.DefaultServiceConfig)) } - adminConnection, err := NewAdminConnection(ctx, cfg, opts...) + adminConnection, err := NewAdminConnection(ctx, cfg, proxyCredentialsFuture, opts...) if err != nil { logger.Panicf(ctx, "failed to initialized Admin connection. Err: %s", err.Error()) } diff --git a/flyteidl/clients/go/admin/config.go b/flyteidl/clients/go/admin/config.go index 1eea36a89eb..03f2f8ecc28 100644 --- a/flyteidl/clients/go/admin/config.go +++ b/flyteidl/clients/go/admin/config.go @@ -74,6 +74,8 @@ type Config struct { Command []string `json:"command" pflag:",Command for external authentication token generation"` + ProxyCommand []string `json:"proxyCommand" pflag:",Command for external proxy-authorization token generation"` + // Set the gRPC service config formatted as a json string https://github.com/grpc/grpc/blob/master/doc/service_config.md // eg. {"loadBalancingConfig": [{"round_robin":{}}], "methodConfig": [{"name":[{"service": "foo", "method": "bar"}, {"service": "baz"}], "timeout": "1.000000001s"}]} // find the full schema here https://github.com/grpc/grpc-proto/blob/master/grpc/service_config/service_config.proto#L625 diff --git a/flyteidl/clients/go/admin/config_flags.go b/flyteidl/clients/go/admin/config_flags.go index 0472413b574..db1305c0b17 100755 --- a/flyteidl/clients/go/admin/config_flags.go +++ b/flyteidl/clients/go/admin/config_flags.go @@ -75,6 +75,7 @@ func (cfg Config) GetPFlagSet(prefix string) *pflag.FlagSet { cmdFlags.String(fmt.Sprintf("%v%v", prefix, "deviceFlowConfig.timeout"), defaultConfig.DeviceFlowConfig.Timeout.String(), "amount of time the device flow should complete or else it will be cancelled.") cmdFlags.String(fmt.Sprintf("%v%v", prefix, "deviceFlowConfig.pollInterval"), defaultConfig.DeviceFlowConfig.PollInterval.String(), "amount of time the device flow would poll the token endpoint if auth server doesn't return a polling interval. Okta and google IDP do return an interval'") cmdFlags.StringSlice(fmt.Sprintf("%v%v", prefix, "command"), defaultConfig.Command, "Command for external authentication token generation") + cmdFlags.StringSlice(fmt.Sprintf("%v%v", prefix, "proxyCommand"), defaultConfig.ProxyCommand, "Command for external proxy-authorization token generation") cmdFlags.String(fmt.Sprintf("%v%v", prefix, "defaultServiceConfig"), defaultConfig.DefaultServiceConfig, "") cmdFlags.String(fmt.Sprintf("%v%v", prefix, "httpProxyURL"), defaultConfig.HTTPProxyURL.String(), "OPTIONAL: HTTP Proxy to be used for OAuth requests.") return cmdFlags diff --git a/flyteidl/clients/go/admin/config_flags_test.go b/flyteidl/clients/go/admin/config_flags_test.go index 1fb1e2a2143..e815bcb5f3d 100755 --- a/flyteidl/clients/go/admin/config_flags_test.go +++ b/flyteidl/clients/go/admin/config_flags_test.go @@ -449,6 +449,20 @@ func TestConfig_SetFlags(t *testing.T) { } }) }) + t.Run("Test_proxyCommand", func(t *testing.T) { + + t.Run("Override", func(t *testing.T) { + testValue := join_Config(defaultConfig.ProxyCommand, ",") + + cmdFlags.Set("proxyCommand", testValue) + if vStringSlice, err := cmdFlags.GetStringSlice("proxyCommand"); err == nil { + testDecodeRaw_Config(t, join_Config(vStringSlice, ","), &actual.ProxyCommand) + + } else { + assert.FailNow(t, err.Error()) + } + }) + }) t.Run("Test_defaultServiceConfig", func(t *testing.T) { t.Run("Override", func(t *testing.T) { diff --git a/flyteidl/clients/go/admin/pkce/auth_flow_orchestrator.go b/flyteidl/clients/go/admin/pkce/auth_flow_orchestrator.go index 0ca48107ae0..cf5a85671ed 100644 --- a/flyteidl/clients/go/admin/pkce/auth_flow_orchestrator.go +++ b/flyteidl/clients/go/admin/pkce/auth_flow_orchestrator.go @@ -2,6 +2,7 @@ package pkce import ( "context" + "errors" "fmt" "net/http" "net/url" @@ -63,8 +64,15 @@ func (f TokenOrchestrator) FetchTokenFromAuthFlow(ctx context.Context) (*oauth2. serveMux := http.NewServeMux() server := &http.Server{Addr: redirectURL.Host, Handler: serveMux, ReadHeaderTimeout: 0} // Register the call back handler + + // Pass along http client used in oauth2 + httpClient, ok := ctx.Value(oauth2.HTTPClient).(*http.Client) + if !ok { + return nil, errors.New("Unable to retrieve httpClient used in oauth2 from context") + } + serveMux.HandleFunc(redirectURL.Path, getAuthServerCallbackHandler(f.ClientConfig, pkceCodeVerifier, - tokenChannel, errorChannel, stateString)) // the oauth2 callback endpoint + tokenChannel, errorChannel, stateString, httpClient)) // the oauth2 callback endpoint defer server.Close() go func() { diff --git a/flyteidl/clients/go/admin/pkce/handle_app_call_back.go b/flyteidl/clients/go/admin/pkce/handle_app_call_back.go index 72ac53883f9..775dcb4795f 100644 --- a/flyteidl/clients/go/admin/pkce/handle_app_call_back.go +++ b/flyteidl/clients/go/admin/pkce/handle_app_call_back.go @@ -11,7 +11,7 @@ import ( ) func getAuthServerCallbackHandler(c *oauth.Config, codeVerifier string, tokenChannel chan *oauth2.Token, - errorChannel chan error, stateString string) func(rw http.ResponseWriter, req *http.Request) { + errorChannel chan error, stateString string, client *http.Client) func(rw http.ResponseWriter, req *http.Request) { return func(rw http.ResponseWriter, req *http.Request) { _, _ = rw.Write([]byte(`

Flyte Authentication

`)) @@ -43,7 +43,8 @@ func getAuthServerCallbackHandler(c *oauth.Config, codeVerifier string, tokenCha var opts []oauth2.AuthCodeOption opts = append(opts, oauth2.SetAuthURLParam("code_verifier", codeVerifier)) - token, err := c.Exchange(context.Background(), req.URL.Query().Get("code"), opts...) + ctx := context.WithValue(context.Background(), oauth2.HTTPClient, client) + token, err := c.Exchange(ctx, req.URL.Query().Get("code"), opts...) if err != nil { errorChannel <- fmt.Errorf("error while exchanging auth code due to %v", err) _, _ = rw.Write([]byte(fmt.Sprintf(`

Couldn't get access token due to error: %s

`, err.Error()))) diff --git a/flyteidl/clients/go/admin/pkce/handle_app_call_back_test.go b/flyteidl/clients/go/admin/pkce/handle_app_call_back_test.go index c28b8333224..30c409002df 100644 --- a/flyteidl/clients/go/admin/pkce/handle_app_call_back_test.go +++ b/flyteidl/clients/go/admin/pkce/handle_app_call_back_test.go @@ -25,7 +25,7 @@ func HandleAppCallBackSetup(t *testing.T, state string) (tokenChannel chan *oaut errorChannel = make(chan error, 1) tokenChannel = make(chan *oauth2.Token) testAuthConfig = &oauth.Config{Config: &oauth2.Config{}, DeviceEndpoint: "dummyDeviceEndpoint"} - callBackFn = getAuthServerCallbackHandler(testAuthConfig, "", tokenChannel, errorChannel, state) + callBackFn = getAuthServerCallbackHandler(testAuthConfig, "", tokenChannel, errorChannel, state, &http.Client{}) assert.NotNil(t, callBackFn) req = &http.Request{ Method: http.MethodGet, diff --git a/flytepropeller/pkg/controller/controller.go b/flytepropeller/pkg/controller/controller.go index 1ff81ef28cf..85d68503256 100644 --- a/flytepropeller/pkg/controller/controller.go +++ b/flytepropeller/pkg/controller/controller.go @@ -315,7 +315,7 @@ func getAdminClient(ctx context.Context) (client service.AdminServiceClient, sig credentialsFuture := admin.NewPerRPCCredentialsFuture() opts := []grpc.DialOption{ - grpc.WithChainUnaryInterceptor(admin.NewAuthInterceptor(cfg, nil, credentialsFuture)), + grpc.WithChainUnaryInterceptor(admin.NewAuthInterceptor(cfg, nil, credentialsFuture, nil)), grpc.WithPerRPCCredentials(credentialsFuture), } From 0ca2d22b5f3abdf491944b67a249108a6cb1e343 Mon Sep 17 00:00:00 2001 From: Andrew Dye Date: Wed, 11 Oct 2023 16:42:44 -0700 Subject: [PATCH 07/18] Implement Spark pod template overrides (#4183) * Build SparkApplicationSpec using ToK8sPodSpec Signed-off-by: Andrew Dye * Comments Signed-off-by: Andrew Dye * Expect merged interruptible node selectors Signed-off-by: Andrew Dye * Fix lints Signed-off-by: Andrew Dye * Rename and cleanup TestBuildResourceContainer Signed-off-by: Andrew Dye * Add more test coverage to TestBuildResourcePodTemplate Signed-off-by: Andrew Dye --------- Signed-off-by: Andrew Dye --- .../flytek8s/non_interruptible.go | 35 ++ .../pluginmachinery/flytek8s/pod_helper.go | 9 + .../go/tasks/plugins/k8s/dask/dask.go | 50 +- .../go/tasks/plugins/k8s/spark/spark.go | 268 ++++++----- .../go/tasks/plugins/k8s/spark/spark_test.go | 437 +++++++++++++----- 5 files changed, 505 insertions(+), 294 deletions(-) create mode 100644 flyteplugins/go/tasks/pluginmachinery/flytek8s/non_interruptible.go diff --git a/flyteplugins/go/tasks/pluginmachinery/flytek8s/non_interruptible.go b/flyteplugins/go/tasks/pluginmachinery/flytek8s/non_interruptible.go new file mode 100644 index 00000000000..d2f5042cf8c --- /dev/null +++ b/flyteplugins/go/tasks/pluginmachinery/flytek8s/non_interruptible.go @@ -0,0 +1,35 @@ +package flytek8s + +import ( + pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" +) + +// Wraps a regular TaskExecutionMetadata and overrides the IsInterruptible method to always return false +// This is useful as the runner and the scheduler pods should never be interruptible +type NonInterruptibleTaskExecutionMetadata struct { + pluginsCore.TaskExecutionMetadata +} + +func (n NonInterruptibleTaskExecutionMetadata) IsInterruptible() bool { + return false +} + +// A wrapper around a regular TaskExecutionContext allowing to inject a custom TaskExecutionMetadata which is +// non-interruptible +type NonInterruptibleTaskExecutionContext struct { + pluginsCore.TaskExecutionContext + metadata NonInterruptibleTaskExecutionMetadata +} + +func (n NonInterruptibleTaskExecutionContext) TaskExecutionMetadata() pluginsCore.TaskExecutionMetadata { + return n.metadata +} + +func NewNonInterruptibleTaskExecutionContext(ctx pluginsCore.TaskExecutionContext) NonInterruptibleTaskExecutionContext { + return NonInterruptibleTaskExecutionContext{ + TaskExecutionContext: ctx, + metadata: NonInterruptibleTaskExecutionMetadata{ + ctx.TaskExecutionMetadata(), + }, + } +} diff --git a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go index 6ee8e417228..93de6a0f390 100644 --- a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go +++ b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go @@ -271,6 +271,15 @@ func ToK8sPodSpec(ctx context.Context, tCtx pluginsCore.TaskExecutionContext) (* return podSpec, objectMeta, primaryContainerName, nil } +func GetContainer(podSpec *v1.PodSpec, name string) (*v1.Container, error) { + for _, container := range podSpec.Containers { + if container.Name == name { + return &container, nil + } + } + return nil, pluginserrors.Errorf(pluginserrors.BadTaskSpecification, "invalid TaskSpecification, container [%s] not defined", name) +} + // getBasePodTemplate attempts to retrieve the PodTemplate to use as the base for k8s Pod configuration. This value can // come from one of the following: // (1) PodTemplate name in the TaskMetadata: This name is then looked up in the PodTemplateStore. diff --git a/flyteplugins/go/tasks/plugins/k8s/dask/dask.go b/flyteplugins/go/tasks/plugins/k8s/dask/dask.go index f8272b919a6..65050f5bb26 100644 --- a/flyteplugins/go/tasks/plugins/k8s/dask/dask.go +++ b/flyteplugins/go/tasks/plugins/k8s/dask/dask.go @@ -6,6 +6,12 @@ import ( "time" daskAPI "github.com/dask/dask-kubernetes/v2023/dask_kubernetes/operator/go_client/pkg/apis/kubernetes.dask.org/v1" + v1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/util/intstr" + "k8s.io/client-go/kubernetes/scheme" + "sigs.k8s.io/controller-runtime/pkg/client" + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins" "github.com/flyteorg/flyte/flyteplugins/go/tasks/errors" "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" @@ -15,11 +21,6 @@ import ( "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/tasklog" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" - v1 "k8s.io/api/core/v1" - metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" - "k8s.io/apimachinery/pkg/util/intstr" - "k8s.io/client-go/kubernetes/scheme" - "sigs.k8s.io/controller-runtime/pkg/client" ) const ( @@ -27,42 +28,12 @@ const ( KindDaskJob = "DaskJob" ) -// Wraps a regular TaskExecutionMetadata and overrides the IsInterruptible method to always return false -// This is useful as the runner and the scheduler pods should never be interruptible -type nonInterruptibleTaskExecutionMetadata struct { - pluginsCore.TaskExecutionMetadata -} - -func (n nonInterruptibleTaskExecutionMetadata) IsInterruptible() bool { - return false -} - -// A wrapper around a regular TaskExecutionContext allowing to inject a custom TaskExecutionMetadata which is -// non-interruptible -type nonInterruptibleTaskExecutionContext struct { - pluginsCore.TaskExecutionContext - metadata nonInterruptibleTaskExecutionMetadata -} - -func (n nonInterruptibleTaskExecutionContext) TaskExecutionMetadata() pluginsCore.TaskExecutionMetadata { - return n.metadata -} - func mergeMapInto(src map[string]string, dst map[string]string) { for key, value := range src { dst[key] = value } } -func getPrimaryContainer(spec *v1.PodSpec, primaryContainerName string) (*v1.Container, error) { - for _, container := range spec.Containers { - if container.Name == primaryContainerName { - return &container, nil - } - } - return nil, errors.Errorf(errors.BadTaskSpecification, "primary container [%v] not found in pod spec", primaryContainerName) -} - func replacePrimaryContainer(spec *v1.PodSpec, primaryContainerName string, container v1.Container) error { for i, c := range spec.Containers { if c.Name == primaryContainerName { @@ -104,8 +75,7 @@ func (p daskResourceHandler) BuildResource(ctx context.Context, taskCtx pluginsC if err != nil { return nil, err } - nonInterruptibleTaskMetadata := nonInterruptibleTaskExecutionMetadata{taskCtx.TaskExecutionMetadata()} - nonInterruptibleTaskCtx := nonInterruptibleTaskExecutionContext{taskCtx, nonInterruptibleTaskMetadata} + nonInterruptibleTaskCtx := flytek8s.NewNonInterruptibleTaskExecutionContext(taskCtx) nonInterruptiblePodSpec, _, _, err := flytek8s.ToK8sPodSpec(ctx, nonInterruptibleTaskCtx) if err != nil { return nil, err @@ -144,7 +114,7 @@ func (p daskResourceHandler) BuildResource(ctx context.Context, taskCtx pluginsC func createWorkerSpec(cluster plugins.DaskWorkerGroup, podSpec *v1.PodSpec, primaryContainerName string) (*daskAPI.WorkerSpec, error) { workerPodSpec := podSpec.DeepCopy() - primaryContainer, err := getPrimaryContainer(workerPodSpec, primaryContainerName) + primaryContainer, err := flytek8s.GetContainer(workerPodSpec, primaryContainerName) if err != nil { return nil, err } @@ -206,7 +176,7 @@ func createWorkerSpec(cluster plugins.DaskWorkerGroup, podSpec *v1.PodSpec, prim func createSchedulerSpec(scheduler plugins.DaskScheduler, clusterName string, podSpec *v1.PodSpec, primaryContainerName string) (*daskAPI.SchedulerSpec, error) { schedulerPodSpec := podSpec.DeepCopy() - primaryContainer, err := getPrimaryContainer(schedulerPodSpec, primaryContainerName) + primaryContainer, err := flytek8s.GetContainer(schedulerPodSpec, primaryContainerName) if err != nil { return nil, err } @@ -283,7 +253,7 @@ func createJobSpec(workerSpec daskAPI.WorkerSpec, schedulerSpec daskAPI.Schedule jobPodSpec := podSpec.DeepCopy() jobPodSpec.RestartPolicy = v1.RestartPolicyNever - primaryContainer, err := getPrimaryContainer(jobPodSpec, primaryContainerName) + primaryContainer, err := flytek8s.GetContainer(jobPodSpec, primaryContainerName) if err != nil { return nil, err } diff --git a/flyteplugins/go/tasks/plugins/k8s/spark/spark.go b/flyteplugins/go/tasks/plugins/k8s/spark/spark.go index 7c7ba34f9a3..d0506ccfb50 100644 --- a/flyteplugins/go/tasks/plugins/k8s/spark/spark.go +++ b/flyteplugins/go/tasks/plugins/k8s/spark/spark.go @@ -3,35 +3,29 @@ package spark import ( "context" "fmt" - - "sigs.k8s.io/controller-runtime/pkg/client" - + "regexp" "strconv" + "strings" + "time" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/tasklog" - - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/template" - - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" + sparkOp "github.com/GoogleCloudPlatform/spark-on-k8s-operator/pkg/apis/sparkoperator.k8s.io/v1beta2" + sparkOpConfig "github.com/GoogleCloudPlatform/spark-on-k8s-operator/pkg/config" + v1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/client-go/kubernetes/scheme" + "sigs.k8s.io/controller-runtime/pkg/client" + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins" "github.com/flyteorg/flyte/flyteplugins/go/tasks/errors" "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery" pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" - + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/tasklog" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" - "k8s.io/client-go/kubernetes/scheme" - - sparkOp "github.com/GoogleCloudPlatform/spark-on-k8s-operator/pkg/apis/sparkoperator.k8s.io/v1beta2" - "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" - "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins" - metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" - - "regexp" - "strings" - "time" ) const KindSparkApplication = "SparkApplication" @@ -80,70 +74,20 @@ func (sparkResourceHandler) BuildResource(ctx context.Context, taskCtx pluginsCo return nil, errors.Wrapf(errors.BadTaskSpecification, err, "invalid TaskSpecification [%v].", taskTemplate.GetCustom()) } - annotations := utils.UnionMaps(config.GetK8sPluginConfig().DefaultAnnotations, utils.CopyMap(taskCtx.TaskExecutionMetadata().GetAnnotations())) - labels := utils.UnionMaps(config.GetK8sPluginConfig().DefaultLabels, utils.CopyMap(taskCtx.TaskExecutionMetadata().GetLabels())) - container := taskTemplate.GetContainer() - - envVars := flytek8s.DecorateEnvVars(ctx, flytek8s.ToK8sEnvVar(container.GetEnv()), - taskCtx.TaskExecutionMetadata().GetEnvironmentVariables(), taskCtx.TaskExecutionMetadata().GetTaskExecutionID()) - - sparkEnvVars := make(map[string]string) - for _, envVar := range envVars { - sparkEnvVars[envVar.Name] = envVar.Value - } - - sparkEnvVars["FLYTE_MAX_ATTEMPTS"] = strconv.Itoa(int(taskCtx.TaskExecutionMetadata().GetMaxAttempts())) - - serviceAccountName := flytek8s.GetServiceAccountNameFromTaskExecutionMetadata(taskCtx.TaskExecutionMetadata()) - - if len(serviceAccountName) == 0 { - serviceAccountName = sparkTaskType - } - driverSpec := sparkOp.DriverSpec{ - SparkPodSpec: sparkOp.SparkPodSpec{ - Affinity: config.GetK8sPluginConfig().DefaultAffinity, - Annotations: annotations, - Labels: labels, - EnvVars: sparkEnvVars, - Image: &container.Image, - SecurityContenxt: config.GetK8sPluginConfig().DefaultPodSecurityContext.DeepCopy(), - DNSConfig: config.GetK8sPluginConfig().DefaultPodDNSConfig.DeepCopy(), - Tolerations: config.GetK8sPluginConfig().DefaultTolerations, - SchedulerName: &config.GetK8sPluginConfig().SchedulerName, - NodeSelector: config.GetK8sPluginConfig().DefaultNodeSelector, - HostNetwork: config.GetK8sPluginConfig().EnableHostNetworkingPod, - }, - ServiceAccount: &serviceAccountName, - } - - executorSpec := sparkOp.ExecutorSpec{ - SparkPodSpec: sparkOp.SparkPodSpec{ - Affinity: config.GetK8sPluginConfig().DefaultAffinity.DeepCopy(), - Annotations: annotations, - Labels: labels, - Image: &container.Image, - EnvVars: sparkEnvVars, - SecurityContenxt: config.GetK8sPluginConfig().DefaultPodSecurityContext.DeepCopy(), - DNSConfig: config.GetK8sPluginConfig().DefaultPodDNSConfig.DeepCopy(), - Tolerations: config.GetK8sPluginConfig().DefaultTolerations, - SchedulerName: &config.GetK8sPluginConfig().SchedulerName, - NodeSelector: config.GetK8sPluginConfig().DefaultNodeSelector, - HostNetwork: config.GetK8sPluginConfig().EnableHostNetworkingPod, - }, + sparkConfig := getSparkConfig(taskCtx, &sparkJob) + driverSpec, err := createDriverSpec(ctx, taskCtx, sparkConfig) + if err != nil { + return nil, err } - - modifiedArgs, err := template.Render(ctx, container.GetArgs(), template.Parameters{ - TaskExecMetadata: taskCtx.TaskExecutionMetadata(), - Inputs: taskCtx.InputReader(), - OutputPath: taskCtx.OutputWriter(), - Task: taskCtx.TaskReader(), - }) + executorSpec, err := createExecutorSpec(ctx, taskCtx, sparkConfig) if err != nil { return nil, err } + app := createSparkApplication(&sparkJob, sparkConfig, driverSpec, executorSpec) + return app, nil +} - // Hack: Retry submit failures in-case of resource limits hit. - submissionFailureRetries := int32(14) +func getSparkConfig(taskCtx pluginsCore.TaskExecutionContext, sparkJob *plugins.SparkJob) map[string]string { // Start with default config values. sparkConfig := make(map[string]string) for k, v := range GetSparkConfig().DefaultSparkConfig { @@ -165,57 +109,145 @@ func (sparkResourceHandler) BuildResource(ctx context.Context, taskCtx pluginsCo } // Set pod limits. - if len(sparkConfig["spark.kubernetes.driver.limit.cores"]) == 0 { + if len(sparkConfig[sparkOpConfig.SparkDriverCoreLimitKey]) == 0 { // spark.kubernetes.driver.request.cores takes precedence over spark.driver.cores - if len(sparkConfig["spark.kubernetes.driver.request.cores"]) != 0 { - sparkConfig["spark.kubernetes.driver.limit.cores"] = sparkConfig["spark.kubernetes.driver.request.cores"] + if len(sparkConfig[sparkOpConfig.SparkDriverCoreRequestKey]) != 0 { + sparkConfig[sparkOpConfig.SparkDriverCoreLimitKey] = sparkConfig[sparkOpConfig.SparkDriverCoreRequestKey] } else if len(sparkConfig["spark.driver.cores"]) != 0 { - sparkConfig["spark.kubernetes.driver.limit.cores"] = sparkConfig["spark.driver.cores"] + sparkConfig[sparkOpConfig.SparkDriverCoreLimitKey] = sparkConfig["spark.driver.cores"] } } - if len(sparkConfig["spark.kubernetes.executor.limit.cores"]) == 0 { + if len(sparkConfig[sparkOpConfig.SparkExecutorCoreLimitKey]) == 0 { // spark.kubernetes.executor.request.cores takes precedence over spark.executor.cores - if len(sparkConfig["spark.kubernetes.executor.request.cores"]) != 0 { - sparkConfig["spark.kubernetes.executor.limit.cores"] = sparkConfig["spark.kubernetes.executor.request.cores"] + if len(sparkConfig[sparkOpConfig.SparkExecutorCoreRequestKey]) != 0 { + sparkConfig[sparkOpConfig.SparkExecutorCoreLimitKey] = sparkConfig[sparkOpConfig.SparkExecutorCoreRequestKey] } else if len(sparkConfig["spark.executor.cores"]) != 0 { - sparkConfig["spark.kubernetes.executor.limit.cores"] = sparkConfig["spark.executor.cores"] + sparkConfig[sparkOpConfig.SparkExecutorCoreLimitKey] = sparkConfig["spark.executor.cores"] } } sparkConfig["spark.kubernetes.executor.podNamePrefix"] = taskCtx.TaskExecutionMetadata().GetTaskExecutionID().GetGeneratedName() sparkConfig["spark.kubernetes.driverEnv.FLYTE_START_TIME"] = strconv.FormatInt(time.Now().UnixNano()/1000000, 10) - // Add driver/executor defaults to CRD Driver/Executor Spec as well. - cores, err := strconv.ParseInt(sparkConfig["spark.driver.cores"], 10, 32) - if err == nil { - driverSpec.Cores = intPtr(int32(cores)) + return sparkConfig +} + +func serviceAccountName(metadata pluginsCore.TaskExecutionMetadata) string { + name := flytek8s.GetServiceAccountNameFromTaskExecutionMetadata(metadata) + if len(name) == 0 { + name = sparkTaskType + } + return name +} + +func createSparkPodSpec(taskCtx pluginsCore.TaskExecutionContext, podSpec *v1.PodSpec, container *v1.Container) *sparkOp.SparkPodSpec { + annotations := utils.UnionMaps(config.GetK8sPluginConfig().DefaultAnnotations, utils.CopyMap(taskCtx.TaskExecutionMetadata().GetAnnotations())) + labels := utils.UnionMaps(config.GetK8sPluginConfig().DefaultLabels, utils.CopyMap(taskCtx.TaskExecutionMetadata().GetLabels())) + + sparkEnvVars := make(map[string]string) + for _, envVar := range container.Env { + sparkEnvVars[envVar.Name] = envVar.Value } - driverSpec.Memory = strPtr(sparkConfig["spark.driver.memory"]) + sparkEnvVars["FLYTE_MAX_ATTEMPTS"] = strconv.Itoa(int(taskCtx.TaskExecutionMetadata().GetMaxAttempts())) + + spec := sparkOp.SparkPodSpec{ + Affinity: podSpec.Affinity, + Annotations: annotations, + Labels: labels, + EnvVars: sparkEnvVars, + Image: &container.Image, + SecurityContenxt: podSpec.SecurityContext.DeepCopy(), + DNSConfig: podSpec.DNSConfig.DeepCopy(), + Tolerations: podSpec.Tolerations, + SchedulerName: &podSpec.SchedulerName, + NodeSelector: podSpec.NodeSelector, + HostNetwork: &podSpec.HostNetwork, + } + return &spec +} - execCores, err := strconv.ParseInt(sparkConfig["spark.executor.cores"], 10, 32) - if err == nil { - executorSpec.Cores = intPtr(int32(execCores)) +type driverSpec struct { + sparkSpec *sparkOp.DriverSpec +} + +func createDriverSpec(ctx context.Context, taskCtx pluginsCore.TaskExecutionContext, sparkConfig map[string]string) (*driverSpec, error) { + // Spark driver pods should always run as non-interruptible + nonInterruptibleTaskCtx := flytek8s.NewNonInterruptibleTaskExecutionContext(taskCtx) + podSpec, _, primaryContainerName, err := flytek8s.ToK8sPodSpec(ctx, nonInterruptibleTaskCtx) + if err != nil { + return nil, err + } + primaryContainer, err := flytek8s.GetContainer(podSpec, primaryContainerName) + if err != nil { + return nil, err + } + sparkPodSpec := createSparkPodSpec(nonInterruptibleTaskCtx, podSpec, primaryContainer) + serviceAccountName := serviceAccountName(nonInterruptibleTaskCtx.TaskExecutionMetadata()) + spec := driverSpec{ + &sparkOp.DriverSpec{ + SparkPodSpec: *sparkPodSpec, + ServiceAccount: &serviceAccountName, + }, + } + if cores, err := strconv.ParseInt(sparkConfig["spark.driver.cores"], 10, 32); err == nil { + spec.sparkSpec.Cores = intPtr(int32(cores)) } + spec.sparkSpec.Memory = strPtr(sparkConfig["spark.driver.memory"]) + return &spec, nil +} - execCount, err := strconv.ParseInt(sparkConfig["spark.executor.instances"], 10, 32) - if err == nil { - executorSpec.Instances = intPtr(int32(execCount)) +type executorSpec struct { + container *v1.Container + sparkSpec *sparkOp.ExecutorSpec + serviceAccountName string +} + +func createExecutorSpec(ctx context.Context, taskCtx pluginsCore.TaskExecutionContext, sparkConfig map[string]string) (*executorSpec, error) { + podSpec, _, primaryContainerName, err := flytek8s.ToK8sPodSpec(ctx, taskCtx) + if err != nil { + return nil, err + } + primaryContainer, err := flytek8s.GetContainer(podSpec, primaryContainerName) + if err != nil { + return nil, err + } + sparkPodSpec := createSparkPodSpec(taskCtx, podSpec, primaryContainer) + serviceAccountName := serviceAccountName(taskCtx.TaskExecutionMetadata()) + spec := executorSpec{ + primaryContainer, + &sparkOp.ExecutorSpec{ + SparkPodSpec: *sparkPodSpec, + }, + serviceAccountName, } - executorSpec.Memory = strPtr(sparkConfig["spark.executor.memory"]) + if execCores, err := strconv.ParseInt(sparkConfig["spark.executor.cores"], 10, 32); err == nil { + spec.sparkSpec.Cores = intPtr(int32(execCores)) + } + if execCount, err := strconv.ParseInt(sparkConfig["spark.executor.instances"], 10, 32); err == nil { + spec.sparkSpec.Instances = intPtr(int32(execCount)) + } + spec.sparkSpec.Memory = strPtr(sparkConfig["spark.executor.memory"]) + return &spec, nil +} + +func createSparkApplication(sparkJob *plugins.SparkJob, sparkConfig map[string]string, driverSpec *driverSpec, + executorSpec *executorSpec) *sparkOp.SparkApplication { + // Hack: Retry submit failures in-case of resource limits hit. + submissionFailureRetries := int32(14) - j := &sparkOp.SparkApplication{ + app := &sparkOp.SparkApplication{ TypeMeta: metav1.TypeMeta{ Kind: KindSparkApplication, APIVersion: sparkOp.SchemeGroupVersion.String(), }, Spec: sparkOp.SparkApplicationSpec{ - ServiceAccount: &serviceAccountName, + ServiceAccount: &executorSpec.serviceAccountName, Type: getApplicationType(sparkJob.GetApplicationType()), - Image: &container.Image, - Arguments: modifiedArgs, - Driver: driverSpec, - Executor: executorSpec, + Image: &executorSpec.container.Image, + Arguments: executorSpec.container.Args, + Driver: *driverSpec.sparkSpec, + Executor: *executorSpec.sparkSpec, SparkConf: sparkConfig, HadoopConf: sparkJob.GetHadoopConf(), // SubmissionFailures handled here. Task Failures handled at Propeller/Job level. @@ -227,32 +259,16 @@ func (sparkResourceHandler) BuildResource(ctx context.Context, taskCtx pluginsCo } if val, ok := sparkConfig["spark.batchScheduler"]; ok { - j.Spec.BatchScheduler = &val + app.Spec.BatchScheduler = &val } if sparkJob.MainApplicationFile != "" { - j.Spec.MainApplicationFile = &sparkJob.MainApplicationFile + app.Spec.MainApplicationFile = &sparkJob.MainApplicationFile } if sparkJob.MainClass != "" { - j.Spec.MainClass = &sparkJob.MainClass - } - - // Spark driver pods should always run as non-interruptible. As such, we hardcode - // `interruptible=false` to explicitly add non-interruptible node selector - // requirements to the driver pods - flytek8s.ApplyInterruptibleNodeSelectorRequirement(false, j.Spec.Driver.Affinity) - - // Add Interruptible Tolerations/NodeSelector to only Executor pods. - // The Interruptible NodeSelector takes precedence over the DefaultNodeSelector - if taskCtx.TaskExecutionMetadata().IsInterruptible() { - j.Spec.Executor.Tolerations = append(j.Spec.Executor.Tolerations, config.GetK8sPluginConfig().InterruptibleTolerations...) - j.Spec.Executor.NodeSelector = config.GetK8sPluginConfig().InterruptibleNodeSelector + app.Spec.MainClass = &sparkJob.MainClass } - - // Add interruptible/non-interruptible node selector requirements to executor pod - flytek8s.ApplyInterruptibleNodeSelectorRequirement(taskCtx.TaskExecutionMetadata().IsInterruptible(), j.Spec.Executor.Affinity) - - return j, nil + return app } func addConfig(sparkConfig map[string]string, key string, value string) { diff --git a/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go b/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go index e981c0dce1e..18565fc0a48 100644 --- a/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go @@ -6,28 +6,25 @@ import ( "strconv" "testing" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" - - "github.com/stretchr/testify/mock" - - "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" - - pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" - - "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" - - pluginIOMocks "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/io/mocks" - sj "github.com/GoogleCloudPlatform/spark-on-k8s-operator/pkg/apis/sparkoperator.k8s.io/v1beta2" - "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" - "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins" + sparkOp "github.com/GoogleCloudPlatform/spark-on-k8s-operator/pkg/apis/sparkoperator.k8s.io/v1beta2" "github.com/golang/protobuf/jsonpb" structpb "github.com/golang/protobuf/ptypes/struct" "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/mock" corev1 "k8s.io/api/core/v1" v1 "k8s.io/apimachinery/pkg/apis/meta/v1" + + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" + "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/plugins" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" + pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" + pluginIOMocks "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/io/mocks" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" ) const sparkMainClass = "MainClass" @@ -87,7 +84,7 @@ func TestGetEventInfo(t *testing.T) { }, }, })) - taskCtx := dummySparkTaskContext(dummySparkTaskTemplate("blah-1", dummySparkConf), false) + taskCtx := dummySparkTaskContext(dummySparkTaskTemplateContainer("blah-1", dummySparkConf), false) info, err := getEventInfoForSpark(taskCtx, dummySparkApplication(sj.RunningState)) assert.NoError(t, err) assert.Len(t, info.Logs, 6) @@ -157,7 +154,7 @@ func TestGetTaskPhase(t *testing.T) { sparkResourceHandler := sparkResourceHandler{} ctx := context.TODO() - taskCtx := dummySparkTaskContext(dummySparkTaskTemplate("", dummySparkConf), false) + taskCtx := dummySparkTaskContext(dummySparkTaskTemplateContainer("", dummySparkConf), false) taskPhase, err := sparkResourceHandler.GetTaskPhase(ctx, taskCtx, dummySparkApplication(sj.NewState)) assert.NoError(t, err) assert.Equal(t, taskPhase.Phase(), pluginsCore.PhaseQueued) @@ -250,8 +247,33 @@ func dummySparkCustomObj(sparkConf map[string]string) *plugins.SparkJob { return &sparkJob } -func dummySparkTaskTemplate(id string, sparkConf map[string]string) *core.TaskTemplate { +func dummyPodSpec() *corev1.PodSpec { + return &corev1.PodSpec{ + InitContainers: []corev1.Container{ + { + Name: "init", + Image: testImage, + Args: testArgs, + }, + }, + Containers: []corev1.Container{ + { + Name: "primary", + Image: testImage, + Args: testArgs, + Env: flytek8s.ToK8sEnvVar(dummyEnvVars), + }, + { + Name: "secondary", + Image: testImage, + Args: testArgs, + Env: flytek8s.ToK8sEnvVar(dummyEnvVars), + }, + }, + } +} +func dummySparkTaskTemplateContainer(id string, sparkConf map[string]string) *core.TaskTemplate { sparkJob := dummySparkCustomObj(sparkConf) sparkJobJSON, err := utils.MarshalToString(sparkJob) if err != nil { @@ -279,6 +301,40 @@ func dummySparkTaskTemplate(id string, sparkConf map[string]string) *core.TaskTe } } +func dummySparkTaskTemplatePod(id string, sparkConf map[string]string, podSpec *corev1.PodSpec) *core.TaskTemplate { + sparkJob := dummySparkCustomObj(sparkConf) + sparkJobJSON, err := utils.MarshalToString(sparkJob) + if err != nil { + panic(err) + } + + structObj := structpb.Struct{} + + err = jsonpb.UnmarshalString(sparkJobJSON, &structObj) + if err != nil { + panic(err) + } + + podSpecPb, err := utils.MarshalObjToStruct(podSpec) + if err != nil { + panic(err) + } + + return &core.TaskTemplate{ + Id: &core.Identifier{Name: id}, + Type: "k8s_pod", + Target: &core.TaskTemplate_K8SPod{ + K8SPod: &core.K8SPod{ + PodSpec: podSpecPb, + }, + }, + Config: map[string]string{ + flytek8s.PrimaryContainerKey: "primary", + }, + Custom: &structObj, + } +} + func dummySparkTaskContext(taskTemplate *core.TaskTemplate, interruptible bool) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} @@ -312,6 +368,9 @@ func dummySparkTaskContext(taskTemplate *core.TaskTemplate, interruptible bool) }) tID.On("GetGeneratedName").Return("some-acceptable-name") + overrides := &mocks.TaskOverrides{} + overrides.On("GetResources").Return(&corev1.ResourceRequirements{}) + taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.On("GetTaskExecutionID").Return(tID) taskExecutionMetadata.On("GetNamespace").Return("test-namespace") @@ -327,30 +386,14 @@ func dummySparkTaskContext(taskTemplate *core.TaskTemplate, interruptible bool) taskExecutionMetadata.On("IsInterruptible").Return(interruptible) taskExecutionMetadata.On("GetMaxAttempts").Return(uint32(1)) taskExecutionMetadata.On("GetEnvironmentVariables").Return(nil) + taskExecutionMetadata.On("GetPlatformResources").Return(nil) + taskExecutionMetadata.On("GetOverrides").Return(overrides) + taskExecutionMetadata.On("GetK8sServiceAccount").Return("new-val") taskCtx.On("TaskExecutionMetadata").Return(taskExecutionMetadata) return taskCtx } -func TestBuildResourceSpark(t *testing.T) { - sparkResourceHandler := sparkResourceHandler{} - - // Case1: Valid Spark Task-Template - taskTemplate := dummySparkTaskTemplate("blah-1", dummySparkConf) - - // Set spark custom feature config. - assert.NoError(t, setSparkConfig(&Config{ - Features: []Feature{ - { - Name: "feature1", - SparkConfig: map[string]string{"spark.hadoop.feature1": "true"}, - }, - { - Name: "feature2", - SparkConfig: map[string]string{"spark.hadoop.feature2": "true"}, - }, - }, - })) - +func defaultPluginConfig() *config.K8sPluginConfig { // Set Interruptible Config runAsUser := int64(1000) dnsOptVal1 := "1" @@ -400,7 +443,7 @@ func TestBuildResourceSpark(t *testing.T) { }, } - // interruptible/non-interruptible nodeselector requirement + // Interruptible/non-interruptible nodeselector requirement interruptibleNodeSelectorRequirement := &corev1.NodeSelectorRequirement{ Key: "x/interruptible", Operator: corev1.NodeSelectorOpIn, @@ -413,9 +456,7 @@ func TestBuildResourceSpark(t *testing.T) { Values: []string{"true"}, } - // NonInterruptibleNodeSelectorRequirement - - assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + config := &config.K8sPluginConfig{ DefaultAffinity: defaultAffinity, DefaultPodSecurityContext: &corev1.PodSecurityContext{ RunAsUser: &runAsUser, @@ -465,8 +506,32 @@ func TestBuildResourceSpark(t *testing.T) { EnableHostNetworkingPod: &defaultPodHostNetwork, DefaultEnvVars: defaultEnvVars, DefaultEnvVarsFromEnv: defaultEnvVarsFromEnv, - }), - ) + } + return config +} + +func TestBuildResourceContainer(t *testing.T) { + sparkResourceHandler := sparkResourceHandler{} + + // Case1: Valid Spark Task-Template + taskTemplate := dummySparkTaskTemplateContainer("blah-1", dummySparkConf) + + // Set spark custom feature config. + assert.NoError(t, setSparkConfig(&Config{ + Features: []Feature{ + { + Name: "feature1", + SparkConfig: map[string]string{"spark.hadoop.feature1": "true"}, + }, + { + Name: "feature2", + SparkConfig: map[string]string{"spark.hadoop.feature2": "true"}, + }, + }, + })) + + defaultConfig := defaultPluginConfig() + assert.NoError(t, config.SetK8sPluginConfig(defaultConfig)) resource, err := sparkResourceHandler.BuildResource(context.TODO(), dummySparkTaskContext(taskTemplate, true)) assert.Nil(t, err) @@ -479,28 +544,16 @@ func TestBuildResourceSpark(t *testing.T) { assert.Equal(t, testArgs, sparkApp.Spec.Arguments) assert.Equal(t, testImage, *sparkApp.Spec.Image) assert.NotNil(t, sparkApp.Spec.Driver.SparkPodSpec.SecurityContenxt) - assert.Equal(t, *sparkApp.Spec.Driver.SparkPodSpec.SecurityContenxt.RunAsUser, runAsUser) + assert.Equal(t, *sparkApp.Spec.Driver.SparkPodSpec.SecurityContenxt.RunAsUser, *defaultConfig.DefaultPodSecurityContext.RunAsUser) assert.NotNil(t, sparkApp.Spec.Driver.DNSConfig) assert.Equal(t, []string{"8.8.8.8", "8.8.4.4"}, sparkApp.Spec.Driver.DNSConfig.Nameservers) - assert.Equal(t, "ndots", sparkApp.Spec.Driver.DNSConfig.Options[0].Name) - assert.Equal(t, dnsOptVal1, *sparkApp.Spec.Driver.DNSConfig.Options[0].Value) - assert.Equal(t, "single-request-reopen", sparkApp.Spec.Driver.DNSConfig.Options[1].Name) - assert.Equal(t, "timeout", sparkApp.Spec.Driver.DNSConfig.Options[2].Name) - assert.Equal(t, dnsOptVal2, *sparkApp.Spec.Driver.DNSConfig.Options[2].Value) - assert.Equal(t, "attempts", sparkApp.Spec.Driver.DNSConfig.Options[3].Name) - assert.Equal(t, dnsOptVal3, *sparkApp.Spec.Driver.DNSConfig.Options[3].Value) + assert.ElementsMatch(t, defaultConfig.DefaultPodDNSConfig.Options, sparkApp.Spec.Driver.DNSConfig.Options) assert.Equal(t, []string{"ns1.svc.cluster-domain.example", "my.dns.search.suffix"}, sparkApp.Spec.Driver.DNSConfig.Searches) assert.NotNil(t, sparkApp.Spec.Executor.SparkPodSpec.SecurityContenxt) - assert.Equal(t, *sparkApp.Spec.Executor.SparkPodSpec.SecurityContenxt.RunAsUser, runAsUser) + assert.Equal(t, *sparkApp.Spec.Executor.SparkPodSpec.SecurityContenxt.RunAsUser, *defaultConfig.DefaultPodSecurityContext.RunAsUser) assert.NotNil(t, sparkApp.Spec.Executor.DNSConfig) assert.NotNil(t, sparkApp.Spec.Executor.DNSConfig) - assert.Equal(t, "ndots", sparkApp.Spec.Executor.DNSConfig.Options[0].Name) - assert.Equal(t, dnsOptVal1, *sparkApp.Spec.Executor.DNSConfig.Options[0].Value) - assert.Equal(t, "single-request-reopen", sparkApp.Spec.Executor.DNSConfig.Options[1].Name) - assert.Equal(t, "timeout", sparkApp.Spec.Executor.DNSConfig.Options[2].Name) - assert.Equal(t, dnsOptVal2, *sparkApp.Spec.Executor.DNSConfig.Options[2].Value) - assert.Equal(t, "attempts", sparkApp.Spec.Executor.DNSConfig.Options[3].Name) - assert.Equal(t, dnsOptVal3, *sparkApp.Spec.Executor.DNSConfig.Options[3].Value) + assert.ElementsMatch(t, defaultConfig.DefaultPodDNSConfig.Options, sparkApp.Spec.Executor.DNSConfig.Options) assert.Equal(t, []string{"ns1.svc.cluster-domain.example", "my.dns.search.suffix"}, sparkApp.Spec.Executor.DNSConfig.Searches) //Validate Driver/Executor Spec. @@ -515,19 +568,19 @@ func TestBuildResourceSpark(t *testing.T) { assert.Equal(t, dummySparkConf["spark.driver.memory"], *sparkApp.Spec.Driver.Memory) assert.Equal(t, dummySparkConf["spark.executor.memory"], *sparkApp.Spec.Executor.Memory) assert.Equal(t, dummySparkConf["spark.batchScheduler"], *sparkApp.Spec.BatchScheduler) - assert.Equal(t, schedulerName, *sparkApp.Spec.Executor.SchedulerName) - assert.Equal(t, schedulerName, *sparkApp.Spec.Driver.SchedulerName) - assert.Equal(t, defaultPodHostNetwork, *sparkApp.Spec.Executor.HostNetwork) - assert.Equal(t, defaultPodHostNetwork, *sparkApp.Spec.Driver.HostNetwork) + assert.Equal(t, defaultConfig.SchedulerName, *sparkApp.Spec.Executor.SchedulerName) + assert.Equal(t, defaultConfig.SchedulerName, *sparkApp.Spec.Driver.SchedulerName) + assert.Equal(t, *defaultConfig.EnableHostNetworkingPod, *sparkApp.Spec.Executor.HostNetwork) + assert.Equal(t, *defaultConfig.EnableHostNetworkingPod, *sparkApp.Spec.Driver.HostNetwork) // Validate - // * Interruptible Toleration and NodeSelector set for Executor but not Driver. - // * Validate Default NodeSelector set for Driver but overwritten with Interruptible NodeSelector for Executor. - // * Default Tolerations set for both Driver and Executor. - // * Interruptible/Non-Interruptible NodeSelectorRequirements set for Executor Affinity but not Driver Affinity. + // * Default tolerations set for both Driver and Executor. + // * Interruptible tolerations and node selector set for Executor but not Driver. + // * Default node selector set for both Driver and Executor. + // * Interruptible node selector requirements set for Executor Affinity, non-interruptiblefir Driver Affinity. assert.Equal(t, 1, len(sparkApp.Spec.Driver.Tolerations)) assert.Equal(t, 1, len(sparkApp.Spec.Driver.NodeSelector)) - assert.Equal(t, defaultNodeSelector, sparkApp.Spec.Driver.NodeSelector) + assert.Equal(t, defaultConfig.DefaultNodeSelector, sparkApp.Spec.Driver.NodeSelector) tolDriverDefault := sparkApp.Spec.Driver.Tolerations[0] assert.Equal(t, tolDriverDefault.Key, "x/flyte") assert.Equal(t, tolDriverDefault.Value, "default") @@ -535,21 +588,23 @@ func TestBuildResourceSpark(t *testing.T) { assert.Equal(t, tolDriverDefault.Effect, corev1.TaintEffect("NoSchedule")) assert.Equal(t, 2, len(sparkApp.Spec.Executor.Tolerations)) - assert.Equal(t, 1, len(sparkApp.Spec.Executor.NodeSelector)) - assert.Equal(t, interruptibleNodeSelector, sparkApp.Spec.Executor.NodeSelector) - - tolExecDefault := sparkApp.Spec.Executor.Tolerations[0] - assert.Equal(t, tolExecDefault.Key, "x/flyte") - assert.Equal(t, tolExecDefault.Value, "default") - assert.Equal(t, tolExecDefault.Operator, corev1.TolerationOperator("Equal")) - assert.Equal(t, tolExecDefault.Effect, corev1.TaintEffect("NoSchedule")) + assert.Equal(t, 2, len(sparkApp.Spec.Executor.NodeSelector)) + assert.Equal(t, map[string]string{ + "x/default": "true", + "x/interruptible": "true", + }, sparkApp.Spec.Executor.NodeSelector) - tolExecInterrupt := sparkApp.Spec.Executor.Tolerations[1] + tolExecInterrupt := sparkApp.Spec.Executor.Tolerations[0] assert.Equal(t, tolExecInterrupt.Key, "x/flyte") assert.Equal(t, tolExecInterrupt.Value, "interruptible") assert.Equal(t, tolExecInterrupt.Operator, corev1.TolerationOperator("Equal")) assert.Equal(t, tolExecInterrupt.Effect, corev1.TaintEffect("NoSchedule")) - assert.Equal(t, "true", sparkApp.Spec.Executor.NodeSelector["x/interruptible"]) + + tolExecDefault := sparkApp.Spec.Executor.Tolerations[1] + assert.Equal(t, tolExecDefault.Key, "x/flyte") + assert.Equal(t, tolExecDefault.Value, "default") + assert.Equal(t, tolExecDefault.Operator, corev1.TolerationOperator("Equal")) + assert.Equal(t, tolExecDefault.Effect, corev1.TaintEffect("NoSchedule")) for confKey, confVal := range dummySparkConf { exists := false @@ -583,31 +638,36 @@ func TestBuildResourceSpark(t *testing.T) { assert.Equal(t, dummySparkConf["spark.flyteorg.feature3.enabled"], sparkApp.Spec.SparkConf["spark.flyteorg.feature3.enabled"]) assert.Equal(t, len(sparkApp.Spec.Driver.EnvVars["FLYTE_MAX_ATTEMPTS"]), 1) - assert.Equal(t, sparkApp.Spec.Driver.EnvVars["foo"], defaultEnvVars["foo"]) - assert.Equal(t, sparkApp.Spec.Executor.EnvVars["foo"], defaultEnvVars["foo"]) - assert.Equal(t, sparkApp.Spec.Driver.EnvVars["fooEnv"], targetValueFromEnv) - assert.Equal(t, sparkApp.Spec.Executor.EnvVars["fooEnv"], targetValueFromEnv) - - assert.Equal( - t, - sparkApp.Spec.Driver.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - defaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - ) - assert.Equal( - t, - sparkApp.Spec.Driver.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[1], - *nonInterruptibleNodeSelectorRequirement, - ) - assert.Equal( - t, - sparkApp.Spec.Executor.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - defaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - ) - assert.Equal( - t, - sparkApp.Spec.Executor.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[1], - *interruptibleNodeSelectorRequirement, - ) + assert.Equal(t, defaultConfig.DefaultEnvVars["foo"], sparkApp.Spec.Driver.EnvVars["foo"]) + assert.Equal(t, defaultConfig.DefaultEnvVars["foo"], sparkApp.Spec.Executor.EnvVars["foo"]) + assert.Equal(t, defaultConfig.DefaultEnvVars["fooEnv"], sparkApp.Spec.Driver.EnvVars["fooEnv"]) + assert.Equal(t, defaultConfig.DefaultEnvVars["fooEnv"], sparkApp.Spec.Executor.EnvVars["fooEnv"]) + + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.NonInterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Driver.Affinity.NodeAffinity) + + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.InterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Executor.Affinity.NodeAffinity) // Case 2: Driver/Executor request cores set. dummyConfWithRequest := make(map[string]string) @@ -619,7 +679,7 @@ func TestBuildResourceSpark(t *testing.T) { dummyConfWithRequest["spark.kubernetes.driver.request.cores"] = "3" dummyConfWithRequest["spark.kubernetes.executor.request.cores"] = "4" - taskTemplate = dummySparkTaskTemplate("blah-1", dummyConfWithRequest) + taskTemplate = dummySparkTaskTemplateContainer("blah-1", dummyConfWithRequest) resource, err = sparkResourceHandler.BuildResource(context.TODO(), dummySparkTaskContext(taskTemplate, false)) assert.Nil(t, err) assert.NotNil(t, resource) @@ -640,36 +700,41 @@ func TestBuildResourceSpark(t *testing.T) { // Validate that the default Toleration and NodeSelector are set for both Driver and Executors. assert.Equal(t, 1, len(sparkApp.Spec.Driver.Tolerations)) assert.Equal(t, 1, len(sparkApp.Spec.Driver.NodeSelector)) - assert.Equal(t, defaultNodeSelector, sparkApp.Spec.Driver.NodeSelector) + assert.Equal(t, defaultConfig.DefaultNodeSelector, sparkApp.Spec.Driver.NodeSelector) assert.Equal(t, 1, len(sparkApp.Spec.Executor.Tolerations)) assert.Equal(t, 1, len(sparkApp.Spec.Executor.NodeSelector)) - assert.Equal(t, defaultNodeSelector, sparkApp.Spec.Executor.NodeSelector) + assert.Equal(t, defaultConfig.DefaultNodeSelector, sparkApp.Spec.Executor.NodeSelector) assert.Equal(t, sparkApp.Spec.Executor.Tolerations[0].Key, "x/flyte") assert.Equal(t, sparkApp.Spec.Executor.Tolerations[0].Value, "default") assert.Equal(t, sparkApp.Spec.Driver.Tolerations[0].Key, "x/flyte") assert.Equal(t, sparkApp.Spec.Driver.Tolerations[0].Value, "default") // Validate correct affinity and nodeselector requirements are set for both Driver and Executors. - assert.Equal( - t, - sparkApp.Spec.Driver.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - defaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - ) - assert.Equal( - t, - sparkApp.Spec.Driver.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[1], - *nonInterruptibleNodeSelectorRequirement, - ) - assert.Equal( - t, - sparkApp.Spec.Executor.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - defaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], - ) - assert.Equal( - t, - sparkApp.Spec.Executor.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[1], - *nonInterruptibleNodeSelectorRequirement, - ) + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.NonInterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Driver.Affinity.NodeAffinity) + + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.NonInterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Executor.Affinity.NodeAffinity) // Case 4: Invalid Spark Task-Template taskTemplate.Custom = nil @@ -678,6 +743,122 @@ func TestBuildResourceSpark(t *testing.T) { assert.Nil(t, resource) } +func TestBuildResourcePodTemplate(t *testing.T) { + defaultConfig := defaultPluginConfig() + assert.NoError(t, config.SetK8sPluginConfig(defaultConfig)) + extraToleration := corev1.Toleration{ + Key: "x/flyte", + Value: "extra", + Operator: "Equal", + } + podSpec := dummyPodSpec() + podSpec.Tolerations = append(podSpec.Tolerations, extraToleration) + podSpec.NodeSelector = map[string]string{"x/custom": "foo"} + taskTemplate := dummySparkTaskTemplatePod("blah-1", dummySparkConf, podSpec) + taskTemplate.GetK8SPod() + sparkResourceHandler := sparkResourceHandler{} + + taskCtx := dummySparkTaskContext(taskTemplate, true) + resource, err := sparkResourceHandler.BuildResource(context.TODO(), taskCtx) + + assert.Nil(t, err) + assert.NotNil(t, resource) + sparkApp, ok := resource.(*sj.SparkApplication) + assert.True(t, ok) + + // Application + assert.Equal(t, v1.TypeMeta{ + Kind: KindSparkApplication, + APIVersion: sparkOp.SchemeGroupVersion.String(), + }, sparkApp.TypeMeta) + + // Application spec + assert.Equal(t, flytek8s.GetServiceAccountNameFromTaskExecutionMetadata(taskCtx.TaskExecutionMetadata()), *sparkApp.Spec.ServiceAccount) + assert.Equal(t, sparkOp.PythonApplicationType, sparkApp.Spec.Type) + assert.Equal(t, testImage, *sparkApp.Spec.Image) + assert.Equal(t, testArgs, sparkApp.Spec.Arguments) + assert.Equal(t, sparkOp.RestartPolicy{ + Type: sparkOp.OnFailure, + OnSubmissionFailureRetries: intPtr(int32(14)), + }, sparkApp.Spec.RestartPolicy) + assert.Equal(t, sparkMainClass, *sparkApp.Spec.MainClass) + assert.Equal(t, sparkApplicationFile, *sparkApp.Spec.MainApplicationFile) + + // Driver + assert.Equal(t, utils.UnionMaps(defaultConfig.DefaultAnnotations, map[string]string{"annotation-1": "val1"}), sparkApp.Spec.Driver.Annotations) + assert.Equal(t, utils.UnionMaps(defaultConfig.DefaultLabels, map[string]string{"label-1": "val1"}), sparkApp.Spec.Driver.Labels) + assert.Equal(t, len(sparkApp.Spec.Driver.EnvVars["FLYTE_MAX_ATTEMPTS"]), 1) + assert.Equal(t, defaultConfig.DefaultEnvVars["foo"], sparkApp.Spec.Driver.EnvVars["foo"]) + assert.Equal(t, defaultConfig.DefaultEnvVars["fooEnv"], sparkApp.Spec.Driver.EnvVars["fooEnv"]) + assert.Equal(t, testImage, *sparkApp.Spec.Driver.Image) + assert.Equal(t, flytek8s.GetServiceAccountNameFromTaskExecutionMetadata(taskCtx.TaskExecutionMetadata()), *sparkApp.Spec.Driver.ServiceAccount) + assert.Equal(t, defaultConfig.DefaultPodSecurityContext, sparkApp.Spec.Driver.SecurityContenxt) + assert.Equal(t, defaultConfig.DefaultPodDNSConfig, sparkApp.Spec.Driver.DNSConfig) + assert.Equal(t, defaultConfig.EnableHostNetworkingPod, sparkApp.Spec.Driver.HostNetwork) + assert.Equal(t, defaultConfig.SchedulerName, *sparkApp.Spec.Driver.SchedulerName) + assert.Equal(t, []corev1.Toleration{ + defaultConfig.DefaultTolerations[0], + extraToleration, + }, sparkApp.Spec.Driver.Tolerations) + assert.Equal(t, map[string]string{ + "x/default": "true", + "x/custom": "foo", + }, sparkApp.Spec.Driver.NodeSelector) + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.NonInterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Driver.Affinity.NodeAffinity) + cores, _ := strconv.ParseInt(dummySparkConf["spark.driver.cores"], 10, 32) + assert.Equal(t, intPtr(int32(cores)), sparkApp.Spec.Driver.Cores) + assert.Equal(t, dummySparkConf["spark.driver.memory"], *sparkApp.Spec.Driver.Memory) + + // Executor + assert.Equal(t, utils.UnionMaps(defaultConfig.DefaultAnnotations, map[string]string{"annotation-1": "val1"}), sparkApp.Spec.Executor.Annotations) + assert.Equal(t, utils.UnionMaps(defaultConfig.DefaultLabels, map[string]string{"label-1": "val1"}), sparkApp.Spec.Executor.Labels) + assert.Equal(t, defaultConfig.DefaultEnvVars["foo"], sparkApp.Spec.Executor.EnvVars["foo"]) + assert.Equal(t, defaultConfig.DefaultEnvVars["fooEnv"], sparkApp.Spec.Executor.EnvVars["fooEnv"]) + assert.Equal(t, testImage, *sparkApp.Spec.Executor.Image) + assert.Equal(t, defaultConfig.DefaultPodSecurityContext, sparkApp.Spec.Executor.SecurityContenxt) + assert.Equal(t, defaultConfig.DefaultPodDNSConfig, sparkApp.Spec.Executor.DNSConfig) + assert.Equal(t, defaultConfig.EnableHostNetworkingPod, sparkApp.Spec.Executor.HostNetwork) + assert.Equal(t, defaultConfig.SchedulerName, *sparkApp.Spec.Executor.SchedulerName) + assert.ElementsMatch(t, []corev1.Toleration{ + defaultConfig.DefaultTolerations[0], + extraToleration, + defaultConfig.InterruptibleTolerations[0], + }, sparkApp.Spec.Executor.Tolerations) + assert.Equal(t, map[string]string{ + "x/default": "true", + "x/custom": "foo", + "x/interruptible": "true", + }, sparkApp.Spec.Executor.NodeSelector) + assert.Equal(t, &corev1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ + NodeSelectorTerms: []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + defaultConfig.DefaultAffinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions[0], + *defaultConfig.InterruptibleNodeSelectorRequirement, + }, + }, + }, + }, + }, sparkApp.Spec.Executor.Affinity.NodeAffinity) + cores, _ = strconv.ParseInt(dummySparkConf["spark.executor.cores"], 10, 32) + instances, _ := strconv.ParseInt(dummySparkConf["spark.executor.instances"], 10, 32) + assert.Equal(t, intPtr(int32(instances)), sparkApp.Spec.Executor.Instances) + assert.Equal(t, intPtr(int32(cores)), sparkApp.Spec.Executor.Cores) + assert.Equal(t, dummySparkConf["spark.executor.memory"], *sparkApp.Spec.Executor.Memory) +} + func TestGetPropertiesSpark(t *testing.T) { sparkResourceHandler := sparkResourceHandler{} expected := k8s.PluginProperties{} From 885435108b5cb1e6f3b8458efbe796402dd57fd2 Mon Sep 17 00:00:00 2001 From: Shovit <119912892+Virtual4087@users.noreply.github.com> Date: Wed, 11 Oct 2023 23:59:27 -0700 Subject: [PATCH 08/18] Fixed bug for incorrect name (#4175) --------- Signed-off-by: Shovit <119912892+Virtual4087@users.noreply.github.com> --- .../pkg/manager/impl/validation/execution_validator.go | 2 +- .../pkg/manager/impl/validation/execution_validator_test.go | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/flyteadmin/pkg/manager/impl/validation/execution_validator.go b/flyteadmin/pkg/manager/impl/validation/execution_validator.go index b5056589a8c..a317dced17a 100644 --- a/flyteadmin/pkg/manager/impl/validation/execution_validator.go +++ b/flyteadmin/pkg/manager/impl/validation/execution_validator.go @@ -131,7 +131,7 @@ func CheckValidExecutionID(executionID, fieldName string) error { matched := executionIDRegex.MatchString(executionID) if !matched { - return errors.NewFlyteAdminErrorf(codes.InvalidArgument, "invalid %s format: %s", fieldName, executionID) + return errors.NewFlyteAdminErrorf(codes.InvalidArgument, "invalid %s format: %s, does not match regex '%s'", fieldName, executionID, executionIDRegex.String()) } return nil diff --git a/flyteadmin/pkg/manager/impl/validation/execution_validator_test.go b/flyteadmin/pkg/manager/impl/validation/execution_validator_test.go index c3c45d4d449..70d78c035d4 100644 --- a/flyteadmin/pkg/manager/impl/validation/execution_validator_test.go +++ b/flyteadmin/pkg/manager/impl/validation/execution_validator_test.go @@ -42,7 +42,7 @@ func TestValidateExecInvalidName(t *testing.T) { request := testutils.GetExecutionRequest() request.Name = "12345" err := ValidateExecutionRequest(context.Background(), request, testutils.GetRepoWithDefaultProject(), execConfig) - assert.EqualError(t, err, "invalid name format: 12345") + assert.EqualError(t, err, "invalid name format: 12345, does not match regex '^[a-z][a-z\\-0-9]*$'") request.Name = "e2345" err = ValidateExecutionRequest(context.Background(), request, testutils.GetRepoWithDefaultProject(), execConfig) @@ -170,10 +170,10 @@ func TestValidExecutionIdInvalidLength(t *testing.T) { func TestValidExecutionIdInvalidChars(t *testing.T) { err := CheckValidExecutionID("a_sdd", "a") assert.NotNil(t, err) - assert.EqualError(t, err, "invalid a format: a_sdd") + assert.EqualError(t, err, "invalid a format: a_sdd, does not match regex '^[a-z][a-z\\-0-9]*$'") err = CheckValidExecutionID("asd@", "a") assert.NotNil(t, err) - assert.EqualError(t, err, "invalid a format: asd@") + assert.EqualError(t, err, "invalid a format: asd@, does not match regex '^[a-z][a-z\\-0-9]*$'") } func TestValidateCreateWorkflowEventRequest(t *testing.T) { From 737ef237b2f37ffa1b0ffc17441a19103027ec81 Mon Sep 17 00:00:00 2001 From: Jeev B Date: Thu, 12 Oct 2023 09:42:33 -0700 Subject: [PATCH 09/18] [Extended Resources] GPU Accelerators (#4172) * Add IDL changes Signed-off-by: Jeev B * Add propeller changes Signed-off-by: Jeev B * Add flyteplugins changes Signed-off-by: Jeev B * Cleanup tests Signed-off-by: Jeev B * Add additional comments to IDL fields Signed-off-by: Jeev B * Fix issues from merge Signed-off-by: Jeev B * First pass through PR Signed-off-by: Jeev B * Address PR comments Signed-off-by: Jeev B * No auto injection of toleration for explicitly requested unpartitioned GPUs Signed-off-by: Jeev B --------- Signed-off-by: Jeev B --- .../gen/pb-cpp/flyteidl/admin/agent.pb.cc | 2 +- flyteidl/gen/pb-cpp/flyteidl/admin/task.pb.cc | 2 +- .../gen/pb-cpp/flyteidl/core/compiler.pb.cc | 2 +- .../pb-cpp/flyteidl/core/dynamic_job.pb.cc | 2 +- flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.cc | 1444 ++++++++--- flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.h | 618 ++++- .../gen/pb-cpp/flyteidl/core/workflow.pb.cc | 98 +- .../gen/pb-cpp/flyteidl/core/workflow.pb.h | 55 + .../flyteidl/core/workflow_closure.pb.cc | 2 +- .../service/external_plugin_service.pb.cc | 2 +- flyteidl/gen/pb-go/flyteidl/core/tasks.pb.go | 397 ++- .../pb-go/flyteidl/core/tasks.pb.validate.go | 164 ++ .../gen/pb-go/flyteidl/core/workflow.pb.go | 204 +- .../flyteidl/core/workflow.pb.validate.go | 10 + .../pb-go/flyteidl/service/admin.swagger.json | 36 + .../pb-go/flyteidl/service/agent.swagger.json | 32 + .../external_plugin_service.swagger.json | 32 + .../flyteidl/service/flyteadmin/README.md | 2 + .../service/flyteadmin/api/swagger.yaml | 507 ++++ .../model_core_extended_resources.go | 16 + .../flyteadmin/model_core_gpu_accelerator.go | 19 + .../model_core_task_node_overrides.go | 2 + .../flyteadmin/model_core_task_template.go | 2 + .../gen/pb-go/flyteidl/service/openapi.go | 4 +- flyteidl/gen/pb-java/flyteidl/core/Tasks.java | 2133 ++++++++++++++++- .../gen/pb-java/flyteidl/core/Workflow.java | 280 ++- flyteidl/gen/pb-js/flyteidl.d.ts | 131 + flyteidl/gen/pb-js/flyteidl.js | 315 +++ .../gen/pb_python/flyteidl/core/tasks_pb2.py | 86 +- .../gen/pb_python/flyteidl/core/tasks_pb2.pyi | 22 +- .../pb_python/flyteidl/core/workflow_pb2.py | 6 +- .../pb_python/flyteidl/core/workflow_pb2.pyi | 6 +- .../flyteidl/service/flyteadmin/README.md | 2 + .../service/flyteadmin/flyteadmin/__init__.py | 2 + .../flyteadmin/flyteadmin/models/__init__.py | 2 + .../models/core_extended_resources.py | 119 + .../flyteadmin/models/core_gpu_accelerator.py | 171 ++ .../models/core_task_node_overrides.py | 35 +- .../flyteadmin/models/core_task_template.py | 31 +- .../test/test_core_extended_resources.py | 40 + .../test/test_core_gpu_accelerator.py | 40 + flyteidl/gen/pb_rust/flyteidl.core.rs | 46 + flyteidl/protos/flyteidl/core/tasks.proto | 29 + flyteidl/protos/flyteidl/core/workflow.proto | 4 + .../pluginmachinery/core/exec_metadata.go | 1 + .../core/mocks/task_overrides.go | 36 + .../pluginmachinery/flytek8s/config/config.go | 16 +- .../pluginmachinery/flytek8s/pod_helper.go | 182 +- .../flytek8s/pod_helper_test.go | 682 +++++- .../plugins/array/k8s/management_test.go | 1 + .../go/tasks/plugins/k8s/dask/dask_test.go | 174 +- .../plugins/k8s/kfoperators/mpi/mpi_test.go | 161 +- .../k8s/kfoperators/pytorch/pytorch_test.go | 215 +- .../kfoperators/tensorflow/tensorflow_test.go | 183 +- .../tasks/plugins/k8s/pod/container_test.go | 167 +- .../go/tasks/plugins/k8s/pod/sidecar_test.go | 215 +- .../go/tasks/plugins/k8s/ray/ray_test.go | 156 +- .../go/tasks/plugins/k8s/spark/spark_test.go | 2 + flyteplugins/tests/end_to_end.go | 1 + .../pkg/apis/flyteworkflow/v1alpha1/iface.go | 1 + .../v1alpha1/mocks/ExecutableNode.go | 39 +- .../pkg/apis/flyteworkflow/v1alpha1/nodes.go | 35 + .../pkg/compiler/transformers/k8s/node.go | 14 +- .../compiler/transformers/k8s/node_test.go | 24 + 64 files changed, 8626 insertions(+), 833 deletions(-) create mode 100644 flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_extended_resources.go create mode 100644 flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_gpu_accelerator.go create mode 100644 flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_extended_resources.py create mode 100644 flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_gpu_accelerator.py create mode 100644 flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_extended_resources.py create mode 100644 flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_gpu_accelerator.py diff --git a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc index d9fe2fed164..e4e12e3543e 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/admin/agent.pb.cc @@ -23,7 +23,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fadmin_2fagent_2eproto ::google::proto extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fadmin_2fagent_2eproto ::google::protobuf::internal::SCCInfo<4> scc_info_TaskExecutionMetadata_flyteidl_2fadmin_2fagent_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fidentifier_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_TaskExecutionIdentifier_flyteidl_2fcore_2fidentifier_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_Literal_flyteidl_2fcore_2fliterals_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; namespace flyteidl { namespace admin { class TaskExecutionMetadata_LabelsEntry_DoNotUseDefaultTypeInternal { diff --git a/flyteidl/gen/pb-cpp/flyteidl/admin/task.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/admin/task.pb.cc index dfedf780d56..b63c75f458a 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/admin/task.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/admin/task.pb.cc @@ -22,7 +22,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fadmin_2ftask_2eproto ::google::protob extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fadmin_2ftask_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_Task_flyteidl_2fadmin_2ftask_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fcompiler_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_CompiledTask_flyteidl_2fcore_2fcompiler_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fidentifier_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_Identifier_flyteidl_2fcore_2fidentifier_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_google_2fprotobuf_2ftimestamp_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_Timestamp_google_2fprotobuf_2ftimestamp_2eproto; namespace flyteidl { namespace admin { diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/compiler.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/core/compiler.pb.cc index 1cdd0ed744f..1b397f1030c 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/compiler.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/core/compiler.pb.cc @@ -22,7 +22,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fcompiler_2eproto ::google::pro extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fcompiler_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_ConnectionSet_UpstreamEntry_DoNotUse_flyteidl_2fcore_2fcompiler_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fcompiler_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_CompiledWorkflow_flyteidl_2fcore_2fcompiler_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fcompiler_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_ConnectionSet_flyteidl_2fcore_2fcompiler_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<6> scc_info_WorkflowTemplate_flyteidl_2fcore_2fworkflow_2eproto; namespace flyteidl { namespace core { diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/dynamic_job.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/core/dynamic_job.pb.cc index b543813104b..40f059d1ba2 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/dynamic_job.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/core/dynamic_job.pb.cc @@ -17,7 +17,7 @@ #include extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_Binding_flyteidl_2fcore_2fliterals_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<6> scc_info_WorkflowTemplate_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<8> scc_info_ArrayNode_flyteidl_2fcore_2fworkflow_2eproto; namespace flyteidl { diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.cc index b983c4338a1..7a06a09340a 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.cc @@ -22,6 +22,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::pro extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_RetryStrategy_flyteidl_2fcore_2fliterals_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fsecurity_2eproto ::google::protobuf::internal::SCCInfo<3> scc_info_SecurityContext_flyteidl_2fcore_2fsecurity_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_ContainerPort_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_IOStrategy_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_K8sObjectMetadata_AnnotationsEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_K8sObjectMetadata_LabelsEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto; @@ -31,6 +32,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protob extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_TaskMetadata_TagsEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_TaskTemplate_ConfigEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_DataLoadingConfig_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_K8sObjectMetadata_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<3> scc_info_K8sPod_flyteidl_2fcore_2ftasks_2eproto; @@ -48,6 +50,16 @@ class ResourcesDefaultTypeInternal { public: ::google::protobuf::internal::ExplicitlyConstructed _instance; } _Resources_default_instance_; +class GPUAcceleratorDefaultTypeInternal { + public: + ::google::protobuf::internal::ExplicitlyConstructed _instance; + bool unpartitioned_; + ::google::protobuf::internal::ArenaStringPtr partition_size_; +} _GPUAccelerator_default_instance_; +class ExtendedResourcesDefaultTypeInternal { + public: + ::google::protobuf::internal::ExplicitlyConstructed _instance; +} _ExtendedResources_default_instance_; class RuntimeMetadataDefaultTypeInternal { public: ::google::protobuf::internal::ExplicitlyConstructed _instance; @@ -139,6 +151,35 @@ ::google::protobuf::internal::SCCInfo<1> scc_info_Resources_flyteidl_2fcore_2fta {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 1, InitDefaultsResources_flyteidl_2fcore_2ftasks_2eproto}, { &scc_info_Resources_ResourceEntry_flyteidl_2fcore_2ftasks_2eproto.base,}}; +static void InitDefaultsGPUAccelerator_flyteidl_2fcore_2ftasks_2eproto() { + GOOGLE_PROTOBUF_VERIFY_VERSION; + + { + void* ptr = &::flyteidl::core::_GPUAccelerator_default_instance_; + new (ptr) ::flyteidl::core::GPUAccelerator(); + ::google::protobuf::internal::OnShutdownDestroyMessage(ptr); + } + ::flyteidl::core::GPUAccelerator::InitAsDefaultInstance(); +} + +::google::protobuf::internal::SCCInfo<0> scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto = + {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 0, InitDefaultsGPUAccelerator_flyteidl_2fcore_2ftasks_2eproto}, {}}; + +static void InitDefaultsExtendedResources_flyteidl_2fcore_2ftasks_2eproto() { + GOOGLE_PROTOBUF_VERIFY_VERSION; + + { + void* ptr = &::flyteidl::core::_ExtendedResources_default_instance_; + new (ptr) ::flyteidl::core::ExtendedResources(); + ::google::protobuf::internal::OnShutdownDestroyMessage(ptr); + } + ::flyteidl::core::ExtendedResources::InitAsDefaultInstance(); +} + +::google::protobuf::internal::SCCInfo<1> scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto = + {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 1, InitDefaultsExtendedResources_flyteidl_2fcore_2ftasks_2eproto}, { + &scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto.base,}}; + static void InitDefaultsRuntimeMetadata_flyteidl_2fcore_2ftasks_2eproto() { GOOGLE_PROTOBUF_VERIFY_VERSION; @@ -208,8 +249,8 @@ static void InitDefaultsTaskTemplate_flyteidl_2fcore_2ftasks_2eproto() { ::flyteidl::core::TaskTemplate::InitAsDefaultInstance(); } -::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto = - {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 9, InitDefaultsTaskTemplate_flyteidl_2fcore_2ftasks_2eproto}, { +::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto = + {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 10, InitDefaultsTaskTemplate_flyteidl_2fcore_2ftasks_2eproto}, { &scc_info_Identifier_flyteidl_2fcore_2fidentifier_2eproto.base, &scc_info_TaskMetadata_flyteidl_2fcore_2ftasks_2eproto.base, &scc_info_TypedInterface_flyteidl_2fcore_2finterface_2eproto.base, @@ -218,6 +259,7 @@ ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2 &scc_info_K8sPod_flyteidl_2fcore_2ftasks_2eproto.base, &scc_info_Sql_flyteidl_2fcore_2ftasks_2eproto.base, &scc_info_SecurityContext_flyteidl_2fcore_2fsecurity_2eproto.base, + &scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto.base, &scc_info_TaskTemplate_ConfigEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto.base,}}; static void InitDefaultsContainerPort_flyteidl_2fcore_2ftasks_2eproto() { @@ -357,6 +399,8 @@ ::google::protobuf::internal::SCCInfo<0> scc_info_Sql_flyteidl_2fcore_2ftasks_2e void InitDefaults_flyteidl_2fcore_2ftasks_2eproto() { ::google::protobuf::internal::InitSCC(&scc_info_Resources_ResourceEntry_flyteidl_2fcore_2ftasks_2eproto.base); ::google::protobuf::internal::InitSCC(&scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base); + ::google::protobuf::internal::InitSCC(&scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto.base); + ::google::protobuf::internal::InitSCC(&scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto.base); ::google::protobuf::internal::InitSCC(&scc_info_RuntimeMetadata_flyteidl_2fcore_2ftasks_2eproto.base); ::google::protobuf::internal::InitSCC(&scc_info_TaskMetadata_TagsEntry_DoNotUse_flyteidl_2fcore_2ftasks_2eproto.base); ::google::protobuf::internal::InitSCC(&scc_info_TaskMetadata_flyteidl_2fcore_2ftasks_2eproto.base); @@ -373,7 +417,7 @@ void InitDefaults_flyteidl_2fcore_2ftasks_2eproto() { ::google::protobuf::internal::InitSCC(&scc_info_Sql_flyteidl_2fcore_2ftasks_2eproto.base); } -::google::protobuf::Metadata file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[16]; +::google::protobuf::Metadata file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[18]; const ::google::protobuf::EnumDescriptor* file_level_enum_descriptors_flyteidl_2fcore_2ftasks_2eproto[7]; constexpr ::google::protobuf::ServiceDescriptor const** file_level_service_descriptors_flyteidl_2fcore_2ftasks_2eproto = nullptr; @@ -393,6 +437,21 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fcore_2ftasks_2eproto::of PROTOBUF_FIELD_OFFSET(::flyteidl::core::Resources, requests_), PROTOBUF_FIELD_OFFSET(::flyteidl::core::Resources, limits_), ~0u, // no _has_bits_ + PROTOBUF_FIELD_OFFSET(::flyteidl::core::GPUAccelerator, _internal_metadata_), + ~0u, // no _extensions_ + PROTOBUF_FIELD_OFFSET(::flyteidl::core::GPUAccelerator, _oneof_case_[0]), + ~0u, // no _weak_field_map_ + PROTOBUF_FIELD_OFFSET(::flyteidl::core::GPUAccelerator, device_), + offsetof(::flyteidl::core::GPUAcceleratorDefaultTypeInternal, unpartitioned_), + offsetof(::flyteidl::core::GPUAcceleratorDefaultTypeInternal, partition_size_), + PROTOBUF_FIELD_OFFSET(::flyteidl::core::GPUAccelerator, partition_size_value_), + ~0u, // no _has_bits_ + PROTOBUF_FIELD_OFFSET(::flyteidl::core::ExtendedResources, _internal_metadata_), + ~0u, // no _extensions_ + ~0u, // no _oneof_case_ + ~0u, // no _weak_field_map_ + PROTOBUF_FIELD_OFFSET(::flyteidl::core::ExtendedResources, gpu_accelerator_), + ~0u, // no _has_bits_ PROTOBUF_FIELD_OFFSET(::flyteidl::core::RuntimeMetadata, _internal_metadata_), ~0u, // no _extensions_ ~0u, // no _oneof_case_ @@ -450,6 +509,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fcore_2ftasks_2eproto::of offsetof(::flyteidl::core::TaskTemplateDefaultTypeInternal, sql_), PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskTemplate, task_type_version_), PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskTemplate, security_context_), + PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskTemplate, extended_resources_), PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskTemplate, config_), PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskTemplate, target_), ~0u, // no _has_bits_ @@ -533,25 +593,29 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fcore_2ftasks_2eproto::of static const ::google::protobuf::internal::MigrationSchema schemas[] PROTOBUF_SECTION_VARIABLE(protodesc_cold) = { { 0, -1, sizeof(::flyteidl::core::Resources_ResourceEntry)}, { 7, -1, sizeof(::flyteidl::core::Resources)}, - { 14, -1, sizeof(::flyteidl::core::RuntimeMetadata)}, - { 22, 29, sizeof(::flyteidl::core::TaskMetadata_TagsEntry_DoNotUse)}, - { 31, -1, sizeof(::flyteidl::core::TaskMetadata)}, - { 48, 55, sizeof(::flyteidl::core::TaskTemplate_ConfigEntry_DoNotUse)}, - { 57, -1, sizeof(::flyteidl::core::TaskTemplate)}, - { 74, -1, sizeof(::flyteidl::core::ContainerPort)}, - { 80, -1, sizeof(::flyteidl::core::Container)}, - { 94, -1, sizeof(::flyteidl::core::IOStrategy)}, - { 101, -1, sizeof(::flyteidl::core::DataLoadingConfig)}, - { 111, -1, sizeof(::flyteidl::core::K8sPod)}, - { 119, 126, sizeof(::flyteidl::core::K8sObjectMetadata_LabelsEntry_DoNotUse)}, - { 128, 135, sizeof(::flyteidl::core::K8sObjectMetadata_AnnotationsEntry_DoNotUse)}, - { 137, -1, sizeof(::flyteidl::core::K8sObjectMetadata)}, - { 144, -1, sizeof(::flyteidl::core::Sql)}, + { 14, -1, sizeof(::flyteidl::core::GPUAccelerator)}, + { 23, -1, sizeof(::flyteidl::core::ExtendedResources)}, + { 29, -1, sizeof(::flyteidl::core::RuntimeMetadata)}, + { 37, 44, sizeof(::flyteidl::core::TaskMetadata_TagsEntry_DoNotUse)}, + { 46, -1, sizeof(::flyteidl::core::TaskMetadata)}, + { 63, 70, sizeof(::flyteidl::core::TaskTemplate_ConfigEntry_DoNotUse)}, + { 72, -1, sizeof(::flyteidl::core::TaskTemplate)}, + { 90, -1, sizeof(::flyteidl::core::ContainerPort)}, + { 96, -1, sizeof(::flyteidl::core::Container)}, + { 110, -1, sizeof(::flyteidl::core::IOStrategy)}, + { 117, -1, sizeof(::flyteidl::core::DataLoadingConfig)}, + { 127, -1, sizeof(::flyteidl::core::K8sPod)}, + { 135, 142, sizeof(::flyteidl::core::K8sObjectMetadata_LabelsEntry_DoNotUse)}, + { 144, 151, sizeof(::flyteidl::core::K8sObjectMetadata_AnnotationsEntry_DoNotUse)}, + { 153, -1, sizeof(::flyteidl::core::K8sObjectMetadata)}, + { 160, -1, sizeof(::flyteidl::core::Sql)}, }; static ::google::protobuf::Message const * const file_default_instances[] = { reinterpret_cast(&::flyteidl::core::_Resources_ResourceEntry_default_instance_), reinterpret_cast(&::flyteidl::core::_Resources_default_instance_), + reinterpret_cast(&::flyteidl::core::_GPUAccelerator_default_instance_), + reinterpret_cast(&::flyteidl::core::_ExtendedResources_default_instance_), reinterpret_cast(&::flyteidl::core::_RuntimeMetadata_default_instance_), reinterpret_cast(&::flyteidl::core::_TaskMetadata_TagsEntry_DoNotUse_default_instance_), reinterpret_cast(&::flyteidl::core::_TaskMetadata_default_instance_), @@ -571,7 +635,7 @@ static ::google::protobuf::Message const * const file_default_instances[] = { ::google::protobuf::internal::AssignDescriptorsTable assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto = { {}, AddDescriptors_flyteidl_2fcore_2ftasks_2eproto, "flyteidl/core/tasks.proto", schemas, file_default_instances, TableStruct_flyteidl_2fcore_2ftasks_2eproto::offsets, - file_level_metadata_flyteidl_2fcore_2ftasks_2eproto, 16, file_level_enum_descriptors_flyteidl_2fcore_2ftasks_2eproto, file_level_service_descriptors_flyteidl_2fcore_2ftasks_2eproto, + file_level_metadata_flyteidl_2fcore_2ftasks_2eproto, 18, file_level_enum_descriptors_flyteidl_2fcore_2ftasks_2eproto, file_level_service_descriptors_flyteidl_2fcore_2ftasks_2eproto, }; const char descriptor_table_protodef_flyteidl_2fcore_2ftasks_2eproto[] = @@ -588,81 +652,87 @@ const char descriptor_table_protodef_flyteidl_2fcore_2ftasks_2eproto[] = "re.Resources.ResourceName\022\r\n\005value\030\002 \001(\t" "\"]\n\014ResourceName\022\013\n\007UNKNOWN\020\000\022\007\n\003CPU\020\001\022\007" "\n\003GPU\020\002\022\n\n\006MEMORY\020\003\022\013\n\007STORAGE\020\004\022\025\n\021EPHE" - "MERAL_STORAGE\020\005\"\225\001\n\017RuntimeMetadata\0228\n\004t" - "ype\030\001 \001(\0162*.flyteidl.core.RuntimeMetadat" - "a.RuntimeType\022\017\n\007version\030\002 \001(\t\022\016\n\006flavor" - "\030\003 \001(\t\"\'\n\013RuntimeType\022\t\n\005OTHER\020\000\022\r\n\tFLYT" - "E_SDK\020\001\"\316\003\n\014TaskMetadata\022\024\n\014discoverable" - "\030\001 \001(\010\022/\n\007runtime\030\002 \001(\0132\036.flyteidl.core." - "RuntimeMetadata\022*\n\007timeout\030\004 \001(\0132\031.googl" - "e.protobuf.Duration\022-\n\007retries\030\005 \001(\0132\034.f" - "lyteidl.core.RetryStrategy\022\031\n\021discovery_" - "version\030\006 \001(\t\022 \n\030deprecated_error_messag" - "e\030\007 \001(\t\022\027\n\rinterruptible\030\010 \001(\010H\000\022\032\n\022cach" - "e_serializable\030\t \001(\010\022\026\n\016generates_deck\030\n" - " \001(\010\0223\n\004tags\030\013 \003(\0132%.flyteidl.core.TaskM" - "etadata.TagsEntry\022\031\n\021pod_template_name\030\014" - " \001(\t\032+\n\tTagsEntry\022\013\n\003key\030\001 \001(\t\022\r\n\005value\030" - "\002 \001(\t:\0028\001B\025\n\023interruptible_value\"\220\004\n\014Tas" - "kTemplate\022%\n\002id\030\001 \001(\0132\031.flyteidl.core.Id" - "entifier\022\014\n\004type\030\002 \001(\t\022-\n\010metadata\030\003 \001(\013" - "2\033.flyteidl.core.TaskMetadata\0220\n\tinterfa" - "ce\030\004 \001(\0132\035.flyteidl.core.TypedInterface\022" - "\'\n\006custom\030\005 \001(\0132\027.google.protobuf.Struct" - "\022-\n\tcontainer\030\006 \001(\0132\030.flyteidl.core.Cont" - "ainerH\000\022(\n\007k8s_pod\030\021 \001(\0132\025.flyteidl.core" - ".K8sPodH\000\022!\n\003sql\030\022 \001(\0132\022.flyteidl.core.S" - "qlH\000\022\031\n\021task_type_version\030\007 \001(\005\0228\n\020secur" - "ity_context\030\010 \001(\0132\036.flyteidl.core.Securi" - "tyContext\0227\n\006config\030\020 \003(\0132\'.flyteidl.cor" - "e.TaskTemplate.ConfigEntry\032-\n\013ConfigEntr" - "y\022\013\n\003key\030\001 \001(\t\022\r\n\005value\030\002 \001(\t:\0028\001B\010\n\006tar" - "get\"\'\n\rContainerPort\022\026\n\016container_port\030\001" - " \001(\r\"\255\003\n\tContainer\022\r\n\005image\030\001 \001(\t\022\017\n\007com" - "mand\030\002 \003(\t\022\014\n\004args\030\003 \003(\t\022+\n\tresources\030\004 " - "\001(\0132\030.flyteidl.core.Resources\022(\n\003env\030\005 \003" - "(\0132\033.flyteidl.core.KeyValuePair\022/\n\006confi" - "g\030\006 \003(\0132\033.flyteidl.core.KeyValuePairB\002\030\001" - "\022+\n\005ports\030\007 \003(\0132\034.flyteidl.core.Containe" - "rPort\0225\n\013data_config\030\t \001(\0132 .flyteidl.co" - "re.DataLoadingConfig\022;\n\014architecture\030\n \001" - "(\0162%.flyteidl.core.Container.Architectur" - "e\"I\n\014Architecture\022\013\n\007UNKNOWN\020\000\022\t\n\005AMD64\020" - "\001\022\t\n\005ARM64\020\002\022\n\n\006ARM_V6\020\003\022\n\n\006ARM_V7\020\004\"\233\002\n" - "\nIOStrategy\022=\n\rdownload_mode\030\001 \001(\0162&.fly" - "teidl.core.IOStrategy.DownloadMode\0229\n\013up" - "load_mode\030\002 \001(\0162$.flyteidl.core.IOStrate" - "gy.UploadMode\"L\n\014DownloadMode\022\022\n\016DOWNLOA" - "D_EAGER\020\000\022\023\n\017DOWNLOAD_STREAM\020\001\022\023\n\017DO_NOT" - "_DOWNLOAD\020\002\"E\n\nUploadMode\022\022\n\016UPLOAD_ON_E" - "XIT\020\000\022\020\n\014UPLOAD_EAGER\020\001\022\021\n\rDO_NOT_UPLOAD" - "\020\002\"\363\001\n\021DataLoadingConfig\022\017\n\007enabled\030\001 \001(" - "\010\022\022\n\ninput_path\030\002 \001(\t\022\023\n\013output_path\030\003 \001" - "(\t\022A\n\006format\030\004 \001(\01621.flyteidl.core.DataL" - "oadingConfig.LiteralMapFormat\022.\n\013io_stra" - "tegy\030\005 \001(\0132\031.flyteidl.core.IOStrategy\"1\n" - "\020LiteralMapFormat\022\010\n\004JSON\020\000\022\010\n\004YAML\020\001\022\t\n" - "\005PROTO\020\002\"\236\001\n\006K8sPod\0222\n\010metadata\030\001 \001(\0132 ." - "flyteidl.core.K8sObjectMetadata\022)\n\010pod_s" - "pec\030\002 \001(\0132\027.google.protobuf.Struct\0225\n\013da" - "ta_config\030\003 \001(\0132 .flyteidl.core.DataLoad" - "ingConfig\"\374\001\n\021K8sObjectMetadata\022<\n\006label" - "s\030\001 \003(\0132,.flyteidl.core.K8sObjectMetadat" - "a.LabelsEntry\022F\n\013annotations\030\002 \003(\01321.fly" - "teidl.core.K8sObjectMetadata.Annotations" - "Entry\032-\n\013LabelsEntry\022\013\n\003key\030\001 \001(\t\022\r\n\005val" - "ue\030\002 \001(\t:\0028\001\0322\n\020AnnotationsEntry\022\013\n\003key\030" - "\001 \001(\t\022\r\n\005value\030\002 \001(\t:\0028\001\"~\n\003Sql\022\021\n\tstate" - "ment\030\001 \001(\t\022+\n\007dialect\030\002 \001(\0162\032.flyteidl.c" - "ore.Sql.Dialect\"7\n\007Dialect\022\r\n\tUNDEFINED\020" - "\000\022\010\n\004ANSI\020\001\022\010\n\004HIVE\020\002\022\t\n\005OTHER\020\003Bname() != 0) { - target = ::google::protobuf::internal::WireFormatLite::WriteEnumToArray( - 1, this->name(), target); + // .flyteidl.core.Resources.ResourceName name = 1; + if (this->name() != 0) { + target = ::google::protobuf::internal::WireFormatLite::WriteEnumToArray( + 1, this->name(), target); + } + + // string value = 2; + if (this->value().size() > 0) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->value().data(), static_cast(this->value().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.core.Resources.ResourceEntry.value"); + target = + ::google::protobuf::internal::WireFormatLite::WriteStringToArray( + 2, this->value(), target); + } + + if (_internal_metadata_.have_unknown_fields()) { + target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( + _internal_metadata_.unknown_fields(), target); + } + // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.Resources.ResourceEntry) + return target; +} + +size_t Resources_ResourceEntry::ByteSizeLong() const { +// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.Resources.ResourceEntry) + size_t total_size = 0; + + if (_internal_metadata_.have_unknown_fields()) { + total_size += + ::google::protobuf::internal::WireFormat::ComputeUnknownFieldsSize( + _internal_metadata_.unknown_fields()); + } + ::google::protobuf::uint32 cached_has_bits = 0; + // Prevent compiler warnings about cached_has_bits being unused + (void) cached_has_bits; + + // string value = 2; + if (this->value().size() > 0) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::StringSize( + this->value()); + } + + // .flyteidl.core.Resources.ResourceName name = 1; + if (this->name() != 0) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::EnumSize(this->name()); + } + + int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); + SetCachedSize(cached_size); + return total_size; +} + +void Resources_ResourceEntry::MergeFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.Resources.ResourceEntry) + GOOGLE_DCHECK_NE(&from, this); + const Resources_ResourceEntry* source = + ::google::protobuf::DynamicCastToGenerated( + &from); + if (source == nullptr) { + // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.Resources.ResourceEntry) + ::google::protobuf::internal::ReflectionOps::Merge(from, this); + } else { + // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.Resources.ResourceEntry) + MergeFrom(*source); + } +} + +void Resources_ResourceEntry::MergeFrom(const Resources_ResourceEntry& from) { +// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.Resources.ResourceEntry) + GOOGLE_DCHECK_NE(&from, this); + _internal_metadata_.MergeFrom(from._internal_metadata_); + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + if (from.value().size() > 0) { + + value_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.value_); + } + if (from.name() != 0) { + set_name(from.name()); + } +} + +void Resources_ResourceEntry::CopyFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.Resources.ResourceEntry) + if (&from == this) return; + Clear(); + MergeFrom(from); +} + +void Resources_ResourceEntry::CopyFrom(const Resources_ResourceEntry& from) { +// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.Resources.ResourceEntry) + if (&from == this) return; + Clear(); + MergeFrom(from); +} + +bool Resources_ResourceEntry::IsInitialized() const { + return true; +} + +void Resources_ResourceEntry::Swap(Resources_ResourceEntry* other) { + if (other == this) return; + InternalSwap(other); +} +void Resources_ResourceEntry::InternalSwap(Resources_ResourceEntry* other) { + using std::swap; + _internal_metadata_.Swap(&other->_internal_metadata_); + value_.Swap(&other->value_, &::google::protobuf::internal::GetEmptyStringAlreadyInited(), + GetArenaNoVirtual()); + swap(name_, other->name_); +} + +::google::protobuf::Metadata Resources_ResourceEntry::GetMetadata() const { + ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[kIndexInFileMessages]; +} + + +// =================================================================== + +void Resources::InitAsDefaultInstance() { +} +class Resources::HasBitSetters { + public: +}; + +#if !defined(_MSC_VER) || _MSC_VER >= 1900 +const int Resources::kRequestsFieldNumber; +const int Resources::kLimitsFieldNumber; +#endif // !defined(_MSC_VER) || _MSC_VER >= 1900 + +Resources::Resources() + : ::google::protobuf::Message(), _internal_metadata_(nullptr) { + SharedCtor(); + // @@protoc_insertion_point(constructor:flyteidl.core.Resources) +} +Resources::Resources(const Resources& from) + : ::google::protobuf::Message(), + _internal_metadata_(nullptr), + requests_(from.requests_), + limits_(from.limits_) { + _internal_metadata_.MergeFrom(from._internal_metadata_); + // @@protoc_insertion_point(copy_constructor:flyteidl.core.Resources) +} + +void Resources::SharedCtor() { + ::google::protobuf::internal::InitSCC( + &scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base); +} + +Resources::~Resources() { + // @@protoc_insertion_point(destructor:flyteidl.core.Resources) + SharedDtor(); +} + +void Resources::SharedDtor() { +} + +void Resources::SetCachedSize(int size) const { + _cached_size_.Set(size); +} +const Resources& Resources::default_instance() { + ::google::protobuf::internal::InitSCC(&::scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base); + return *internal_default_instance(); +} + + +void Resources::Clear() { +// @@protoc_insertion_point(message_clear_start:flyteidl.core.Resources) + ::google::protobuf::uint32 cached_has_bits = 0; + // Prevent compiler warnings about cached_has_bits being unused + (void) cached_has_bits; + + requests_.Clear(); + limits_.Clear(); + _internal_metadata_.Clear(); +} + +#if GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER +const char* Resources::_InternalParse(const char* begin, const char* end, void* object, + ::google::protobuf::internal::ParseContext* ctx) { + auto msg = static_cast(object); + ::google::protobuf::int32 size; (void)size; + int depth; (void)depth; + ::google::protobuf::uint32 tag; + ::google::protobuf::internal::ParseFunc parser_till_end; (void)parser_till_end; + auto ptr = begin; + while (ptr < end) { + ptr = ::google::protobuf::io::Parse32(ptr, &tag); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + switch (tag >> 3) { + // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + case 1: { + if (static_cast<::google::protobuf::uint8>(tag) != 10) goto handle_unusual; + do { + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::core::Resources_ResourceEntry::_InternalParse; + object = msg->add_requests(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); + if (ptr >= end) break; + } while ((::google::protobuf::io::UnalignedLoad<::google::protobuf::uint64>(ptr) & 255) == 10 && (ptr += 1)); + break; + } + // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; + case 2: { + if (static_cast<::google::protobuf::uint8>(tag) != 18) goto handle_unusual; + do { + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::core::Resources_ResourceEntry::_InternalParse; + object = msg->add_limits(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); + if (ptr >= end) break; + } while ((::google::protobuf::io::UnalignedLoad<::google::protobuf::uint64>(ptr) & 255) == 18 && (ptr += 1)); + break; + } + default: { + handle_unusual: + if ((tag & 7) == 4 || tag == 0) { + ctx->EndGroup(tag); + return ptr; + } + auto res = UnknownFieldParse(tag, {_InternalParse, msg}, + ptr, end, msg->_internal_metadata_.mutable_unknown_fields(), ctx); + ptr = res.first; + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr != nullptr); + if (res.second) return ptr; + } + } // switch + } // while + return ptr; +len_delim_till_end: + return ctx->StoreAndTailCall(ptr, end, {_InternalParse, msg}, + {parser_till_end, object}, size); +} +#else // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER +bool Resources::MergePartialFromCodedStream( + ::google::protobuf::io::CodedInputStream* input) { +#define DO_(EXPRESSION) if (!PROTOBUF_PREDICT_TRUE(EXPRESSION)) goto failure + ::google::protobuf::uint32 tag; + // @@protoc_insertion_point(parse_start:flyteidl.core.Resources) + for (;;) { + ::std::pair<::google::protobuf::uint32, bool> p = input->ReadTagWithCutoffNoLastTag(127u); + tag = p.first; + if (!p.second) goto handle_unusual; + switch (::google::protobuf::internal::WireFormatLite::GetTagFieldNumber(tag)) { + // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + case 1: { + if (static_cast< ::google::protobuf::uint8>(tag) == (10 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( + input, add_requests())); + } else { + goto handle_unusual; + } + break; + } + + // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; + case 2: { + if (static_cast< ::google::protobuf::uint8>(tag) == (18 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( + input, add_limits())); + } else { + goto handle_unusual; + } + break; + } + + default: { + handle_unusual: + if (tag == 0) { + goto success; + } + DO_(::google::protobuf::internal::WireFormat::SkipField( + input, tag, _internal_metadata_.mutable_unknown_fields())); + break; + } + } + } +success: + // @@protoc_insertion_point(parse_success:flyteidl.core.Resources) + return true; +failure: + // @@protoc_insertion_point(parse_failure:flyteidl.core.Resources) + return false; +#undef DO_ +} +#endif // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + +void Resources::SerializeWithCachedSizes( + ::google::protobuf::io::CodedOutputStream* output) const { + // @@protoc_insertion_point(serialize_start:flyteidl.core.Resources) + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + for (unsigned int i = 0, + n = static_cast(this->requests_size()); i < n; i++) { + ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( + 1, + this->requests(static_cast(i)), + output); + } + + // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; + for (unsigned int i = 0, + n = static_cast(this->limits_size()); i < n; i++) { + ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( + 2, + this->limits(static_cast(i)), + output); + } + + if (_internal_metadata_.have_unknown_fields()) { + ::google::protobuf::internal::WireFormat::SerializeUnknownFields( + _internal_metadata_.unknown_fields(), output); + } + // @@protoc_insertion_point(serialize_end:flyteidl.core.Resources) +} + +::google::protobuf::uint8* Resources::InternalSerializeWithCachedSizesToArray( + ::google::protobuf::uint8* target) const { + // @@protoc_insertion_point(serialize_to_array_start:flyteidl.core.Resources) + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + for (unsigned int i = 0, + n = static_cast(this->requests_size()); i < n; i++) { + target = ::google::protobuf::internal::WireFormatLite:: + InternalWriteMessageToArray( + 1, this->requests(static_cast(i)), target); + } + + // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; + for (unsigned int i = 0, + n = static_cast(this->limits_size()); i < n; i++) { + target = ::google::protobuf::internal::WireFormatLite:: + InternalWriteMessageToArray( + 2, this->limits(static_cast(i)), target); + } + + if (_internal_metadata_.have_unknown_fields()) { + target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( + _internal_metadata_.unknown_fields(), target); + } + // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.Resources) + return target; +} + +size_t Resources::ByteSizeLong() const { +// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.Resources) + size_t total_size = 0; + + if (_internal_metadata_.have_unknown_fields()) { + total_size += + ::google::protobuf::internal::WireFormat::ComputeUnknownFieldsSize( + _internal_metadata_.unknown_fields()); + } + ::google::protobuf::uint32 cached_has_bits = 0; + // Prevent compiler warnings about cached_has_bits being unused + (void) cached_has_bits; + + // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + { + unsigned int count = static_cast(this->requests_size()); + total_size += 1UL * count; + for (unsigned int i = 0; i < count; i++) { + total_size += + ::google::protobuf::internal::WireFormatLite::MessageSize( + this->requests(static_cast(i))); + } + } + + // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; + { + unsigned int count = static_cast(this->limits_size()); + total_size += 1UL * count; + for (unsigned int i = 0; i < count; i++) { + total_size += + ::google::protobuf::internal::WireFormatLite::MessageSize( + this->limits(static_cast(i))); + } + } + + int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); + SetCachedSize(cached_size); + return total_size; +} + +void Resources::MergeFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.Resources) + GOOGLE_DCHECK_NE(&from, this); + const Resources* source = + ::google::protobuf::DynamicCastToGenerated( + &from); + if (source == nullptr) { + // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.Resources) + ::google::protobuf::internal::ReflectionOps::Merge(from, this); + } else { + // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.Resources) + MergeFrom(*source); + } +} + +void Resources::MergeFrom(const Resources& from) { +// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.Resources) + GOOGLE_DCHECK_NE(&from, this); + _internal_metadata_.MergeFrom(from._internal_metadata_); + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + requests_.MergeFrom(from.requests_); + limits_.MergeFrom(from.limits_); +} + +void Resources::CopyFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.Resources) + if (&from == this) return; + Clear(); + MergeFrom(from); +} + +void Resources::CopyFrom(const Resources& from) { +// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.Resources) + if (&from == this) return; + Clear(); + MergeFrom(from); +} + +bool Resources::IsInitialized() const { + return true; +} + +void Resources::Swap(Resources* other) { + if (other == this) return; + InternalSwap(other); +} +void Resources::InternalSwap(Resources* other) { + using std::swap; + _internal_metadata_.Swap(&other->_internal_metadata_); + CastToBase(&requests_)->InternalSwap(CastToBase(&other->requests_)); + CastToBase(&limits_)->InternalSwap(CastToBase(&other->limits_)); +} + +::google::protobuf::Metadata Resources::GetMetadata() const { + ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[kIndexInFileMessages]; +} + + +// =================================================================== + +void GPUAccelerator::InitAsDefaultInstance() { + ::flyteidl::core::_GPUAccelerator_default_instance_.unpartitioned_ = false; + ::flyteidl::core::_GPUAccelerator_default_instance_.partition_size_.UnsafeSetDefault( + &::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +class GPUAccelerator::HasBitSetters { + public: +}; + +#if !defined(_MSC_VER) || _MSC_VER >= 1900 +const int GPUAccelerator::kDeviceFieldNumber; +const int GPUAccelerator::kUnpartitionedFieldNumber; +const int GPUAccelerator::kPartitionSizeFieldNumber; +#endif // !defined(_MSC_VER) || _MSC_VER >= 1900 + +GPUAccelerator::GPUAccelerator() + : ::google::protobuf::Message(), _internal_metadata_(nullptr) { + SharedCtor(); + // @@protoc_insertion_point(constructor:flyteidl.core.GPUAccelerator) +} +GPUAccelerator::GPUAccelerator(const GPUAccelerator& from) + : ::google::protobuf::Message(), + _internal_metadata_(nullptr) { + _internal_metadata_.MergeFrom(from._internal_metadata_); + device_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + if (from.device().size() > 0) { + device_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.device_); + } + clear_has_partition_size_value(); + switch (from.partition_size_value_case()) { + case kUnpartitioned: { + set_unpartitioned(from.unpartitioned()); + break; + } + case kPartitionSize: { + set_partition_size(from.partition_size()); + break; + } + case PARTITION_SIZE_VALUE_NOT_SET: { + break; + } + } + // @@protoc_insertion_point(copy_constructor:flyteidl.core.GPUAccelerator) +} + +void GPUAccelerator::SharedCtor() { + ::google::protobuf::internal::InitSCC( + &scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto.base); + device_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + clear_has_partition_size_value(); +} + +GPUAccelerator::~GPUAccelerator() { + // @@protoc_insertion_point(destructor:flyteidl.core.GPUAccelerator) + SharedDtor(); +} + +void GPUAccelerator::SharedDtor() { + device_.DestroyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + if (has_partition_size_value()) { + clear_partition_size_value(); + } +} + +void GPUAccelerator::SetCachedSize(int size) const { + _cached_size_.Set(size); +} +const GPUAccelerator& GPUAccelerator::default_instance() { + ::google::protobuf::internal::InitSCC(&::scc_info_GPUAccelerator_flyteidl_2fcore_2ftasks_2eproto.base); + return *internal_default_instance(); +} + + +void GPUAccelerator::clear_partition_size_value() { +// @@protoc_insertion_point(one_of_clear_start:flyteidl.core.GPUAccelerator) + switch (partition_size_value_case()) { + case kUnpartitioned: { + // No need to clear + break; + } + case kPartitionSize: { + partition_size_value_.partition_size_.DestroyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + break; + } + case PARTITION_SIZE_VALUE_NOT_SET: { + break; + } + } + _oneof_case_[0] = PARTITION_SIZE_VALUE_NOT_SET; +} + + +void GPUAccelerator::Clear() { +// @@protoc_insertion_point(message_clear_start:flyteidl.core.GPUAccelerator) + ::google::protobuf::uint32 cached_has_bits = 0; + // Prevent compiler warnings about cached_has_bits being unused + (void) cached_has_bits; + + device_.ClearToEmptyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + clear_partition_size_value(); + _internal_metadata_.Clear(); +} + +#if GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER +const char* GPUAccelerator::_InternalParse(const char* begin, const char* end, void* object, + ::google::protobuf::internal::ParseContext* ctx) { + auto msg = static_cast(object); + ::google::protobuf::int32 size; (void)size; + int depth; (void)depth; + ::google::protobuf::uint32 tag; + ::google::protobuf::internal::ParseFunc parser_till_end; (void)parser_till_end; + auto ptr = begin; + while (ptr < end) { + ptr = ::google::protobuf::io::Parse32(ptr, &tag); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + switch (tag >> 3) { + // string device = 1; + case 1: { + if (static_cast<::google::protobuf::uint8>(tag) != 10) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + ctx->extra_parse_data().SetFieldName("flyteidl.core.GPUAccelerator.device"); + object = msg->mutable_device(); + if (size > end - ptr + ::google::protobuf::internal::ParseContext::kSlopBytes) { + parser_till_end = ::google::protobuf::internal::GreedyStringParserUTF8; + goto string_till_end; + } + GOOGLE_PROTOBUF_PARSER_ASSERT(::google::protobuf::internal::StringCheckUTF8(ptr, size, ctx)); + ::google::protobuf::internal::InlineGreedyStringParser(object, ptr, size, ctx); + ptr += size; + break; + } + // bool unpartitioned = 2; + case 2: { + if (static_cast<::google::protobuf::uint8>(tag) != 16) goto handle_unusual; + msg->set_unpartitioned(::google::protobuf::internal::ReadVarint(&ptr)); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + break; + } + // string partition_size = 3; + case 3: { + if (static_cast<::google::protobuf::uint8>(tag) != 26) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + ctx->extra_parse_data().SetFieldName("flyteidl.core.GPUAccelerator.partition_size"); + object = msg->mutable_partition_size(); + if (size > end - ptr + ::google::protobuf::internal::ParseContext::kSlopBytes) { + parser_till_end = ::google::protobuf::internal::GreedyStringParserUTF8; + goto string_till_end; + } + GOOGLE_PROTOBUF_PARSER_ASSERT(::google::protobuf::internal::StringCheckUTF8(ptr, size, ctx)); + ::google::protobuf::internal::InlineGreedyStringParser(object, ptr, size, ctx); + ptr += size; + break; + } + default: { + handle_unusual: + if ((tag & 7) == 4 || tag == 0) { + ctx->EndGroup(tag); + return ptr; + } + auto res = UnknownFieldParse(tag, {_InternalParse, msg}, + ptr, end, msg->_internal_metadata_.mutable_unknown_fields(), ctx); + ptr = res.first; + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr != nullptr); + if (res.second) return ptr; + } + } // switch + } // while + return ptr; +string_till_end: + static_cast<::std::string*>(object)->clear(); + static_cast<::std::string*>(object)->reserve(size); + goto len_delim_till_end; +len_delim_till_end: + return ctx->StoreAndTailCall(ptr, end, {_InternalParse, msg}, + {parser_till_end, object}, size); +} +#else // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER +bool GPUAccelerator::MergePartialFromCodedStream( + ::google::protobuf::io::CodedInputStream* input) { +#define DO_(EXPRESSION) if (!PROTOBUF_PREDICT_TRUE(EXPRESSION)) goto failure + ::google::protobuf::uint32 tag; + // @@protoc_insertion_point(parse_start:flyteidl.core.GPUAccelerator) + for (;;) { + ::std::pair<::google::protobuf::uint32, bool> p = input->ReadTagWithCutoffNoLastTag(127u); + tag = p.first; + if (!p.second) goto handle_unusual; + switch (::google::protobuf::internal::WireFormatLite::GetTagFieldNumber(tag)) { + // string device = 1; + case 1: { + if (static_cast< ::google::protobuf::uint8>(tag) == (10 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadString( + input, this->mutable_device())); + DO_(::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->device().data(), static_cast(this->device().length()), + ::google::protobuf::internal::WireFormatLite::PARSE, + "flyteidl.core.GPUAccelerator.device")); + } else { + goto handle_unusual; + } + break; + } + + // bool unpartitioned = 2; + case 2: { + if (static_cast< ::google::protobuf::uint8>(tag) == (16 & 0xFF)) { + clear_partition_size_value(); + DO_((::google::protobuf::internal::WireFormatLite::ReadPrimitive< + bool, ::google::protobuf::internal::WireFormatLite::TYPE_BOOL>( + input, &partition_size_value_.unpartitioned_))); + set_has_unpartitioned(); + } else { + goto handle_unusual; + } + break; + } + + // string partition_size = 3; + case 3: { + if (static_cast< ::google::protobuf::uint8>(tag) == (26 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadString( + input, this->mutable_partition_size())); + DO_(::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->partition_size().data(), static_cast(this->partition_size().length()), + ::google::protobuf::internal::WireFormatLite::PARSE, + "flyteidl.core.GPUAccelerator.partition_size")); + } else { + goto handle_unusual; + } + break; + } + + default: { + handle_unusual: + if (tag == 0) { + goto success; + } + DO_(::google::protobuf::internal::WireFormat::SkipField( + input, tag, _internal_metadata_.mutable_unknown_fields())); + break; + } + } + } +success: + // @@protoc_insertion_point(parse_success:flyteidl.core.GPUAccelerator) + return true; +failure: + // @@protoc_insertion_point(parse_failure:flyteidl.core.GPUAccelerator) + return false; +#undef DO_ +} +#endif // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + +void GPUAccelerator::SerializeWithCachedSizes( + ::google::protobuf::io::CodedOutputStream* output) const { + // @@protoc_insertion_point(serialize_start:flyteidl.core.GPUAccelerator) + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + // string device = 1; + if (this->device().size() > 0) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->device().data(), static_cast(this->device().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.core.GPUAccelerator.device"); + ::google::protobuf::internal::WireFormatLite::WriteStringMaybeAliased( + 1, this->device(), output); + } + + // bool unpartitioned = 2; + if (has_unpartitioned()) { + ::google::protobuf::internal::WireFormatLite::WriteBool(2, this->unpartitioned(), output); + } + + // string partition_size = 3; + if (has_partition_size()) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->partition_size().data(), static_cast(this->partition_size().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.core.GPUAccelerator.partition_size"); + ::google::protobuf::internal::WireFormatLite::WriteStringMaybeAliased( + 3, this->partition_size(), output); + } + + if (_internal_metadata_.have_unknown_fields()) { + ::google::protobuf::internal::WireFormat::SerializeUnknownFields( + _internal_metadata_.unknown_fields(), output); + } + // @@protoc_insertion_point(serialize_end:flyteidl.core.GPUAccelerator) +} + +::google::protobuf::uint8* GPUAccelerator::InternalSerializeWithCachedSizesToArray( + ::google::protobuf::uint8* target) const { + // @@protoc_insertion_point(serialize_to_array_start:flyteidl.core.GPUAccelerator) + ::google::protobuf::uint32 cached_has_bits = 0; + (void) cached_has_bits; + + // string device = 1; + if (this->device().size() > 0) { + ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( + this->device().data(), static_cast(this->device().length()), + ::google::protobuf::internal::WireFormatLite::SERIALIZE, + "flyteidl.core.GPUAccelerator.device"); + target = + ::google::protobuf::internal::WireFormatLite::WriteStringToArray( + 1, this->device(), target); + } + + // bool unpartitioned = 2; + if (has_unpartitioned()) { + target = ::google::protobuf::internal::WireFormatLite::WriteBoolToArray(2, this->unpartitioned(), target); } - // string value = 2; - if (this->value().size() > 0) { + // string partition_size = 3; + if (has_partition_size()) { ::google::protobuf::internal::WireFormatLite::VerifyUtf8String( - this->value().data(), static_cast(this->value().length()), + this->partition_size().data(), static_cast(this->partition_size().length()), ::google::protobuf::internal::WireFormatLite::SERIALIZE, - "flyteidl.core.Resources.ResourceEntry.value"); + "flyteidl.core.GPUAccelerator.partition_size"); target = ::google::protobuf::internal::WireFormatLite::WriteStringToArray( - 2, this->value(), target); + 3, this->partition_size(), target); } if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); } - // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.Resources.ResourceEntry) + // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.GPUAccelerator) return target; } -size_t Resources_ResourceEntry::ByteSizeLong() const { -// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.Resources.ResourceEntry) +size_t GPUAccelerator::ByteSizeLong() const { +// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.GPUAccelerator) size_t total_size = 0; if (_internal_metadata_.have_unknown_fields()) { @@ -1114,86 +1957,108 @@ size_t Resources_ResourceEntry::ByteSizeLong() const { // Prevent compiler warnings about cached_has_bits being unused (void) cached_has_bits; - // string value = 2; - if (this->value().size() > 0) { + // string device = 1; + if (this->device().size() > 0) { total_size += 1 + ::google::protobuf::internal::WireFormatLite::StringSize( - this->value()); + this->device()); } - // .flyteidl.core.Resources.ResourceName name = 1; - if (this->name() != 0) { - total_size += 1 + - ::google::protobuf::internal::WireFormatLite::EnumSize(this->name()); + switch (partition_size_value_case()) { + // bool unpartitioned = 2; + case kUnpartitioned: { + total_size += 1 + 1; + break; + } + // string partition_size = 3; + case kPartitionSize: { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::StringSize( + this->partition_size()); + break; + } + case PARTITION_SIZE_VALUE_NOT_SET: { + break; + } } - int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); SetCachedSize(cached_size); return total_size; } -void Resources_ResourceEntry::MergeFrom(const ::google::protobuf::Message& from) { -// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.Resources.ResourceEntry) +void GPUAccelerator::MergeFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.GPUAccelerator) GOOGLE_DCHECK_NE(&from, this); - const Resources_ResourceEntry* source = - ::google::protobuf::DynamicCastToGenerated( + const GPUAccelerator* source = + ::google::protobuf::DynamicCastToGenerated( &from); if (source == nullptr) { - // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.Resources.ResourceEntry) + // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.GPUAccelerator) ::google::protobuf::internal::ReflectionOps::Merge(from, this); } else { - // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.Resources.ResourceEntry) + // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.GPUAccelerator) MergeFrom(*source); } } -void Resources_ResourceEntry::MergeFrom(const Resources_ResourceEntry& from) { -// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.Resources.ResourceEntry) +void GPUAccelerator::MergeFrom(const GPUAccelerator& from) { +// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.GPUAccelerator) GOOGLE_DCHECK_NE(&from, this); _internal_metadata_.MergeFrom(from._internal_metadata_); ::google::protobuf::uint32 cached_has_bits = 0; (void) cached_has_bits; - if (from.value().size() > 0) { + if (from.device().size() > 0) { - value_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.value_); + device_.AssignWithDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), from.device_); } - if (from.name() != 0) { - set_name(from.name()); + switch (from.partition_size_value_case()) { + case kUnpartitioned: { + set_unpartitioned(from.unpartitioned()); + break; + } + case kPartitionSize: { + set_partition_size(from.partition_size()); + break; + } + case PARTITION_SIZE_VALUE_NOT_SET: { + break; + } } } -void Resources_ResourceEntry::CopyFrom(const ::google::protobuf::Message& from) { -// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.Resources.ResourceEntry) +void GPUAccelerator::CopyFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.GPUAccelerator) if (&from == this) return; Clear(); MergeFrom(from); } -void Resources_ResourceEntry::CopyFrom(const Resources_ResourceEntry& from) { -// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.Resources.ResourceEntry) +void GPUAccelerator::CopyFrom(const GPUAccelerator& from) { +// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.GPUAccelerator) if (&from == this) return; Clear(); MergeFrom(from); } -bool Resources_ResourceEntry::IsInitialized() const { +bool GPUAccelerator::IsInitialized() const { return true; } -void Resources_ResourceEntry::Swap(Resources_ResourceEntry* other) { +void GPUAccelerator::Swap(GPUAccelerator* other) { if (other == this) return; InternalSwap(other); } -void Resources_ResourceEntry::InternalSwap(Resources_ResourceEntry* other) { +void GPUAccelerator::InternalSwap(GPUAccelerator* other) { using std::swap; _internal_metadata_.Swap(&other->_internal_metadata_); - value_.Swap(&other->value_, &::google::protobuf::internal::GetEmptyStringAlreadyInited(), + device_.Swap(&other->device_, &::google::protobuf::internal::GetEmptyStringAlreadyInited(), GetArenaNoVirtual()); - swap(name_, other->name_); + swap(partition_size_value_, other->partition_size_value_); + swap(_oneof_case_[0], other->_oneof_case_[0]); } -::google::protobuf::Metadata Resources_ResourceEntry::GetMetadata() const { +::google::protobuf::Metadata GPUAccelerator::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[kIndexInFileMessages]; } @@ -1201,68 +2066,81 @@ ::google::protobuf::Metadata Resources_ResourceEntry::GetMetadata() const { // =================================================================== -void Resources::InitAsDefaultInstance() { +void ExtendedResources::InitAsDefaultInstance() { + ::flyteidl::core::_ExtendedResources_default_instance_._instance.get_mutable()->gpu_accelerator_ = const_cast< ::flyteidl::core::GPUAccelerator*>( + ::flyteidl::core::GPUAccelerator::internal_default_instance()); } -class Resources::HasBitSetters { +class ExtendedResources::HasBitSetters { public: + static const ::flyteidl::core::GPUAccelerator& gpu_accelerator(const ExtendedResources* msg); }; +const ::flyteidl::core::GPUAccelerator& +ExtendedResources::HasBitSetters::gpu_accelerator(const ExtendedResources* msg) { + return *msg->gpu_accelerator_; +} #if !defined(_MSC_VER) || _MSC_VER >= 1900 -const int Resources::kRequestsFieldNumber; -const int Resources::kLimitsFieldNumber; +const int ExtendedResources::kGpuAcceleratorFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 -Resources::Resources() +ExtendedResources::ExtendedResources() : ::google::protobuf::Message(), _internal_metadata_(nullptr) { SharedCtor(); - // @@protoc_insertion_point(constructor:flyteidl.core.Resources) + // @@protoc_insertion_point(constructor:flyteidl.core.ExtendedResources) } -Resources::Resources(const Resources& from) +ExtendedResources::ExtendedResources(const ExtendedResources& from) : ::google::protobuf::Message(), - _internal_metadata_(nullptr), - requests_(from.requests_), - limits_(from.limits_) { + _internal_metadata_(nullptr) { _internal_metadata_.MergeFrom(from._internal_metadata_); - // @@protoc_insertion_point(copy_constructor:flyteidl.core.Resources) + if (from.has_gpu_accelerator()) { + gpu_accelerator_ = new ::flyteidl::core::GPUAccelerator(*from.gpu_accelerator_); + } else { + gpu_accelerator_ = nullptr; + } + // @@protoc_insertion_point(copy_constructor:flyteidl.core.ExtendedResources) } -void Resources::SharedCtor() { +void ExtendedResources::SharedCtor() { ::google::protobuf::internal::InitSCC( - &scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base); + &scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto.base); + gpu_accelerator_ = nullptr; } -Resources::~Resources() { - // @@protoc_insertion_point(destructor:flyteidl.core.Resources) +ExtendedResources::~ExtendedResources() { + // @@protoc_insertion_point(destructor:flyteidl.core.ExtendedResources) SharedDtor(); } -void Resources::SharedDtor() { +void ExtendedResources::SharedDtor() { + if (this != internal_default_instance()) delete gpu_accelerator_; } -void Resources::SetCachedSize(int size) const { +void ExtendedResources::SetCachedSize(int size) const { _cached_size_.Set(size); } -const Resources& Resources::default_instance() { - ::google::protobuf::internal::InitSCC(&::scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base); +const ExtendedResources& ExtendedResources::default_instance() { + ::google::protobuf::internal::InitSCC(&::scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto.base); return *internal_default_instance(); } -void Resources::Clear() { -// @@protoc_insertion_point(message_clear_start:flyteidl.core.Resources) +void ExtendedResources::Clear() { +// @@protoc_insertion_point(message_clear_start:flyteidl.core.ExtendedResources) ::google::protobuf::uint32 cached_has_bits = 0; // Prevent compiler warnings about cached_has_bits being unused (void) cached_has_bits; - requests_.Clear(); - limits_.Clear(); + if (GetArenaNoVirtual() == nullptr && gpu_accelerator_ != nullptr) { + delete gpu_accelerator_; + } + gpu_accelerator_ = nullptr; _internal_metadata_.Clear(); } #if GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER -const char* Resources::_InternalParse(const char* begin, const char* end, void* object, +const char* ExtendedResources::_InternalParse(const char* begin, const char* end, void* object, ::google::protobuf::internal::ParseContext* ctx) { - auto msg = static_cast(object); + auto msg = static_cast(object); ::google::protobuf::int32 size; (void)size; int depth; (void)depth; ::google::protobuf::uint32 tag; @@ -1272,36 +2150,17 @@ const char* Resources::_InternalParse(const char* begin, const char* end, void* ptr = ::google::protobuf::io::Parse32(ptr, &tag); GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); switch (tag >> 3) { - // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; case 1: { if (static_cast<::google::protobuf::uint8>(tag) != 10) goto handle_unusual; - do { - ptr = ::google::protobuf::io::ReadSize(ptr, &size); - GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); - parser_till_end = ::flyteidl::core::Resources_ResourceEntry::_InternalParse; - object = msg->add_requests(); - if (size > end - ptr) goto len_delim_till_end; - ptr += size; - GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( - {parser_till_end, object}, ptr - size, ptr)); - if (ptr >= end) break; - } while ((::google::protobuf::io::UnalignedLoad<::google::protobuf::uint64>(ptr) & 255) == 10 && (ptr += 1)); - break; - } - // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; - case 2: { - if (static_cast<::google::protobuf::uint8>(tag) != 18) goto handle_unusual; - do { - ptr = ::google::protobuf::io::ReadSize(ptr, &size); - GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); - parser_till_end = ::flyteidl::core::Resources_ResourceEntry::_InternalParse; - object = msg->add_limits(); - if (size > end - ptr) goto len_delim_till_end; - ptr += size; - GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( - {parser_till_end, object}, ptr - size, ptr)); - if (ptr >= end) break; - } while ((::google::protobuf::io::UnalignedLoad<::google::protobuf::uint64>(ptr) & 255) == 18 && (ptr += 1)); + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::core::GPUAccelerator::_InternalParse; + object = msg->mutable_gpu_accelerator(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); break; } default: { @@ -1324,32 +2183,21 @@ const char* Resources::_InternalParse(const char* begin, const char* end, void* {parser_till_end, object}, size); } #else // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER -bool Resources::MergePartialFromCodedStream( +bool ExtendedResources::MergePartialFromCodedStream( ::google::protobuf::io::CodedInputStream* input) { #define DO_(EXPRESSION) if (!PROTOBUF_PREDICT_TRUE(EXPRESSION)) goto failure ::google::protobuf::uint32 tag; - // @@protoc_insertion_point(parse_start:flyteidl.core.Resources) + // @@protoc_insertion_point(parse_start:flyteidl.core.ExtendedResources) for (;;) { ::std::pair<::google::protobuf::uint32, bool> p = input->ReadTagWithCutoffNoLastTag(127u); tag = p.first; if (!p.second) goto handle_unusual; switch (::google::protobuf::internal::WireFormatLite::GetTagFieldNumber(tag)) { - // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; case 1: { if (static_cast< ::google::protobuf::uint8>(tag) == (10 & 0xFF)) { DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( - input, add_requests())); - } else { - goto handle_unusual; - } - break; - } - - // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; - case 2: { - if (static_cast< ::google::protobuf::uint8>(tag) == (18 & 0xFF)) { - DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( - input, add_limits())); + input, mutable_gpu_accelerator())); } else { goto handle_unusual; } @@ -1368,78 +2216,57 @@ bool Resources::MergePartialFromCodedStream( } } success: - // @@protoc_insertion_point(parse_success:flyteidl.core.Resources) + // @@protoc_insertion_point(parse_success:flyteidl.core.ExtendedResources) return true; failure: - // @@protoc_insertion_point(parse_failure:flyteidl.core.Resources) + // @@protoc_insertion_point(parse_failure:flyteidl.core.ExtendedResources) return false; #undef DO_ } #endif // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER -void Resources::SerializeWithCachedSizes( +void ExtendedResources::SerializeWithCachedSizes( ::google::protobuf::io::CodedOutputStream* output) const { - // @@protoc_insertion_point(serialize_start:flyteidl.core.Resources) + // @@protoc_insertion_point(serialize_start:flyteidl.core.ExtendedResources) ::google::protobuf::uint32 cached_has_bits = 0; (void) cached_has_bits; - // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; - for (unsigned int i = 0, - n = static_cast(this->requests_size()); i < n; i++) { - ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( - 1, - this->requests(static_cast(i)), - output); - } - - // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; - for (unsigned int i = 0, - n = static_cast(this->limits_size()); i < n; i++) { + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + if (this->has_gpu_accelerator()) { ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( - 2, - this->limits(static_cast(i)), - output); + 1, HasBitSetters::gpu_accelerator(this), output); } if (_internal_metadata_.have_unknown_fields()) { ::google::protobuf::internal::WireFormat::SerializeUnknownFields( _internal_metadata_.unknown_fields(), output); } - // @@protoc_insertion_point(serialize_end:flyteidl.core.Resources) + // @@protoc_insertion_point(serialize_end:flyteidl.core.ExtendedResources) } -::google::protobuf::uint8* Resources::InternalSerializeWithCachedSizesToArray( +::google::protobuf::uint8* ExtendedResources::InternalSerializeWithCachedSizesToArray( ::google::protobuf::uint8* target) const { - // @@protoc_insertion_point(serialize_to_array_start:flyteidl.core.Resources) + // @@protoc_insertion_point(serialize_to_array_start:flyteidl.core.ExtendedResources) ::google::protobuf::uint32 cached_has_bits = 0; (void) cached_has_bits; - // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; - for (unsigned int i = 0, - n = static_cast(this->requests_size()); i < n; i++) { - target = ::google::protobuf::internal::WireFormatLite:: - InternalWriteMessageToArray( - 1, this->requests(static_cast(i)), target); - } - - // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; - for (unsigned int i = 0, - n = static_cast(this->limits_size()); i < n; i++) { + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + if (this->has_gpu_accelerator()) { target = ::google::protobuf::internal::WireFormatLite:: InternalWriteMessageToArray( - 2, this->limits(static_cast(i)), target); + 1, HasBitSetters::gpu_accelerator(this), target); } if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); } - // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.Resources) + // @@protoc_insertion_point(serialize_to_array_end:flyteidl.core.ExtendedResources) return target; } -size_t Resources::ByteSizeLong() const { -// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.Resources) +size_t ExtendedResources::ByteSizeLong() const { +// @@protoc_insertion_point(message_byte_size_start:flyteidl.core.ExtendedResources) size_t total_size = 0; if (_internal_metadata_.have_unknown_fields()) { @@ -1451,26 +2278,11 @@ size_t Resources::ByteSizeLong() const { // Prevent compiler warnings about cached_has_bits being unused (void) cached_has_bits; - // repeated .flyteidl.core.Resources.ResourceEntry requests = 1; - { - unsigned int count = static_cast(this->requests_size()); - total_size += 1UL * count; - for (unsigned int i = 0; i < count; i++) { - total_size += - ::google::protobuf::internal::WireFormatLite::MessageSize( - this->requests(static_cast(i))); - } - } - - // repeated .flyteidl.core.Resources.ResourceEntry limits = 2; - { - unsigned int count = static_cast(this->limits_size()); - total_size += 1UL * count; - for (unsigned int i = 0; i < count; i++) { - total_size += - ::google::protobuf::internal::WireFormatLite::MessageSize( - this->limits(static_cast(i))); - } + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + if (this->has_gpu_accelerator()) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::MessageSize( + *gpu_accelerator_); } int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); @@ -1478,62 +2290,62 @@ size_t Resources::ByteSizeLong() const { return total_size; } -void Resources::MergeFrom(const ::google::protobuf::Message& from) { -// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.Resources) +void ExtendedResources::MergeFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_merge_from_start:flyteidl.core.ExtendedResources) GOOGLE_DCHECK_NE(&from, this); - const Resources* source = - ::google::protobuf::DynamicCastToGenerated( + const ExtendedResources* source = + ::google::protobuf::DynamicCastToGenerated( &from); if (source == nullptr) { - // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.Resources) + // @@protoc_insertion_point(generalized_merge_from_cast_fail:flyteidl.core.ExtendedResources) ::google::protobuf::internal::ReflectionOps::Merge(from, this); } else { - // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.Resources) + // @@protoc_insertion_point(generalized_merge_from_cast_success:flyteidl.core.ExtendedResources) MergeFrom(*source); } } -void Resources::MergeFrom(const Resources& from) { -// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.Resources) +void ExtendedResources::MergeFrom(const ExtendedResources& from) { +// @@protoc_insertion_point(class_specific_merge_from_start:flyteidl.core.ExtendedResources) GOOGLE_DCHECK_NE(&from, this); _internal_metadata_.MergeFrom(from._internal_metadata_); ::google::protobuf::uint32 cached_has_bits = 0; (void) cached_has_bits; - requests_.MergeFrom(from.requests_); - limits_.MergeFrom(from.limits_); + if (from.has_gpu_accelerator()) { + mutable_gpu_accelerator()->::flyteidl::core::GPUAccelerator::MergeFrom(from.gpu_accelerator()); + } } -void Resources::CopyFrom(const ::google::protobuf::Message& from) { -// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.Resources) +void ExtendedResources::CopyFrom(const ::google::protobuf::Message& from) { +// @@protoc_insertion_point(generalized_copy_from_start:flyteidl.core.ExtendedResources) if (&from == this) return; Clear(); MergeFrom(from); } -void Resources::CopyFrom(const Resources& from) { -// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.Resources) +void ExtendedResources::CopyFrom(const ExtendedResources& from) { +// @@protoc_insertion_point(class_specific_copy_from_start:flyteidl.core.ExtendedResources) if (&from == this) return; Clear(); MergeFrom(from); } -bool Resources::IsInitialized() const { +bool ExtendedResources::IsInitialized() const { return true; } -void Resources::Swap(Resources* other) { +void ExtendedResources::Swap(ExtendedResources* other) { if (other == this) return; InternalSwap(other); } -void Resources::InternalSwap(Resources* other) { +void ExtendedResources::InternalSwap(ExtendedResources* other) { using std::swap; _internal_metadata_.Swap(&other->_internal_metadata_); - CastToBase(&requests_)->InternalSwap(CastToBase(&other->requests_)); - CastToBase(&limits_)->InternalSwap(CastToBase(&other->limits_)); + swap(gpu_accelerator_, other->gpu_accelerator_); } -::google::protobuf::Metadata Resources::GetMetadata() const { +::google::protobuf::Metadata ExtendedResources::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[kIndexInFileMessages]; } @@ -1967,7 +2779,7 @@ void TaskMetadata_TagsEntry_DoNotUse::MergeFrom(const TaskMetadata_TagsEntry_DoN } ::google::protobuf::Metadata TaskMetadata_TagsEntry_DoNotUse::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); - return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[3]; + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[5]; } void TaskMetadata_TagsEntry_DoNotUse::MergeFrom( const ::google::protobuf::Message& other) { @@ -3036,7 +3848,7 @@ void TaskTemplate_ConfigEntry_DoNotUse::MergeFrom(const TaskTemplate_ConfigEntry } ::google::protobuf::Metadata TaskTemplate_ConfigEntry_DoNotUse::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); - return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[5]; + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[7]; } void TaskTemplate_ConfigEntry_DoNotUse::MergeFrom( const ::google::protobuf::Message& other) { @@ -3086,6 +3898,8 @@ void TaskTemplate::InitAsDefaultInstance() { ::flyteidl::core::Sql::internal_default_instance()); ::flyteidl::core::_TaskTemplate_default_instance_._instance.get_mutable()->security_context_ = const_cast< ::flyteidl::core::SecurityContext*>( ::flyteidl::core::SecurityContext::internal_default_instance()); + ::flyteidl::core::_TaskTemplate_default_instance_._instance.get_mutable()->extended_resources_ = const_cast< ::flyteidl::core::ExtendedResources*>( + ::flyteidl::core::ExtendedResources::internal_default_instance()); } class TaskTemplate::HasBitSetters { public: @@ -3097,6 +3911,7 @@ class TaskTemplate::HasBitSetters { static const ::flyteidl::core::K8sPod& k8s_pod(const TaskTemplate* msg); static const ::flyteidl::core::Sql& sql(const TaskTemplate* msg); static const ::flyteidl::core::SecurityContext& security_context(const TaskTemplate* msg); + static const ::flyteidl::core::ExtendedResources& extended_resources(const TaskTemplate* msg); }; const ::flyteidl::core::Identifier& @@ -3131,6 +3946,10 @@ const ::flyteidl::core::SecurityContext& TaskTemplate::HasBitSetters::security_context(const TaskTemplate* msg) { return *msg->security_context_; } +const ::flyteidl::core::ExtendedResources& +TaskTemplate::HasBitSetters::extended_resources(const TaskTemplate* msg) { + return *msg->extended_resources_; +} void TaskTemplate::clear_id() { if (GetArenaNoVirtual() == nullptr && id_ != nullptr) { delete id_; @@ -3208,6 +4027,7 @@ const int TaskTemplate::kK8SPodFieldNumber; const int TaskTemplate::kSqlFieldNumber; const int TaskTemplate::kTaskTypeVersionFieldNumber; const int TaskTemplate::kSecurityContextFieldNumber; +const int TaskTemplate::kExtendedResourcesFieldNumber; const int TaskTemplate::kConfigFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 @@ -3250,6 +4070,11 @@ TaskTemplate::TaskTemplate(const TaskTemplate& from) } else { security_context_ = nullptr; } + if (from.has_extended_resources()) { + extended_resources_ = new ::flyteidl::core::ExtendedResources(*from.extended_resources_); + } else { + extended_resources_ = nullptr; + } task_type_version_ = from.task_type_version_; clear_has_target(); switch (from.target_case()) { @@ -3294,6 +4119,7 @@ void TaskTemplate::SharedDtor() { if (this != internal_default_instance()) delete interface_; if (this != internal_default_instance()) delete custom_; if (this != internal_default_instance()) delete security_context_; + if (this != internal_default_instance()) delete extended_resources_; if (has_target()) { clear_target(); } @@ -3359,6 +4185,10 @@ void TaskTemplate::Clear() { delete security_context_; } security_context_ = nullptr; + if (GetArenaNoVirtual() == nullptr && extended_resources_ != nullptr) { + delete extended_resources_; + } + extended_resources_ = nullptr; task_type_version_ = 0; clear_target(); _internal_metadata_.Clear(); @@ -3478,6 +4308,19 @@ const char* TaskTemplate::_InternalParse(const char* begin, const char* end, voi {parser_till_end, object}, ptr - size, ptr)); break; } + // .flyteidl.core.ExtendedResources extended_resources = 9; + case 9: { + if (static_cast<::google::protobuf::uint8>(tag) != 74) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::core::ExtendedResources::_InternalParse; + object = msg->mutable_extended_resources(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); + break; + } // map config = 16; case 16: { if (static_cast<::google::protobuf::uint8>(tag) != 130) goto handle_unusual; @@ -3651,6 +4494,17 @@ bool TaskTemplate::MergePartialFromCodedStream( break; } + // .flyteidl.core.ExtendedResources extended_resources = 9; + case 9: { + if (static_cast< ::google::protobuf::uint8>(tag) == (74 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( + input, mutable_extended_resources())); + } else { + goto handle_unusual; + } + break; + } + // map config = 16; case 16: { if (static_cast< ::google::protobuf::uint8>(tag) == (130 & 0xFF)) { @@ -3777,6 +4631,12 @@ void TaskTemplate::SerializeWithCachedSizes( 8, HasBitSetters::security_context(this), output); } + // .flyteidl.core.ExtendedResources extended_resources = 9; + if (this->has_extended_resources()) { + ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( + 9, HasBitSetters::extended_resources(this), output); + } + // map config = 16; if (!this->config().empty()) { typedef ::google::protobuf::Map< ::std::string, ::std::string >::const_pointer @@ -3909,6 +4769,13 @@ ::google::protobuf::uint8* TaskTemplate::InternalSerializeWithCachedSizesToArray 8, HasBitSetters::security_context(this), target); } + // .flyteidl.core.ExtendedResources extended_resources = 9; + if (this->has_extended_resources()) { + target = ::google::protobuf::internal::WireFormatLite:: + InternalWriteMessageToArray( + 9, HasBitSetters::extended_resources(this), target); + } + // map config = 16; if (!this->config().empty()) { typedef ::google::protobuf::Map< ::std::string, ::std::string >::const_pointer @@ -4049,6 +4916,13 @@ size_t TaskTemplate::ByteSizeLong() const { *security_context_); } + // .flyteidl.core.ExtendedResources extended_resources = 9; + if (this->has_extended_resources()) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::MessageSize( + *extended_resources_); + } + // int32 task_type_version = 7; if (this->task_type_version() != 0) { total_size += 1 + @@ -4129,6 +5003,9 @@ void TaskTemplate::MergeFrom(const TaskTemplate& from) { if (from.has_security_context()) { mutable_security_context()->::flyteidl::core::SecurityContext::MergeFrom(from.security_context()); } + if (from.has_extended_resources()) { + mutable_extended_resources()->::flyteidl::core::ExtendedResources::MergeFrom(from.extended_resources()); + } if (from.task_type_version() != 0) { set_task_type_version(from.task_type_version()); } @@ -4184,6 +5061,7 @@ void TaskTemplate::InternalSwap(TaskTemplate* other) { swap(interface_, other->interface_); swap(custom_, other->custom_); swap(security_context_, other->security_context_); + swap(extended_resources_, other->extended_resources_); swap(task_type_version_, other->task_type_version_); swap(target_, other->target_); swap(_oneof_case_[0], other->_oneof_case_[0]); @@ -6537,7 +7415,7 @@ void K8sObjectMetadata_LabelsEntry_DoNotUse::MergeFrom(const K8sObjectMetadata_L } ::google::protobuf::Metadata K8sObjectMetadata_LabelsEntry_DoNotUse::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); - return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[12]; + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[14]; } void K8sObjectMetadata_LabelsEntry_DoNotUse::MergeFrom( const ::google::protobuf::Message& other) { @@ -6578,7 +7456,7 @@ void K8sObjectMetadata_AnnotationsEntry_DoNotUse::MergeFrom(const K8sObjectMetad } ::google::protobuf::Metadata K8sObjectMetadata_AnnotationsEntry_DoNotUse::GetMetadata() const { ::google::protobuf::internal::AssignDescriptors(&::assign_descriptors_table_flyteidl_2fcore_2ftasks_2eproto); - return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[13]; + return ::file_level_metadata_flyteidl_2fcore_2ftasks_2eproto[15]; } void K8sObjectMetadata_AnnotationsEntry_DoNotUse::MergeFrom( const ::google::protobuf::Message& other) { @@ -7509,6 +8387,12 @@ template<> PROTOBUF_NOINLINE ::flyteidl::core::Resources_ResourceEntry* Arena::C template<> PROTOBUF_NOINLINE ::flyteidl::core::Resources* Arena::CreateMaybeMessage< ::flyteidl::core::Resources >(Arena* arena) { return Arena::CreateInternal< ::flyteidl::core::Resources >(arena); } +template<> PROTOBUF_NOINLINE ::flyteidl::core::GPUAccelerator* Arena::CreateMaybeMessage< ::flyteidl::core::GPUAccelerator >(Arena* arena) { + return Arena::CreateInternal< ::flyteidl::core::GPUAccelerator >(arena); +} +template<> PROTOBUF_NOINLINE ::flyteidl::core::ExtendedResources* Arena::CreateMaybeMessage< ::flyteidl::core::ExtendedResources >(Arena* arena) { + return Arena::CreateInternal< ::flyteidl::core::ExtendedResources >(arena); +} template<> PROTOBUF_NOINLINE ::flyteidl::core::RuntimeMetadata* Arena::CreateMaybeMessage< ::flyteidl::core::RuntimeMetadata >(Arena* arena) { return Arena::CreateInternal< ::flyteidl::core::RuntimeMetadata >(arena); } diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.h b/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.h index 430d867123d..37446ea3459 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.h +++ b/flyteidl/gen/pb-cpp/flyteidl/core/tasks.pb.h @@ -51,7 +51,7 @@ struct TableStruct_flyteidl_2fcore_2ftasks_2eproto { PROTOBUF_SECTION_VARIABLE(protodesc_cold); static const ::google::protobuf::internal::AuxillaryParseTableField aux[] PROTOBUF_SECTION_VARIABLE(protodesc_cold); - static const ::google::protobuf::internal::ParseTable schema[16] + static const ::google::protobuf::internal::ParseTable schema[18] PROTOBUF_SECTION_VARIABLE(protodesc_cold); static const ::google::protobuf::internal::FieldMetadata field_metadata[]; static const ::google::protobuf::internal::SerializationTable serialization_table[]; @@ -69,6 +69,12 @@ extern ContainerPortDefaultTypeInternal _ContainerPort_default_instance_; class DataLoadingConfig; class DataLoadingConfigDefaultTypeInternal; extern DataLoadingConfigDefaultTypeInternal _DataLoadingConfig_default_instance_; +class ExtendedResources; +class ExtendedResourcesDefaultTypeInternal; +extern ExtendedResourcesDefaultTypeInternal _ExtendedResources_default_instance_; +class GPUAccelerator; +class GPUAcceleratorDefaultTypeInternal; +extern GPUAcceleratorDefaultTypeInternal _GPUAccelerator_default_instance_; class IOStrategy; class IOStrategyDefaultTypeInternal; extern IOStrategyDefaultTypeInternal _IOStrategy_default_instance_; @@ -115,6 +121,8 @@ namespace protobuf { template<> ::flyteidl::core::Container* Arena::CreateMaybeMessage<::flyteidl::core::Container>(Arena*); template<> ::flyteidl::core::ContainerPort* Arena::CreateMaybeMessage<::flyteidl::core::ContainerPort>(Arena*); template<> ::flyteidl::core::DataLoadingConfig* Arena::CreateMaybeMessage<::flyteidl::core::DataLoadingConfig>(Arena*); +template<> ::flyteidl::core::ExtendedResources* Arena::CreateMaybeMessage<::flyteidl::core::ExtendedResources>(Arena*); +template<> ::flyteidl::core::GPUAccelerator* Arena::CreateMaybeMessage<::flyteidl::core::GPUAccelerator>(Arena*); template<> ::flyteidl::core::IOStrategy* Arena::CreateMaybeMessage<::flyteidl::core::IOStrategy>(Arena*); template<> ::flyteidl::core::K8sObjectMetadata* Arena::CreateMaybeMessage<::flyteidl::core::K8sObjectMetadata>(Arena*); template<> ::flyteidl::core::K8sObjectMetadata_AnnotationsEntry_DoNotUse* Arena::CreateMaybeMessage<::flyteidl::core::K8sObjectMetadata_AnnotationsEntry_DoNotUse>(Arena*); @@ -588,6 +596,287 @@ class Resources final : }; // ------------------------------------------------------------------- +class GPUAccelerator final : + public ::google::protobuf::Message /* @@protoc_insertion_point(class_definition:flyteidl.core.GPUAccelerator) */ { + public: + GPUAccelerator(); + virtual ~GPUAccelerator(); + + GPUAccelerator(const GPUAccelerator& from); + + inline GPUAccelerator& operator=(const GPUAccelerator& from) { + CopyFrom(from); + return *this; + } + #if LANG_CXX11 + GPUAccelerator(GPUAccelerator&& from) noexcept + : GPUAccelerator() { + *this = ::std::move(from); + } + + inline GPUAccelerator& operator=(GPUAccelerator&& from) noexcept { + if (GetArenaNoVirtual() == from.GetArenaNoVirtual()) { + if (this != &from) InternalSwap(&from); + } else { + CopyFrom(from); + } + return *this; + } + #endif + static const ::google::protobuf::Descriptor* descriptor() { + return default_instance().GetDescriptor(); + } + static const GPUAccelerator& default_instance(); + + enum PartitionSizeValueCase { + kUnpartitioned = 2, + kPartitionSize = 3, + PARTITION_SIZE_VALUE_NOT_SET = 0, + }; + + static void InitAsDefaultInstance(); // FOR INTERNAL USE ONLY + static inline const GPUAccelerator* internal_default_instance() { + return reinterpret_cast( + &_GPUAccelerator_default_instance_); + } + static constexpr int kIndexInFileMessages = + 2; + + void Swap(GPUAccelerator* other); + friend void swap(GPUAccelerator& a, GPUAccelerator& b) { + a.Swap(&b); + } + + // implements Message ---------------------------------------------- + + inline GPUAccelerator* New() const final { + return CreateMaybeMessage(nullptr); + } + + GPUAccelerator* New(::google::protobuf::Arena* arena) const final { + return CreateMaybeMessage(arena); + } + void CopyFrom(const ::google::protobuf::Message& from) final; + void MergeFrom(const ::google::protobuf::Message& from) final; + void CopyFrom(const GPUAccelerator& from); + void MergeFrom(const GPUAccelerator& from); + PROTOBUF_ATTRIBUTE_REINITIALIZES void Clear() final; + bool IsInitialized() const final; + + size_t ByteSizeLong() const final; + #if GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + static const char* _InternalParse(const char* begin, const char* end, void* object, ::google::protobuf::internal::ParseContext* ctx); + ::google::protobuf::internal::ParseFunc _ParseFunc() const final { return _InternalParse; } + #else + bool MergePartialFromCodedStream( + ::google::protobuf::io::CodedInputStream* input) final; + #endif // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + void SerializeWithCachedSizes( + ::google::protobuf::io::CodedOutputStream* output) const final; + ::google::protobuf::uint8* InternalSerializeWithCachedSizesToArray( + ::google::protobuf::uint8* target) const final; + int GetCachedSize() const final { return _cached_size_.Get(); } + + private: + void SharedCtor(); + void SharedDtor(); + void SetCachedSize(int size) const final; + void InternalSwap(GPUAccelerator* other); + private: + inline ::google::protobuf::Arena* GetArenaNoVirtual() const { + return nullptr; + } + inline void* MaybeArenaPtr() const { + return nullptr; + } + public: + + ::google::protobuf::Metadata GetMetadata() const final; + + // nested types ---------------------------------------------------- + + // accessors ------------------------------------------------------- + + // string device = 1; + void clear_device(); + static const int kDeviceFieldNumber = 1; + const ::std::string& device() const; + void set_device(const ::std::string& value); + #if LANG_CXX11 + void set_device(::std::string&& value); + #endif + void set_device(const char* value); + void set_device(const char* value, size_t size); + ::std::string* mutable_device(); + ::std::string* release_device(); + void set_allocated_device(::std::string* device); + + // bool unpartitioned = 2; + private: + bool has_unpartitioned() const; + public: + void clear_unpartitioned(); + static const int kUnpartitionedFieldNumber = 2; + bool unpartitioned() const; + void set_unpartitioned(bool value); + + // string partition_size = 3; + private: + bool has_partition_size() const; + public: + void clear_partition_size(); + static const int kPartitionSizeFieldNumber = 3; + const ::std::string& partition_size() const; + void set_partition_size(const ::std::string& value); + #if LANG_CXX11 + void set_partition_size(::std::string&& value); + #endif + void set_partition_size(const char* value); + void set_partition_size(const char* value, size_t size); + ::std::string* mutable_partition_size(); + ::std::string* release_partition_size(); + void set_allocated_partition_size(::std::string* partition_size); + + void clear_partition_size_value(); + PartitionSizeValueCase partition_size_value_case() const; + // @@protoc_insertion_point(class_scope:flyteidl.core.GPUAccelerator) + private: + class HasBitSetters; + void set_has_unpartitioned(); + void set_has_partition_size(); + + inline bool has_partition_size_value() const; + inline void clear_has_partition_size_value(); + + ::google::protobuf::internal::InternalMetadataWithArena _internal_metadata_; + ::google::protobuf::internal::ArenaStringPtr device_; + union PartitionSizeValueUnion { + PartitionSizeValueUnion() {} + bool unpartitioned_; + ::google::protobuf::internal::ArenaStringPtr partition_size_; + } partition_size_value_; + mutable ::google::protobuf::internal::CachedSize _cached_size_; + ::google::protobuf::uint32 _oneof_case_[1]; + + friend struct ::TableStruct_flyteidl_2fcore_2ftasks_2eproto; +}; +// ------------------------------------------------------------------- + +class ExtendedResources final : + public ::google::protobuf::Message /* @@protoc_insertion_point(class_definition:flyteidl.core.ExtendedResources) */ { + public: + ExtendedResources(); + virtual ~ExtendedResources(); + + ExtendedResources(const ExtendedResources& from); + + inline ExtendedResources& operator=(const ExtendedResources& from) { + CopyFrom(from); + return *this; + } + #if LANG_CXX11 + ExtendedResources(ExtendedResources&& from) noexcept + : ExtendedResources() { + *this = ::std::move(from); + } + + inline ExtendedResources& operator=(ExtendedResources&& from) noexcept { + if (GetArenaNoVirtual() == from.GetArenaNoVirtual()) { + if (this != &from) InternalSwap(&from); + } else { + CopyFrom(from); + } + return *this; + } + #endif + static const ::google::protobuf::Descriptor* descriptor() { + return default_instance().GetDescriptor(); + } + static const ExtendedResources& default_instance(); + + static void InitAsDefaultInstance(); // FOR INTERNAL USE ONLY + static inline const ExtendedResources* internal_default_instance() { + return reinterpret_cast( + &_ExtendedResources_default_instance_); + } + static constexpr int kIndexInFileMessages = + 3; + + void Swap(ExtendedResources* other); + friend void swap(ExtendedResources& a, ExtendedResources& b) { + a.Swap(&b); + } + + // implements Message ---------------------------------------------- + + inline ExtendedResources* New() const final { + return CreateMaybeMessage(nullptr); + } + + ExtendedResources* New(::google::protobuf::Arena* arena) const final { + return CreateMaybeMessage(arena); + } + void CopyFrom(const ::google::protobuf::Message& from) final; + void MergeFrom(const ::google::protobuf::Message& from) final; + void CopyFrom(const ExtendedResources& from); + void MergeFrom(const ExtendedResources& from); + PROTOBUF_ATTRIBUTE_REINITIALIZES void Clear() final; + bool IsInitialized() const final; + + size_t ByteSizeLong() const final; + #if GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + static const char* _InternalParse(const char* begin, const char* end, void* object, ::google::protobuf::internal::ParseContext* ctx); + ::google::protobuf::internal::ParseFunc _ParseFunc() const final { return _InternalParse; } + #else + bool MergePartialFromCodedStream( + ::google::protobuf::io::CodedInputStream* input) final; + #endif // GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER + void SerializeWithCachedSizes( + ::google::protobuf::io::CodedOutputStream* output) const final; + ::google::protobuf::uint8* InternalSerializeWithCachedSizesToArray( + ::google::protobuf::uint8* target) const final; + int GetCachedSize() const final { return _cached_size_.Get(); } + + private: + void SharedCtor(); + void SharedDtor(); + void SetCachedSize(int size) const final; + void InternalSwap(ExtendedResources* other); + private: + inline ::google::protobuf::Arena* GetArenaNoVirtual() const { + return nullptr; + } + inline void* MaybeArenaPtr() const { + return nullptr; + } + public: + + ::google::protobuf::Metadata GetMetadata() const final; + + // nested types ---------------------------------------------------- + + // accessors ------------------------------------------------------- + + // .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + bool has_gpu_accelerator() const; + void clear_gpu_accelerator(); + static const int kGpuAcceleratorFieldNumber = 1; + const ::flyteidl::core::GPUAccelerator& gpu_accelerator() const; + ::flyteidl::core::GPUAccelerator* release_gpu_accelerator(); + ::flyteidl::core::GPUAccelerator* mutable_gpu_accelerator(); + void set_allocated_gpu_accelerator(::flyteidl::core::GPUAccelerator* gpu_accelerator); + + // @@protoc_insertion_point(class_scope:flyteidl.core.ExtendedResources) + private: + class HasBitSetters; + + ::google::protobuf::internal::InternalMetadataWithArena _internal_metadata_; + ::flyteidl::core::GPUAccelerator* gpu_accelerator_; + mutable ::google::protobuf::internal::CachedSize _cached_size_; + friend struct ::TableStruct_flyteidl_2fcore_2ftasks_2eproto; +}; +// ------------------------------------------------------------------- + class RuntimeMetadata final : public ::google::protobuf::Message /* @@protoc_insertion_point(class_definition:flyteidl.core.RuntimeMetadata) */ { public: @@ -626,7 +915,7 @@ class RuntimeMetadata final : &_RuntimeMetadata_default_instance_); } static constexpr int kIndexInFileMessages = - 2; + 4; void Swap(RuntimeMetadata* other); friend void swap(RuntimeMetadata& a, RuntimeMetadata& b) { @@ -823,7 +1112,7 @@ class TaskMetadata final : &_TaskMetadata_default_instance_); } static constexpr int kIndexInFileMessages = - 4; + 6; void Swap(TaskMetadata* other); friend void swap(TaskMetadata& a, TaskMetadata& b) { @@ -1092,7 +1381,7 @@ class TaskTemplate final : &_TaskTemplate_default_instance_); } static constexpr int kIndexInFileMessages = - 6; + 8; void Swap(TaskTemplate* other); friend void swap(TaskTemplate& a, TaskTemplate& b) { @@ -1218,6 +1507,15 @@ class TaskTemplate final : ::flyteidl::core::SecurityContext* mutable_security_context(); void set_allocated_security_context(::flyteidl::core::SecurityContext* security_context); + // .flyteidl.core.ExtendedResources extended_resources = 9; + bool has_extended_resources() const; + void clear_extended_resources(); + static const int kExtendedResourcesFieldNumber = 9; + const ::flyteidl::core::ExtendedResources& extended_resources() const; + ::flyteidl::core::ExtendedResources* release_extended_resources(); + ::flyteidl::core::ExtendedResources* mutable_extended_resources(); + void set_allocated_extended_resources(::flyteidl::core::ExtendedResources* extended_resources); + // int32 task_type_version = 7; void clear_task_type_version(); static const int kTaskTypeVersionFieldNumber = 7; @@ -1276,6 +1574,7 @@ class TaskTemplate final : ::flyteidl::core::TypedInterface* interface_; ::google::protobuf::Struct* custom_; ::flyteidl::core::SecurityContext* security_context_; + ::flyteidl::core::ExtendedResources* extended_resources_; ::google::protobuf::int32 task_type_version_; union TargetUnion { TargetUnion() {} @@ -1328,7 +1627,7 @@ class ContainerPort final : &_ContainerPort_default_instance_); } static constexpr int kIndexInFileMessages = - 7; + 9; void Swap(ContainerPort* other); friend void swap(ContainerPort& a, ContainerPort& b) { @@ -1440,7 +1739,7 @@ class Container final : &_Container_default_instance_); } static constexpr int kIndexInFileMessages = - 8; + 10; void Swap(Container* other); friend void swap(Container& a, Container& b) { @@ -1704,7 +2003,7 @@ class IOStrategy final : &_IOStrategy_default_instance_); } static constexpr int kIndexInFileMessages = - 9; + 11; void Swap(IOStrategy* other); friend void swap(IOStrategy& a, IOStrategy& b) { @@ -1879,7 +2178,7 @@ class DataLoadingConfig final : &_DataLoadingConfig_default_instance_); } static constexpr int kIndexInFileMessages = - 10; + 12; void Swap(DataLoadingConfig* other); friend void swap(DataLoadingConfig& a, DataLoadingConfig& b) { @@ -2066,7 +2365,7 @@ class K8sPod final : &_K8sPod_default_instance_); } static constexpr int kIndexInFileMessages = - 11; + 13; void Swap(K8sPod* other); friend void swap(K8sPod& a, K8sPod& b) { @@ -2249,7 +2548,7 @@ class K8sObjectMetadata final : &_K8sObjectMetadata_default_instance_); } static constexpr int kIndexInFileMessages = - 14; + 16; void Swap(K8sObjectMetadata* other); friend void swap(K8sObjectMetadata& a, K8sObjectMetadata& b) { @@ -2385,7 +2684,7 @@ class Sql final : &_Sql_default_instance_); } static constexpr int kIndexInFileMessages = - 15; + 17; void Swap(Sql* other); friend void swap(Sql& a, Sql& b) { @@ -2646,6 +2945,248 @@ Resources::limits() const { // ------------------------------------------------------------------- +// GPUAccelerator + +// string device = 1; +inline void GPUAccelerator::clear_device() { + device_.ClearToEmptyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline const ::std::string& GPUAccelerator::device() const { + // @@protoc_insertion_point(field_get:flyteidl.core.GPUAccelerator.device) + return device_.GetNoArena(); +} +inline void GPUAccelerator::set_device(const ::std::string& value) { + + device_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), value); + // @@protoc_insertion_point(field_set:flyteidl.core.GPUAccelerator.device) +} +#if LANG_CXX11 +inline void GPUAccelerator::set_device(::std::string&& value) { + + device_.SetNoArena( + &::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::move(value)); + // @@protoc_insertion_point(field_set_rvalue:flyteidl.core.GPUAccelerator.device) +} +#endif +inline void GPUAccelerator::set_device(const char* value) { + GOOGLE_DCHECK(value != nullptr); + + device_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::string(value)); + // @@protoc_insertion_point(field_set_char:flyteidl.core.GPUAccelerator.device) +} +inline void GPUAccelerator::set_device(const char* value, size_t size) { + + device_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), + ::std::string(reinterpret_cast(value), size)); + // @@protoc_insertion_point(field_set_pointer:flyteidl.core.GPUAccelerator.device) +} +inline ::std::string* GPUAccelerator::mutable_device() { + + // @@protoc_insertion_point(field_mutable:flyteidl.core.GPUAccelerator.device) + return device_.MutableNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline ::std::string* GPUAccelerator::release_device() { + // @@protoc_insertion_point(field_release:flyteidl.core.GPUAccelerator.device) + + return device_.ReleaseNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline void GPUAccelerator::set_allocated_device(::std::string* device) { + if (device != nullptr) { + + } else { + + } + device_.SetAllocatedNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), device); + // @@protoc_insertion_point(field_set_allocated:flyteidl.core.GPUAccelerator.device) +} + +// bool unpartitioned = 2; +inline bool GPUAccelerator::has_unpartitioned() const { + return partition_size_value_case() == kUnpartitioned; +} +inline void GPUAccelerator::set_has_unpartitioned() { + _oneof_case_[0] = kUnpartitioned; +} +inline void GPUAccelerator::clear_unpartitioned() { + if (has_unpartitioned()) { + partition_size_value_.unpartitioned_ = false; + clear_has_partition_size_value(); + } +} +inline bool GPUAccelerator::unpartitioned() const { + // @@protoc_insertion_point(field_get:flyteidl.core.GPUAccelerator.unpartitioned) + if (has_unpartitioned()) { + return partition_size_value_.unpartitioned_; + } + return false; +} +inline void GPUAccelerator::set_unpartitioned(bool value) { + if (!has_unpartitioned()) { + clear_partition_size_value(); + set_has_unpartitioned(); + } + partition_size_value_.unpartitioned_ = value; + // @@protoc_insertion_point(field_set:flyteidl.core.GPUAccelerator.unpartitioned) +} + +// string partition_size = 3; +inline bool GPUAccelerator::has_partition_size() const { + return partition_size_value_case() == kPartitionSize; +} +inline void GPUAccelerator::set_has_partition_size() { + _oneof_case_[0] = kPartitionSize; +} +inline void GPUAccelerator::clear_partition_size() { + if (has_partition_size()) { + partition_size_value_.partition_size_.DestroyNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + clear_has_partition_size_value(); + } +} +inline const ::std::string& GPUAccelerator::partition_size() const { + // @@protoc_insertion_point(field_get:flyteidl.core.GPUAccelerator.partition_size) + if (has_partition_size()) { + return partition_size_value_.partition_size_.GetNoArena(); + } + return *&::google::protobuf::internal::GetEmptyStringAlreadyInited(); +} +inline void GPUAccelerator::set_partition_size(const ::std::string& value) { + // @@protoc_insertion_point(field_set:flyteidl.core.GPUAccelerator.partition_size) + if (!has_partition_size()) { + clear_partition_size_value(); + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } + partition_size_value_.partition_size_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), value); + // @@protoc_insertion_point(field_set:flyteidl.core.GPUAccelerator.partition_size) +} +#if LANG_CXX11 +inline void GPUAccelerator::set_partition_size(::std::string&& value) { + // @@protoc_insertion_point(field_set:flyteidl.core.GPUAccelerator.partition_size) + if (!has_partition_size()) { + clear_partition_size_value(); + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } + partition_size_value_.partition_size_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::move(value)); + // @@protoc_insertion_point(field_set_rvalue:flyteidl.core.GPUAccelerator.partition_size) +} +#endif +inline void GPUAccelerator::set_partition_size(const char* value) { + GOOGLE_DCHECK(value != nullptr); + if (!has_partition_size()) { + clear_partition_size_value(); + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } + partition_size_value_.partition_size_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), + ::std::string(value)); + // @@protoc_insertion_point(field_set_char:flyteidl.core.GPUAccelerator.partition_size) +} +inline void GPUAccelerator::set_partition_size(const char* value, size_t size) { + if (!has_partition_size()) { + clear_partition_size_value(); + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } + partition_size_value_.partition_size_.SetNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited(), ::std::string( + reinterpret_cast(value), size)); + // @@protoc_insertion_point(field_set_pointer:flyteidl.core.GPUAccelerator.partition_size) +} +inline ::std::string* GPUAccelerator::mutable_partition_size() { + if (!has_partition_size()) { + clear_partition_size_value(); + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } + // @@protoc_insertion_point(field_mutable:flyteidl.core.GPUAccelerator.partition_size) + return partition_size_value_.partition_size_.MutableNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); +} +inline ::std::string* GPUAccelerator::release_partition_size() { + // @@protoc_insertion_point(field_release:flyteidl.core.GPUAccelerator.partition_size) + if (has_partition_size()) { + clear_has_partition_size_value(); + return partition_size_value_.partition_size_.ReleaseNoArena(&::google::protobuf::internal::GetEmptyStringAlreadyInited()); + } else { + return nullptr; + } +} +inline void GPUAccelerator::set_allocated_partition_size(::std::string* partition_size) { + if (has_partition_size_value()) { + clear_partition_size_value(); + } + if (partition_size != nullptr) { + set_has_partition_size(); + partition_size_value_.partition_size_.UnsafeSetDefault(partition_size); + } + // @@protoc_insertion_point(field_set_allocated:flyteidl.core.GPUAccelerator.partition_size) +} + +inline bool GPUAccelerator::has_partition_size_value() const { + return partition_size_value_case() != PARTITION_SIZE_VALUE_NOT_SET; +} +inline void GPUAccelerator::clear_has_partition_size_value() { + _oneof_case_[0] = PARTITION_SIZE_VALUE_NOT_SET; +} +inline GPUAccelerator::PartitionSizeValueCase GPUAccelerator::partition_size_value_case() const { + return GPUAccelerator::PartitionSizeValueCase(_oneof_case_[0]); +} +// ------------------------------------------------------------------- + +// ExtendedResources + +// .flyteidl.core.GPUAccelerator gpu_accelerator = 1; +inline bool ExtendedResources::has_gpu_accelerator() const { + return this != internal_default_instance() && gpu_accelerator_ != nullptr; +} +inline void ExtendedResources::clear_gpu_accelerator() { + if (GetArenaNoVirtual() == nullptr && gpu_accelerator_ != nullptr) { + delete gpu_accelerator_; + } + gpu_accelerator_ = nullptr; +} +inline const ::flyteidl::core::GPUAccelerator& ExtendedResources::gpu_accelerator() const { + const ::flyteidl::core::GPUAccelerator* p = gpu_accelerator_; + // @@protoc_insertion_point(field_get:flyteidl.core.ExtendedResources.gpu_accelerator) + return p != nullptr ? *p : *reinterpret_cast( + &::flyteidl::core::_GPUAccelerator_default_instance_); +} +inline ::flyteidl::core::GPUAccelerator* ExtendedResources::release_gpu_accelerator() { + // @@protoc_insertion_point(field_release:flyteidl.core.ExtendedResources.gpu_accelerator) + + ::flyteidl::core::GPUAccelerator* temp = gpu_accelerator_; + gpu_accelerator_ = nullptr; + return temp; +} +inline ::flyteidl::core::GPUAccelerator* ExtendedResources::mutable_gpu_accelerator() { + + if (gpu_accelerator_ == nullptr) { + auto* p = CreateMaybeMessage<::flyteidl::core::GPUAccelerator>(GetArenaNoVirtual()); + gpu_accelerator_ = p; + } + // @@protoc_insertion_point(field_mutable:flyteidl.core.ExtendedResources.gpu_accelerator) + return gpu_accelerator_; +} +inline void ExtendedResources::set_allocated_gpu_accelerator(::flyteidl::core::GPUAccelerator* gpu_accelerator) { + ::google::protobuf::Arena* message_arena = GetArenaNoVirtual(); + if (message_arena == nullptr) { + delete gpu_accelerator_; + } + if (gpu_accelerator) { + ::google::protobuf::Arena* submessage_arena = nullptr; + if (message_arena != submessage_arena) { + gpu_accelerator = ::google::protobuf::internal::GetOwnedMessage( + message_arena, gpu_accelerator, submessage_arena); + } + + } else { + + } + gpu_accelerator_ = gpu_accelerator; + // @@protoc_insertion_point(field_set_allocated:flyteidl.core.ExtendedResources.gpu_accelerator) +} + +// ------------------------------------------------------------------- + // RuntimeMetadata // .flyteidl.core.RuntimeMetadata.RuntimeType type = 1; @@ -3601,6 +4142,57 @@ inline void TaskTemplate::set_allocated_security_context(::flyteidl::core::Secur // @@protoc_insertion_point(field_set_allocated:flyteidl.core.TaskTemplate.security_context) } +// .flyteidl.core.ExtendedResources extended_resources = 9; +inline bool TaskTemplate::has_extended_resources() const { + return this != internal_default_instance() && extended_resources_ != nullptr; +} +inline void TaskTemplate::clear_extended_resources() { + if (GetArenaNoVirtual() == nullptr && extended_resources_ != nullptr) { + delete extended_resources_; + } + extended_resources_ = nullptr; +} +inline const ::flyteidl::core::ExtendedResources& TaskTemplate::extended_resources() const { + const ::flyteidl::core::ExtendedResources* p = extended_resources_; + // @@protoc_insertion_point(field_get:flyteidl.core.TaskTemplate.extended_resources) + return p != nullptr ? *p : *reinterpret_cast( + &::flyteidl::core::_ExtendedResources_default_instance_); +} +inline ::flyteidl::core::ExtendedResources* TaskTemplate::release_extended_resources() { + // @@protoc_insertion_point(field_release:flyteidl.core.TaskTemplate.extended_resources) + + ::flyteidl::core::ExtendedResources* temp = extended_resources_; + extended_resources_ = nullptr; + return temp; +} +inline ::flyteidl::core::ExtendedResources* TaskTemplate::mutable_extended_resources() { + + if (extended_resources_ == nullptr) { + auto* p = CreateMaybeMessage<::flyteidl::core::ExtendedResources>(GetArenaNoVirtual()); + extended_resources_ = p; + } + // @@protoc_insertion_point(field_mutable:flyteidl.core.TaskTemplate.extended_resources) + return extended_resources_; +} +inline void TaskTemplate::set_allocated_extended_resources(::flyteidl::core::ExtendedResources* extended_resources) { + ::google::protobuf::Arena* message_arena = GetArenaNoVirtual(); + if (message_arena == nullptr) { + delete extended_resources_; + } + if (extended_resources) { + ::google::protobuf::Arena* submessage_arena = nullptr; + if (message_arena != submessage_arena) { + extended_resources = ::google::protobuf::internal::GetOwnedMessage( + message_arena, extended_resources, submessage_arena); + } + + } else { + + } + extended_resources_ = extended_resources; + // @@protoc_insertion_point(field_set_allocated:flyteidl.core.TaskTemplate.extended_resources) +} + // map config = 16; inline int TaskTemplate::config_size() const { return config_.size(); @@ -4562,6 +5154,10 @@ inline void Sql::set_dialect(::flyteidl::core::Sql_Dialect value) { // ------------------------------------------------------------------- +// ------------------------------------------------------------------- + +// ------------------------------------------------------------------- + // @@protoc_insertion_point(namespace_scope) diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.cc index 5248fdd5b7b..849b2faa207 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.cc @@ -22,6 +22,7 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fidentifier_2eproto ::google::p extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2finterface_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_TypedInterface_flyteidl_2fcore_2finterface_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_RetryStrategy_flyteidl_2fcore_2fliterals_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_Binding_flyteidl_2fcore_2fliterals_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftypes_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_Error_flyteidl_2fcore_2ftypes_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftypes_2eproto ::google::protobuf::internal::SCCInfo<6> scc_info_LiteralType_flyteidl_2fcore_2ftypes_2eproto; @@ -31,9 +32,9 @@ extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::pro extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<0> scc_info_WorkflowMetadata_TagsEntry_DoNotUse_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_SignalCondition_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_SleepCondition_flyteidl_2fcore_2fworkflow_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_TaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<1> scc_info_WorkflowNode_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_NodeMetadata_flyteidl_2fcore_2fworkflow_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_TaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_TaskNode_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<2> scc_info_WorkflowMetadata_flyteidl_2fcore_2fworkflow_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<3> scc_info_GateNode_flyteidl_2fcore_2fworkflow_2eproto; @@ -373,9 +374,10 @@ static void InitDefaultsTaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto() { ::flyteidl::core::TaskNodeOverrides::InitAsDefaultInstance(); } -::google::protobuf::internal::SCCInfo<1> scc_info_TaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto = - {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 1, InitDefaultsTaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto}, { - &scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base,}}; +::google::protobuf::internal::SCCInfo<2> scc_info_TaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto = + {{ATOMIC_VAR_INIT(::google::protobuf::internal::SCCInfoBase::kUninitialized), 2, InitDefaultsTaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto}, { + &scc_info_Resources_flyteidl_2fcore_2ftasks_2eproto.base, + &scc_info_ExtendedResources_flyteidl_2fcore_2ftasks_2eproto.base,}}; void InitDefaults_flyteidl_2fcore_2fworkflow_2eproto() { ::google::protobuf::internal::InitSCC(&scc_info_TaskNode_flyteidl_2fcore_2fworkflow_2eproto.base); @@ -551,6 +553,7 @@ const ::google::protobuf::uint32 TableStruct_flyteidl_2fcore_2fworkflow_2eproto: ~0u, // no _oneof_case_ ~0u, // no _weak_field_map_ PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskNodeOverrides, resources_), + PROTOBUF_FIELD_OFFSET(::flyteidl::core::TaskNodeOverrides, extended_resources_), }; static const ::google::protobuf::internal::MigrationSchema schemas[] PROTOBUF_SECTION_VARIABLE(protodesc_cold) = { { 0, -1, sizeof(::flyteidl::core::IfBlock)}, @@ -672,15 +675,16 @@ const char descriptor_table_protodef_flyteidl_2fcore_2fworkflow_2eproto[] = "re.Binding\022)\n\014failure_node\030\006 \001(\0132\023.flyte" "idl.core.Node\022B\n\021metadata_defaults\030\007 \001(\013" "2\'.flyteidl.core.WorkflowMetadataDefault" - "s\"@\n\021TaskNodeOverrides\022+\n\tresources\030\001 \001(" - "\0132\030.flyteidl.core.ResourcesBresources_ = const_cast< ::flyteidl::core::Resources*>( ::flyteidl::core::Resources::internal_default_instance()); + ::flyteidl::core::_TaskNodeOverrides_default_instance_._instance.get_mutable()->extended_resources_ = const_cast< ::flyteidl::core::ExtendedResources*>( + ::flyteidl::core::ExtendedResources::internal_default_instance()); } class TaskNodeOverrides::HasBitSetters { public: static const ::flyteidl::core::Resources& resources(const TaskNodeOverrides* msg); + static const ::flyteidl::core::ExtendedResources& extended_resources(const TaskNodeOverrides* msg); }; const ::flyteidl::core::Resources& TaskNodeOverrides::HasBitSetters::resources(const TaskNodeOverrides* msg) { return *msg->resources_; } +const ::flyteidl::core::ExtendedResources& +TaskNodeOverrides::HasBitSetters::extended_resources(const TaskNodeOverrides* msg) { + return *msg->extended_resources_; +} void TaskNodeOverrides::clear_resources() { if (GetArenaNoVirtual() == nullptr && resources_ != nullptr) { delete resources_; } resources_ = nullptr; } +void TaskNodeOverrides::clear_extended_resources() { + if (GetArenaNoVirtual() == nullptr && extended_resources_ != nullptr) { + delete extended_resources_; + } + extended_resources_ = nullptr; +} #if !defined(_MSC_VER) || _MSC_VER >= 1900 const int TaskNodeOverrides::kResourcesFieldNumber; +const int TaskNodeOverrides::kExtendedResourcesFieldNumber; #endif // !defined(_MSC_VER) || _MSC_VER >= 1900 TaskNodeOverrides::TaskNodeOverrides() @@ -8140,13 +8158,20 @@ TaskNodeOverrides::TaskNodeOverrides(const TaskNodeOverrides& from) } else { resources_ = nullptr; } + if (from.has_extended_resources()) { + extended_resources_ = new ::flyteidl::core::ExtendedResources(*from.extended_resources_); + } else { + extended_resources_ = nullptr; + } // @@protoc_insertion_point(copy_constructor:flyteidl.core.TaskNodeOverrides) } void TaskNodeOverrides::SharedCtor() { ::google::protobuf::internal::InitSCC( &scc_info_TaskNodeOverrides_flyteidl_2fcore_2fworkflow_2eproto.base); - resources_ = nullptr; + ::memset(&resources_, 0, static_cast( + reinterpret_cast(&extended_resources_) - + reinterpret_cast(&resources_)) + sizeof(extended_resources_)); } TaskNodeOverrides::~TaskNodeOverrides() { @@ -8156,6 +8181,7 @@ TaskNodeOverrides::~TaskNodeOverrides() { void TaskNodeOverrides::SharedDtor() { if (this != internal_default_instance()) delete resources_; + if (this != internal_default_instance()) delete extended_resources_; } void TaskNodeOverrides::SetCachedSize(int size) const { @@ -8177,6 +8203,10 @@ void TaskNodeOverrides::Clear() { delete resources_; } resources_ = nullptr; + if (GetArenaNoVirtual() == nullptr && extended_resources_ != nullptr) { + delete extended_resources_; + } + extended_resources_ = nullptr; _internal_metadata_.Clear(); } @@ -8206,6 +8236,19 @@ const char* TaskNodeOverrides::_InternalParse(const char* begin, const char* end {parser_till_end, object}, ptr - size, ptr)); break; } + // .flyteidl.core.ExtendedResources extended_resources = 2; + case 2: { + if (static_cast<::google::protobuf::uint8>(tag) != 18) goto handle_unusual; + ptr = ::google::protobuf::io::ReadSize(ptr, &size); + GOOGLE_PROTOBUF_PARSER_ASSERT(ptr); + parser_till_end = ::flyteidl::core::ExtendedResources::_InternalParse; + object = msg->mutable_extended_resources(); + if (size > end - ptr) goto len_delim_till_end; + ptr += size; + GOOGLE_PROTOBUF_PARSER_ASSERT(ctx->ParseExactRange( + {parser_till_end, object}, ptr - size, ptr)); + break; + } default: { handle_unusual: if ((tag & 7) == 4 || tag == 0) { @@ -8247,6 +8290,17 @@ bool TaskNodeOverrides::MergePartialFromCodedStream( break; } + // .flyteidl.core.ExtendedResources extended_resources = 2; + case 2: { + if (static_cast< ::google::protobuf::uint8>(tag) == (18 & 0xFF)) { + DO_(::google::protobuf::internal::WireFormatLite::ReadMessage( + input, mutable_extended_resources())); + } else { + goto handle_unusual; + } + break; + } + default: { handle_unusual: if (tag == 0) { @@ -8280,6 +8334,12 @@ void TaskNodeOverrides::SerializeWithCachedSizes( 1, HasBitSetters::resources(this), output); } + // .flyteidl.core.ExtendedResources extended_resources = 2; + if (this->has_extended_resources()) { + ::google::protobuf::internal::WireFormatLite::WriteMessageMaybeToArray( + 2, HasBitSetters::extended_resources(this), output); + } + if (_internal_metadata_.have_unknown_fields()) { ::google::protobuf::internal::WireFormat::SerializeUnknownFields( _internal_metadata_.unknown_fields(), output); @@ -8300,6 +8360,13 @@ ::google::protobuf::uint8* TaskNodeOverrides::InternalSerializeWithCachedSizesTo 1, HasBitSetters::resources(this), target); } + // .flyteidl.core.ExtendedResources extended_resources = 2; + if (this->has_extended_resources()) { + target = ::google::protobuf::internal::WireFormatLite:: + InternalWriteMessageToArray( + 2, HasBitSetters::extended_resources(this), target); + } + if (_internal_metadata_.have_unknown_fields()) { target = ::google::protobuf::internal::WireFormat::SerializeUnknownFieldsToArray( _internal_metadata_.unknown_fields(), target); @@ -8328,6 +8395,13 @@ size_t TaskNodeOverrides::ByteSizeLong() const { *resources_); } + // .flyteidl.core.ExtendedResources extended_resources = 2; + if (this->has_extended_resources()) { + total_size += 1 + + ::google::protobuf::internal::WireFormatLite::MessageSize( + *extended_resources_); + } + int cached_size = ::google::protobuf::internal::ToCachedSize(total_size); SetCachedSize(cached_size); return total_size; @@ -8358,6 +8432,9 @@ void TaskNodeOverrides::MergeFrom(const TaskNodeOverrides& from) { if (from.has_resources()) { mutable_resources()->::flyteidl::core::Resources::MergeFrom(from.resources()); } + if (from.has_extended_resources()) { + mutable_extended_resources()->::flyteidl::core::ExtendedResources::MergeFrom(from.extended_resources()); + } } void TaskNodeOverrides::CopyFrom(const ::google::protobuf::Message& from) { @@ -8386,6 +8463,7 @@ void TaskNodeOverrides::InternalSwap(TaskNodeOverrides* other) { using std::swap; _internal_metadata_.Swap(&other->_internal_metadata_); swap(resources_, other->resources_); + swap(extended_resources_, other->extended_resources_); } ::google::protobuf::Metadata TaskNodeOverrides::GetMetadata() const { diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.h b/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.h index 9ee5eb92c8a..390fd2e31f4 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.h +++ b/flyteidl/gen/pb-cpp/flyteidl/core/workflow.pb.h @@ -2691,12 +2691,22 @@ class TaskNodeOverrides final : ::flyteidl::core::Resources* mutable_resources(); void set_allocated_resources(::flyteidl::core::Resources* resources); + // .flyteidl.core.ExtendedResources extended_resources = 2; + bool has_extended_resources() const; + void clear_extended_resources(); + static const int kExtendedResourcesFieldNumber = 2; + const ::flyteidl::core::ExtendedResources& extended_resources() const; + ::flyteidl::core::ExtendedResources* release_extended_resources(); + ::flyteidl::core::ExtendedResources* mutable_extended_resources(); + void set_allocated_extended_resources(::flyteidl::core::ExtendedResources* extended_resources); + // @@protoc_insertion_point(class_scope:flyteidl.core.TaskNodeOverrides) private: class HasBitSetters; ::google::protobuf::internal::InternalMetadataWithArena _internal_metadata_; ::flyteidl::core::Resources* resources_; + ::flyteidl::core::ExtendedResources* extended_resources_; mutable ::google::protobuf::internal::CachedSize _cached_size_; friend struct ::TableStruct_flyteidl_2fcore_2fworkflow_2eproto; }; @@ -4946,6 +4956,51 @@ inline void TaskNodeOverrides::set_allocated_resources(::flyteidl::core::Resourc // @@protoc_insertion_point(field_set_allocated:flyteidl.core.TaskNodeOverrides.resources) } +// .flyteidl.core.ExtendedResources extended_resources = 2; +inline bool TaskNodeOverrides::has_extended_resources() const { + return this != internal_default_instance() && extended_resources_ != nullptr; +} +inline const ::flyteidl::core::ExtendedResources& TaskNodeOverrides::extended_resources() const { + const ::flyteidl::core::ExtendedResources* p = extended_resources_; + // @@protoc_insertion_point(field_get:flyteidl.core.TaskNodeOverrides.extended_resources) + return p != nullptr ? *p : *reinterpret_cast( + &::flyteidl::core::_ExtendedResources_default_instance_); +} +inline ::flyteidl::core::ExtendedResources* TaskNodeOverrides::release_extended_resources() { + // @@protoc_insertion_point(field_release:flyteidl.core.TaskNodeOverrides.extended_resources) + + ::flyteidl::core::ExtendedResources* temp = extended_resources_; + extended_resources_ = nullptr; + return temp; +} +inline ::flyteidl::core::ExtendedResources* TaskNodeOverrides::mutable_extended_resources() { + + if (extended_resources_ == nullptr) { + auto* p = CreateMaybeMessage<::flyteidl::core::ExtendedResources>(GetArenaNoVirtual()); + extended_resources_ = p; + } + // @@protoc_insertion_point(field_mutable:flyteidl.core.TaskNodeOverrides.extended_resources) + return extended_resources_; +} +inline void TaskNodeOverrides::set_allocated_extended_resources(::flyteidl::core::ExtendedResources* extended_resources) { + ::google::protobuf::Arena* message_arena = GetArenaNoVirtual(); + if (message_arena == nullptr) { + delete reinterpret_cast< ::google::protobuf::MessageLite*>(extended_resources_); + } + if (extended_resources) { + ::google::protobuf::Arena* submessage_arena = nullptr; + if (message_arena != submessage_arena) { + extended_resources = ::google::protobuf::internal::GetOwnedMessage( + message_arena, extended_resources, submessage_arena); + } + + } else { + + } + extended_resources_ = extended_resources; + // @@protoc_insertion_point(field_set_allocated:flyteidl.core.TaskNodeOverrides.extended_resources) +} + #ifdef __GNUC__ #pragma GCC diagnostic pop #endif // __GNUC__ diff --git a/flyteidl/gen/pb-cpp/flyteidl/core/workflow_closure.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/core/workflow_closure.pb.cc index 71d86496327..d515bf74505 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/core/workflow_closure.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/core/workflow_closure.pb.cc @@ -16,7 +16,7 @@ // @@protoc_insertion_point(includes) #include -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fworkflow_2eproto ::google::protobuf::internal::SCCInfo<6> scc_info_WorkflowTemplate_flyteidl_2fcore_2fworkflow_2eproto; namespace flyteidl { namespace core { diff --git a/flyteidl/gen/pb-cpp/flyteidl/service/external_plugin_service.pb.cc b/flyteidl/gen/pb-cpp/flyteidl/service/external_plugin_service.pb.cc index 5dee1180386..e73d9c9a21c 100644 --- a/flyteidl/gen/pb-cpp/flyteidl/service/external_plugin_service.pb.cc +++ b/flyteidl/gen/pb-cpp/flyteidl/service/external_plugin_service.pb.cc @@ -17,7 +17,7 @@ #include extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2fliterals_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_Literal_flyteidl_2fcore_2fliterals_2eproto; -extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<9> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; +extern PROTOBUF_INTERNAL_EXPORT_flyteidl_2fcore_2ftasks_2eproto ::google::protobuf::internal::SCCInfo<10> scc_info_TaskTemplate_flyteidl_2fcore_2ftasks_2eproto; namespace flyteidl { namespace service { class TaskCreateRequestDefaultTypeInternal { diff --git a/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.go b/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.go index 8a67fb2670e..d7310cce314 100644 --- a/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.go +++ b/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.go @@ -83,7 +83,7 @@ func (x RuntimeMetadata_RuntimeType) String() string { } func (RuntimeMetadata_RuntimeType) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{1, 0} + return fileDescriptor_bd8423ba58d6ed80, []int{3, 0} } // Architecture-type the container image supports. @@ -118,7 +118,7 @@ func (x Container_Architecture) String() string { } func (Container_Architecture) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{5, 0} + return fileDescriptor_bd8423ba58d6ed80, []int{7, 0} } // Mode to use for downloading @@ -150,7 +150,7 @@ func (x IOStrategy_DownloadMode) String() string { } func (IOStrategy_DownloadMode) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{6, 0} + return fileDescriptor_bd8423ba58d6ed80, []int{8, 0} } // Mode to use for uploading @@ -182,7 +182,7 @@ func (x IOStrategy_UploadMode) String() string { } func (IOStrategy_UploadMode) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{6, 1} + return fileDescriptor_bd8423ba58d6ed80, []int{8, 1} } // LiteralMapFormat decides the encoding format in which the input metadata should be made available to the containers. @@ -216,7 +216,7 @@ func (x DataLoadingConfig_LiteralMapFormat) String() string { } func (DataLoadingConfig_LiteralMapFormat) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{7, 0} + return fileDescriptor_bd8423ba58d6ed80, []int{9, 0} } // The dialect of the SQL statement. This is used to validate and parse SQL statements at compilation time to avoid @@ -250,7 +250,7 @@ func (x Sql_Dialect) String() string { } func (Sql_Dialect) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{10, 0} + return fileDescriptor_bd8423ba58d6ed80, []int{12, 0} } // A customizable interface to convey resources requested for a container. This can be interpreted differently for different @@ -356,6 +356,143 @@ func (m *Resources_ResourceEntry) GetValue() string { return "" } +// Metadata associated with the GPU accelerator to allocate to a task. Contains +// information about device type, and for multi-instance GPUs, the partition size to +// use. +type GPUAccelerator struct { + // This can be any arbitrary string, and should be informed by the labels or taints + // associated with the nodes in question. Default cloud provider labels typically + // use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. + Device string `protobuf:"bytes,1,opt,name=device,proto3" json:"device,omitempty"` + // Types that are valid to be assigned to PartitionSizeValue: + // *GPUAccelerator_Unpartitioned + // *GPUAccelerator_PartitionSize + PartitionSizeValue isGPUAccelerator_PartitionSizeValue `protobuf_oneof:"partition_size_value"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` +} + +func (m *GPUAccelerator) Reset() { *m = GPUAccelerator{} } +func (m *GPUAccelerator) String() string { return proto.CompactTextString(m) } +func (*GPUAccelerator) ProtoMessage() {} +func (*GPUAccelerator) Descriptor() ([]byte, []int) { + return fileDescriptor_bd8423ba58d6ed80, []int{1} +} + +func (m *GPUAccelerator) XXX_Unmarshal(b []byte) error { + return xxx_messageInfo_GPUAccelerator.Unmarshal(m, b) +} +func (m *GPUAccelerator) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + return xxx_messageInfo_GPUAccelerator.Marshal(b, m, deterministic) +} +func (m *GPUAccelerator) XXX_Merge(src proto.Message) { + xxx_messageInfo_GPUAccelerator.Merge(m, src) +} +func (m *GPUAccelerator) XXX_Size() int { + return xxx_messageInfo_GPUAccelerator.Size(m) +} +func (m *GPUAccelerator) XXX_DiscardUnknown() { + xxx_messageInfo_GPUAccelerator.DiscardUnknown(m) +} + +var xxx_messageInfo_GPUAccelerator proto.InternalMessageInfo + +func (m *GPUAccelerator) GetDevice() string { + if m != nil { + return m.Device + } + return "" +} + +type isGPUAccelerator_PartitionSizeValue interface { + isGPUAccelerator_PartitionSizeValue() +} + +type GPUAccelerator_Unpartitioned struct { + Unpartitioned bool `protobuf:"varint,2,opt,name=unpartitioned,proto3,oneof"` +} + +type GPUAccelerator_PartitionSize struct { + PartitionSize string `protobuf:"bytes,3,opt,name=partition_size,json=partitionSize,proto3,oneof"` +} + +func (*GPUAccelerator_Unpartitioned) isGPUAccelerator_PartitionSizeValue() {} + +func (*GPUAccelerator_PartitionSize) isGPUAccelerator_PartitionSizeValue() {} + +func (m *GPUAccelerator) GetPartitionSizeValue() isGPUAccelerator_PartitionSizeValue { + if m != nil { + return m.PartitionSizeValue + } + return nil +} + +func (m *GPUAccelerator) GetUnpartitioned() bool { + if x, ok := m.GetPartitionSizeValue().(*GPUAccelerator_Unpartitioned); ok { + return x.Unpartitioned + } + return false +} + +func (m *GPUAccelerator) GetPartitionSize() string { + if x, ok := m.GetPartitionSizeValue().(*GPUAccelerator_PartitionSize); ok { + return x.PartitionSize + } + return "" +} + +// XXX_OneofWrappers is for the internal use of the proto package. +func (*GPUAccelerator) XXX_OneofWrappers() []interface{} { + return []interface{}{ + (*GPUAccelerator_Unpartitioned)(nil), + (*GPUAccelerator_PartitionSize)(nil), + } +} + +// Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to +// allocate to a task. +type ExtendedResources struct { + // GPU accelerator to select for task. Contains information about device type, and + // for multi-instance GPUs, the partition size to use. + GpuAccelerator *GPUAccelerator `protobuf:"bytes,1,opt,name=gpu_accelerator,json=gpuAccelerator,proto3" json:"gpu_accelerator,omitempty"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` +} + +func (m *ExtendedResources) Reset() { *m = ExtendedResources{} } +func (m *ExtendedResources) String() string { return proto.CompactTextString(m) } +func (*ExtendedResources) ProtoMessage() {} +func (*ExtendedResources) Descriptor() ([]byte, []int) { + return fileDescriptor_bd8423ba58d6ed80, []int{2} +} + +func (m *ExtendedResources) XXX_Unmarshal(b []byte) error { + return xxx_messageInfo_ExtendedResources.Unmarshal(m, b) +} +func (m *ExtendedResources) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + return xxx_messageInfo_ExtendedResources.Marshal(b, m, deterministic) +} +func (m *ExtendedResources) XXX_Merge(src proto.Message) { + xxx_messageInfo_ExtendedResources.Merge(m, src) +} +func (m *ExtendedResources) XXX_Size() int { + return xxx_messageInfo_ExtendedResources.Size(m) +} +func (m *ExtendedResources) XXX_DiscardUnknown() { + xxx_messageInfo_ExtendedResources.DiscardUnknown(m) +} + +var xxx_messageInfo_ExtendedResources proto.InternalMessageInfo + +func (m *ExtendedResources) GetGpuAccelerator() *GPUAccelerator { + if m != nil { + return m.GpuAccelerator + } + return nil +} + // Runtime information. This is loosely defined to allow for extensibility. type RuntimeMetadata struct { // Type of runtime. @@ -374,7 +511,7 @@ func (m *RuntimeMetadata) Reset() { *m = RuntimeMetadata{} } func (m *RuntimeMetadata) String() string { return proto.CompactTextString(m) } func (*RuntimeMetadata) ProtoMessage() {} func (*RuntimeMetadata) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{1} + return fileDescriptor_bd8423ba58d6ed80, []int{3} } func (m *RuntimeMetadata) XXX_Unmarshal(b []byte) error { @@ -455,7 +592,7 @@ func (m *TaskMetadata) Reset() { *m = TaskMetadata{} } func (m *TaskMetadata) String() string { return proto.CompactTextString(m) } func (*TaskMetadata) ProtoMessage() {} func (*TaskMetadata) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{2} + return fileDescriptor_bd8423ba58d6ed80, []int{4} } func (m *TaskMetadata) XXX_Unmarshal(b []byte) error { @@ -606,6 +743,9 @@ type TaskTemplate struct { TaskTypeVersion int32 `protobuf:"varint,7,opt,name=task_type_version,json=taskTypeVersion,proto3" json:"task_type_version,omitempty"` // security_context encapsulates security attributes requested to run this task. SecurityContext *SecurityContext `protobuf:"bytes,8,opt,name=security_context,json=securityContext,proto3" json:"security_context,omitempty"` + // Encapsulates all non-standard resources, not captured by + // v1.ResourceRequirements, to allocate to a task. + ExtendedResources *ExtendedResources `protobuf:"bytes,9,opt,name=extended_resources,json=extendedResources,proto3" json:"extended_resources,omitempty"` // Metadata about the custom defined for this task. This is extensible to allow various plugins in the system // to use as required. // reserve the field numbers 1 through 15 for very frequently occurring message elements @@ -619,7 +759,7 @@ func (m *TaskTemplate) Reset() { *m = TaskTemplate{} } func (m *TaskTemplate) String() string { return proto.CompactTextString(m) } func (*TaskTemplate) ProtoMessage() {} func (*TaskTemplate) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{3} + return fileDescriptor_bd8423ba58d6ed80, []int{5} } func (m *TaskTemplate) XXX_Unmarshal(b []byte) error { @@ -739,6 +879,13 @@ func (m *TaskTemplate) GetSecurityContext() *SecurityContext { return nil } +func (m *TaskTemplate) GetExtendedResources() *ExtendedResources { + if m != nil { + return m.ExtendedResources + } + return nil +} + func (m *TaskTemplate) GetConfig() map[string]string { if m != nil { return m.Config @@ -769,7 +916,7 @@ func (m *ContainerPort) Reset() { *m = ContainerPort{} } func (m *ContainerPort) String() string { return proto.CompactTextString(m) } func (*ContainerPort) ProtoMessage() {} func (*ContainerPort) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{4} + return fileDescriptor_bd8423ba58d6ed80, []int{6} } func (m *ContainerPort) XXX_Unmarshal(b []byte) error { @@ -836,7 +983,7 @@ func (m *Container) Reset() { *m = Container{} } func (m *Container) String() string { return proto.CompactTextString(m) } func (*Container) ProtoMessage() {} func (*Container) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{5} + return fileDescriptor_bd8423ba58d6ed80, []int{7} } func (m *Container) XXX_Unmarshal(b []byte) error { @@ -936,7 +1083,7 @@ func (m *IOStrategy) Reset() { *m = IOStrategy{} } func (m *IOStrategy) String() string { return proto.CompactTextString(m) } func (*IOStrategy) ProtoMessage() {} func (*IOStrategy) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{6} + return fileDescriptor_bd8423ba58d6ed80, []int{8} } func (m *IOStrategy) XXX_Unmarshal(b []byte) error { @@ -1000,7 +1147,7 @@ func (m *DataLoadingConfig) Reset() { *m = DataLoadingConfig{} } func (m *DataLoadingConfig) String() string { return proto.CompactTextString(m) } func (*DataLoadingConfig) ProtoMessage() {} func (*DataLoadingConfig) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{7} + return fileDescriptor_bd8423ba58d6ed80, []int{9} } func (m *DataLoadingConfig) XXX_Unmarshal(b []byte) error { @@ -1082,7 +1229,7 @@ func (m *K8SPod) Reset() { *m = K8SPod{} } func (m *K8SPod) String() string { return proto.CompactTextString(m) } func (*K8SPod) ProtoMessage() {} func (*K8SPod) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{8} + return fileDescriptor_bd8423ba58d6ed80, []int{10} } func (m *K8SPod) XXX_Unmarshal(b []byte) error { @@ -1139,7 +1286,7 @@ func (m *K8SObjectMetadata) Reset() { *m = K8SObjectMetadata{} } func (m *K8SObjectMetadata) String() string { return proto.CompactTextString(m) } func (*K8SObjectMetadata) ProtoMessage() {} func (*K8SObjectMetadata) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{9} + return fileDescriptor_bd8423ba58d6ed80, []int{11} } func (m *K8SObjectMetadata) XXX_Unmarshal(b []byte) error { @@ -1196,7 +1343,7 @@ func (m *Sql) Reset() { *m = Sql{} } func (m *Sql) String() string { return proto.CompactTextString(m) } func (*Sql) ProtoMessage() {} func (*Sql) Descriptor() ([]byte, []int) { - return fileDescriptor_bd8423ba58d6ed80, []int{10} + return fileDescriptor_bd8423ba58d6ed80, []int{12} } func (m *Sql) XXX_Unmarshal(b []byte) error { @@ -1241,6 +1388,8 @@ func init() { proto.RegisterEnum("flyteidl.core.Sql_Dialect", Sql_Dialect_name, Sql_Dialect_value) proto.RegisterType((*Resources)(nil), "flyteidl.core.Resources") proto.RegisterType((*Resources_ResourceEntry)(nil), "flyteidl.core.Resources.ResourceEntry") + proto.RegisterType((*GPUAccelerator)(nil), "flyteidl.core.GPUAccelerator") + proto.RegisterType((*ExtendedResources)(nil), "flyteidl.core.ExtendedResources") proto.RegisterType((*RuntimeMetadata)(nil), "flyteidl.core.RuntimeMetadata") proto.RegisterType((*TaskMetadata)(nil), "flyteidl.core.TaskMetadata") proto.RegisterMapType((map[string]string)(nil), "flyteidl.core.TaskMetadata.TagsEntry") @@ -1260,109 +1409,117 @@ func init() { func init() { proto.RegisterFile("flyteidl/core/tasks.proto", fileDescriptor_bd8423ba58d6ed80) } var fileDescriptor_bd8423ba58d6ed80 = []byte{ - // 1664 bytes of a gzipped FileDescriptorProto + // 1787 bytes of a gzipped FileDescriptorProto 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x94, 0x57, 0xdd, 0x72, 0xdb, 0xb8, - 0x15, 0xb6, 0x24, 0x5b, 0xb2, 0x8e, 0xa4, 0x98, 0xc6, 0x36, 0x5b, 0xc6, 0x4d, 0xd2, 0x0c, 0xa7, - 0x9b, 0x4d, 0xb7, 0x8d, 0xd4, 0x78, 0x33, 0xb6, 0x37, 0xdd, 0x49, 0x47, 0x89, 0x98, 0x58, 0xb5, - 0xf5, 0x33, 0x90, 0x9c, 0x6d, 0x3a, 0xd3, 0x61, 0x61, 0x12, 0x96, 0x59, 0x53, 0x04, 0x03, 0x82, - 0x6e, 0xd5, 0xfb, 0xbe, 0x40, 0x5f, 0xa2, 0xd3, 0x99, 0xde, 0xf6, 0xb2, 0xef, 0xd0, 0x97, 0xe9, - 0x5d, 0x2f, 0x76, 0x00, 0x90, 0xd4, 0x8f, 0xed, 0xf5, 0xf8, 0x4a, 0xc0, 0x39, 0xdf, 0x77, 0x40, - 0x1c, 0x7d, 0x07, 0x07, 0x80, 0x07, 0x67, 0xc1, 0x4c, 0x50, 0xdf, 0x0b, 0x5a, 0x2e, 0xe3, 0xb4, - 0x25, 0x48, 0x7c, 0x11, 0x37, 0x23, 0xce, 0x04, 0x43, 0x8d, 0xcc, 0xd5, 0x94, 0xae, 0x9d, 0xc7, - 0xcb, 0x48, 0xdf, 0xa3, 0xa1, 0xf0, 0xcf, 0x7c, 0xca, 0x35, 0x7c, 0xe7, 0xd1, 0x8a, 0x3f, 0x14, - 0x94, 0x9f, 0x11, 0x97, 0xa6, 0xee, 0x87, 0xcb, 0xee, 0xc0, 0x17, 0x94, 0x93, 0x20, 0xbe, 0xde, - 0x1b, 0x53, 0x37, 0xe1, 0xbe, 0x98, 0xa5, 0xde, 0xc7, 0x13, 0xc6, 0x26, 0x01, 0x6d, 0xa9, 0xd9, - 0x69, 0x72, 0xd6, 0xf2, 0x12, 0x4e, 0x84, 0xcf, 0xc2, 0x8c, 0xbd, 0xea, 0x8f, 0x05, 0x4f, 0x5c, - 0xa1, 0xbd, 0xd6, 0x7f, 0x8b, 0x50, 0xc5, 0x34, 0x66, 0x09, 0x77, 0x69, 0x8c, 0xde, 0xc0, 0x26, - 0xa7, 0x9f, 0x12, 0x1a, 0x8b, 0xd8, 0x2c, 0x3c, 0x29, 0x3d, 0xab, 0xed, 0x3e, 0x6d, 0x2e, 0x6d, - 0xb4, 0x99, 0x63, 0xf3, 0x91, 0x1d, 0x0a, 0x3e, 0xc3, 0x39, 0x0f, 0xbd, 0x86, 0x72, 0xe0, 0x4f, - 0x7d, 0x11, 0x9b, 0xc5, 0x3b, 0x45, 0x48, 0x59, 0x3b, 0x7f, 0x84, 0xc6, 0x92, 0x03, 0x7d, 0x03, - 0xeb, 0x21, 0x99, 0x52, 0xb3, 0xf0, 0xa4, 0xf0, 0xec, 0xde, 0xee, 0x17, 0xb7, 0x86, 0xeb, 0x93, - 0x29, 0xc5, 0x8a, 0x82, 0x7e, 0x04, 0x1b, 0x97, 0x24, 0x48, 0xa8, 0x59, 0x7c, 0x52, 0x78, 0x56, - 0xc5, 0x7a, 0x62, 0xfd, 0x01, 0xea, 0x8b, 0x58, 0x54, 0x83, 0xca, 0x49, 0xff, 0xa8, 0x3f, 0xf8, - 0xae, 0x6f, 0xac, 0xa1, 0x0a, 0x94, 0xde, 0x0e, 0x4f, 0x8c, 0x82, 0x1c, 0xbc, 0x1f, 0x9e, 0x18, - 0x45, 0x04, 0x50, 0xee, 0xd9, 0xbd, 0x01, 0xfe, 0x68, 0x94, 0x24, 0x74, 0x34, 0x1e, 0xe0, 0xf6, - 0x7b, 0xdb, 0x58, 0x47, 0xf7, 0x61, 0xdb, 0x1e, 0x1e, 0xda, 0x3d, 0x1b, 0xb7, 0x8f, 0x9d, 0xcc, - 0xbc, 0x61, 0xfd, 0xab, 0x00, 0x5b, 0x38, 0x09, 0x85, 0x3f, 0xa5, 0x3d, 0x2a, 0x88, 0x47, 0x04, - 0x41, 0xaf, 0x61, 0x5d, 0xcc, 0xa2, 0x6c, 0x0f, 0x5f, 0xad, 0xee, 0x61, 0x19, 0x9d, 0xcd, 0xc7, - 0xb3, 0x88, 0x62, 0xc5, 0x43, 0x26, 0x54, 0x2e, 0x29, 0x8f, 0x7d, 0x16, 0xa6, 0x5b, 0xc9, 0xa6, - 0xe8, 0x73, 0x28, 0x9f, 0x05, 0xe4, 0x92, 0x71, 0xb3, 0xa4, 0x1c, 0xe9, 0xcc, 0xfa, 0x12, 0x6a, - 0x0b, 0x61, 0x50, 0x15, 0x36, 0x06, 0xe3, 0x43, 0x1b, 0x1b, 0x6b, 0xa8, 0x01, 0xd5, 0x77, 0xc7, - 0x1f, 0xc7, 0xb6, 0x33, 0xea, 0x1c, 0x19, 0x05, 0xeb, 0x7f, 0xeb, 0x50, 0x1f, 0x93, 0xf8, 0x22, - 0xff, 0x56, 0x0b, 0xea, 0x9e, 0x1f, 0xbb, 0xec, 0x92, 0x72, 0x72, 0x1a, 0xe8, 0x6f, 0xde, 0xc4, - 0x4b, 0x36, 0x74, 0x00, 0x15, 0xae, 0xa3, 0xab, 0xef, 0xa9, 0xed, 0x3e, 0xfe, 0xe1, 0x2d, 0xe1, - 0x0c, 0x8e, 0xbe, 0x86, 0x8a, 0xfc, 0x65, 0x89, 0x30, 0xd7, 0x15, 0xf3, 0x41, 0x53, 0x0b, 0xb4, - 0x99, 0x09, 0xb4, 0xd9, 0x49, 0x05, 0x8c, 0x33, 0x24, 0xda, 0x83, 0x0a, 0xa7, 0x82, 0xfb, 0x34, - 0x36, 0x37, 0x14, 0xe9, 0xe1, 0x15, 0x15, 0x08, 0x3e, 0x1b, 0x09, 0x4e, 0x04, 0x9d, 0xcc, 0x70, - 0x06, 0x46, 0xbf, 0x80, 0xed, 0xec, 0xb3, 0x67, 0x4e, 0x96, 0xc0, 0xb2, 0xca, 0x93, 0x91, 0x3b, - 0x3e, 0xa4, 0x99, 0x3c, 0x00, 0xd3, 0xa3, 0x11, 0xa7, 0x2e, 0x11, 0xd4, 0x73, 0x28, 0xe7, 0x8c, - 0x3b, 0x53, 0x1a, 0xc7, 0x64, 0x42, 0xcd, 0x8a, 0xe2, 0x7c, 0x3e, 0xf7, 0xdb, 0xd2, 0xdd, 0xd3, - 0x5e, 0xf4, 0x14, 0x1a, 0xaa, 0xa2, 0x79, 0x12, 0x09, 0x5f, 0xa6, 0x6c, 0x53, 0xa6, 0xec, 0x70, - 0x0d, 0x2f, 0x9b, 0xd1, 0x73, 0x40, 0x2e, 0x71, 0xcf, 0xa9, 0x13, 0x53, 0xee, 0x93, 0xc0, 0xff, - 0xab, 0xca, 0x6f, 0x55, 0xe5, 0x77, 0x5b, 0x79, 0x46, 0x0b, 0x0e, 0xf4, 0x05, 0xdc, 0x9b, 0xd0, - 0x90, 0xca, 0x4d, 0xc5, 0x8e, 0x47, 0xdd, 0x0b, 0x13, 0x14, 0xb4, 0x91, 0x5b, 0x3b, 0xd4, 0xbd, - 0x90, 0xf5, 0x21, 0xc8, 0x24, 0x36, 0x6b, 0xaa, 0xdc, 0x56, 0xeb, 0x63, 0xf1, 0xaf, 0x6d, 0x8e, - 0xc9, 0x24, 0xd6, 0xd5, 0xa6, 0x28, 0xe8, 0x2b, 0xd8, 0x8e, 0x98, 0xe7, 0x08, 0x3a, 0x8d, 0x02, - 0x22, 0xa8, 0xa3, 0xea, 0xac, 0xae, 0xf6, 0xba, 0x15, 0x31, 0x6f, 0x9c, 0xda, 0x65, 0x95, 0xec, - 0xec, 0x43, 0x35, 0xa7, 0x23, 0x03, 0x4a, 0x17, 0x74, 0xa6, 0xa4, 0x51, 0xc5, 0x72, 0x78, 0x7d, - 0xa9, 0xbd, 0x2a, 0x1e, 0x14, 0xde, 0xdc, 0x87, 0xcf, 0x96, 0xd2, 0xe0, 0xe8, 0x2a, 0xfc, 0xdb, - 0x86, 0xd6, 0x5d, 0xb6, 0x08, 0xfa, 0x39, 0x14, 0x7d, 0x4f, 0x85, 0x94, 0xa2, 0x58, 0xde, 0x45, - 0x37, 0x3f, 0x50, 0x71, 0xd1, 0xf7, 0x10, 0x4a, 0xcb, 0x49, 0xaf, 0xa5, 0x4b, 0x64, 0x1f, 0x36, - 0xa7, 0xe9, 0x3e, 0x55, 0x29, 0xd4, 0x76, 0x7f, 0xf2, 0x03, 0xa9, 0xc0, 0x39, 0x18, 0xfd, 0x1a, - 0xaa, 0xf9, 0x79, 0x9c, 0x6a, 0xf2, 0xd1, 0x2a, 0x73, 0x16, 0x51, 0xaf, 0x9b, 0x81, 0xf0, 0x1c, - 0x8f, 0x5a, 0x50, 0x76, 0x93, 0x58, 0xb0, 0x69, 0x2a, 0xcc, 0x1f, 0x5f, 0x51, 0xf3, 0x48, 0x1d, - 0xb7, 0x38, 0x85, 0xa1, 0x03, 0xa8, 0xba, 0x2c, 0x14, 0xc4, 0x0f, 0x29, 0x57, 0x52, 0xac, 0xed, - 0x9a, 0x2b, 0xab, 0xbd, 0xcd, 0xfc, 0x87, 0x6b, 0x78, 0x0e, 0x46, 0xbf, 0x82, 0xca, 0xc5, 0x41, - 0xec, 0x44, 0xcc, 0x33, 0xb7, 0x15, 0xef, 0xfe, 0x0a, 0xef, 0xe8, 0x20, 0x1e, 0x32, 0xef, 0x70, - 0x0d, 0x97, 0x2f, 0xd4, 0x08, 0x3d, 0x85, 0x52, 0xfc, 0x29, 0x30, 0x91, 0x42, 0xa3, 0x15, 0xf4, - 0xe8, 0x53, 0x70, 0xb8, 0x86, 0x25, 0x40, 0xca, 0x40, 0xf6, 0x36, 0x47, 0xe6, 0x31, 0x2f, 0x13, - 0x29, 0xf9, 0x0d, 0xbc, 0x25, 0x1d, 0x72, 0xfb, 0x59, 0x95, 0x74, 0xc1, 0xc8, 0x1a, 0x90, 0x23, - 0xbf, 0x8d, 0xfe, 0x45, 0x28, 0xb9, 0x5f, 0x3d, 0x02, 0x46, 0x29, 0xec, 0xad, 0x46, 0xe1, 0xad, - 0x78, 0xd9, 0x80, 0x7e, 0x03, 0x65, 0x97, 0x85, 0x67, 0xfe, 0xc4, 0x34, 0x94, 0x74, 0xbf, 0xbc, - 0xe6, 0xff, 0xca, 0xd4, 0x21, 0x93, 0x72, 0xe6, 0x4f, 0xd2, 0x56, 0xa1, 0x69, 0x3b, 0xdf, 0x40, - 0x6d, 0xc1, 0x7c, 0x27, 0x51, 0x6e, 0x42, 0x59, 0x10, 0x3e, 0xa1, 0xc2, 0xda, 0x83, 0x46, 0x9e, - 0xf0, 0x21, 0xe3, 0x42, 0x96, 0x5d, 0x9e, 0x74, 0x27, 0x62, 0x5c, 0xa8, 0x88, 0x0d, 0xdc, 0x70, - 0x17, 0x61, 0xd6, 0xff, 0x4b, 0x50, 0xcd, 0x89, 0x72, 0x25, 0x7f, 0x2a, 0x4f, 0x0a, 0xbd, 0xba, - 0x9e, 0xc8, 0x63, 0xdb, 0x65, 0xd3, 0x29, 0x09, 0x3d, 0xd5, 0x0c, 0xab, 0x38, 0x9b, 0x4a, 0x05, - 0x13, 0x3e, 0x89, 0xcd, 0x92, 0x32, 0xab, 0x31, 0xda, 0x83, 0x2a, 0xcf, 0xba, 0x59, 0x2a, 0x44, - 0xf3, 0xa6, 0x6e, 0x87, 0xe7, 0x50, 0xf4, 0x1c, 0x4a, 0x34, 0xbc, 0x34, 0x37, 0x54, 0x12, 0x57, - 0x45, 0x7f, 0x44, 0x67, 0x1f, 0xe4, 0xae, 0x87, 0xc4, 0xe7, 0x58, 0xe2, 0xd0, 0x7e, 0x9e, 0xf6, - 0xf2, 0xad, 0x8c, 0x37, 0x45, 0xb3, 0x90, 0xa5, 0x1b, 0xed, 0xc2, 0x86, 0x4c, 0x47, 0x6c, 0x56, - 0x14, 0xef, 0xe1, 0x4d, 0xb2, 0x95, 0xe9, 0xc1, 0x1a, 0x8a, 0xda, 0x50, 0x93, 0x45, 0xe6, 0xa4, - 0x2b, 0x56, 0xd5, 0xae, 0x9e, 0xac, 0x30, 0x3b, 0x44, 0x90, 0x63, 0x46, 0x3c, 0x3f, 0x9c, 0xe8, - 0xff, 0x13, 0x83, 0x24, 0xe9, 0x31, 0xea, 0x42, 0x9d, 0x70, 0xf7, 0xdc, 0x17, 0xd4, 0x15, 0x09, - 0xa7, 0xea, 0x10, 0xbc, 0x7a, 0x0f, 0xc8, 0x57, 0x6f, 0xb6, 0x17, 0xc0, 0x78, 0x89, 0x6a, 0x75, - 0xa1, 0xbe, 0xe8, 0x5d, 0xee, 0xfc, 0x55, 0xd8, 0x68, 0xf7, 0x3a, 0x7b, 0x2f, 0x8d, 0x82, 0x1a, - 0xe2, 0xde, 0xde, 0x4b, 0xdd, 0xfd, 0xdb, 0xb8, 0xe7, 0x7c, 0xd8, 0x33, 0x4a, 0xf9, 0x78, 0xdf, - 0x58, 0xb7, 0xfe, 0x5d, 0x04, 0xe8, 0x0e, 0xb2, 0x96, 0x83, 0x8e, 0xa0, 0xe1, 0xb1, 0x3f, 0x87, - 0x01, 0x23, 0x9e, 0x33, 0x65, 0x5e, 0xd6, 0xe9, 0x57, 0x2f, 0x3f, 0x73, 0x46, 0xb3, 0x93, 0xc2, - 0x7b, 0xcc, 0xa3, 0xb8, 0xee, 0x2d, 0xcc, 0x90, 0x0d, 0xb5, 0x24, 0x9a, 0x87, 0x2a, 0xaa, 0x50, - 0x3f, 0xbb, 0x39, 0xd4, 0x49, 0x94, 0x07, 0x82, 0x24, 0x1f, 0x5b, 0xc7, 0x50, 0x5f, 0x5c, 0x04, - 0x21, 0xb8, 0xd7, 0x19, 0x7c, 0xd7, 0x3f, 0x1e, 0xb4, 0x3b, 0x8e, 0xdd, 0x7e, 0xaf, 0x2e, 0x03, - 0x9f, 0xc1, 0x56, 0x6e, 0x1b, 0x8d, 0xb1, 0xdd, 0xee, 0x19, 0x05, 0x6d, 0x74, 0xfa, 0x83, 0xb1, - 0x93, 0xf9, 0x8c, 0xa2, 0x65, 0x03, 0xcc, 0xd7, 0x91, 0xb1, 0x4e, 0x86, 0x8a, 0x35, 0xe8, 0x3b, - 0xf6, 0xef, 0xba, 0x63, 0x63, 0x0d, 0x19, 0x50, 0x4f, 0x6d, 0x3a, 0x7a, 0x01, 0x6d, 0x43, 0x23, - 0x0d, 0xa4, 0x1d, 0x46, 0xd1, 0xfa, 0x47, 0x11, 0xb6, 0xaf, 0xfc, 0xdf, 0xb2, 0x50, 0x68, 0x28, - 0x9b, 0x9e, 0x97, 0x5e, 0x37, 0xb2, 0x29, 0x7a, 0x04, 0xe0, 0x87, 0x51, 0x22, 0x9c, 0x88, 0x88, - 0xf3, 0xb4, 0x8e, 0xab, 0xca, 0x32, 0x24, 0xe2, 0x1c, 0xfd, 0x14, 0x6a, 0x2c, 0x11, 0xb9, 0x5f, - 0xdf, 0x81, 0x40, 0x9b, 0x14, 0xa0, 0x0b, 0xe5, 0x33, 0xc6, 0xa7, 0x44, 0x5f, 0x37, 0xee, 0xed, - 0xbe, 0xb8, 0x4d, 0x7b, 0xcd, 0x63, 0x7d, 0xfb, 0xee, 0x91, 0xe8, 0x9d, 0x22, 0xe2, 0x34, 0x00, - 0x7a, 0x05, 0x35, 0x9f, 0x39, 0x71, 0x9a, 0xf5, 0xf4, 0xc0, 0x7f, 0x70, 0xe3, 0xdf, 0x82, 0xc1, - 0x67, 0xd9, 0xd8, 0x7a, 0x01, 0xc6, 0x6a, 0x5c, 0xb4, 0x09, 0xeb, 0xbf, 0x1d, 0x0d, 0xa4, 0xf4, - 0x36, 0x61, 0xfd, 0x63, 0xbb, 0x77, 0xac, 0x95, 0x37, 0xc4, 0x83, 0xf1, 0xc0, 0x28, 0x5a, 0xff, - 0x29, 0x40, 0x59, 0x1f, 0xe9, 0xe8, 0xdb, 0x85, 0xde, 0x56, 0xb8, 0xb6, 0x84, 0x8e, 0x0e, 0xe2, - 0xc1, 0xe9, 0x9f, 0xa8, 0x2b, 0xae, 0x69, 0x70, 0xbb, 0xb0, 0x29, 0xbb, 0x7c, 0x1c, 0x51, 0x37, - 0xbd, 0xad, 0xdd, 0xd8, 0xa5, 0x2a, 0x11, 0xf3, 0x46, 0x11, 0x75, 0x57, 0xeb, 0xb6, 0x74, 0xf7, - 0xba, 0xb5, 0xfe, 0x59, 0x84, 0xed, 0x2b, 0x9f, 0x85, 0x3a, 0x50, 0x0e, 0xc8, 0x29, 0x0d, 0xb2, - 0x07, 0xc6, 0x2f, 0x6f, 0xdb, 0x48, 0xf3, 0x58, 0xc1, 0xb3, 0x47, 0x82, 0x9a, 0xa0, 0x11, 0xd4, - 0x48, 0x18, 0x32, 0xa1, 0xee, 0x89, 0xd9, 0x4b, 0xe3, 0xc5, 0xad, 0xa1, 0xda, 0x73, 0x8e, 0x8e, - 0xb7, 0x18, 0x45, 0xb6, 0x93, 0x85, 0xb5, 0xee, 0xd2, 0x4e, 0x76, 0x5e, 0x83, 0xb1, 0x1a, 0xfb, - 0x2e, 0x7c, 0xeb, 0xef, 0x05, 0x28, 0x8d, 0x3e, 0x05, 0xe8, 0x21, 0x54, 0x63, 0x41, 0x04, 0x9d, - 0xd2, 0x50, 0xa4, 0xcc, 0xb9, 0x01, 0xbd, 0x84, 0x8a, 0xe7, 0x93, 0x80, 0xba, 0x22, 0x3d, 0x13, - 0x76, 0xae, 0xf6, 0xf4, 0x66, 0x47, 0x23, 0x70, 0x06, 0xb5, 0xf6, 0xa1, 0x92, 0xda, 0xe4, 0xd5, - 0xff, 0xa4, 0xdf, 0xb1, 0xdf, 0x75, 0xfb, 0x76, 0x47, 0xcb, 0xae, 0xdd, 0x1f, 0x75, 0x8d, 0x82, - 0x1c, 0x1d, 0x76, 0x3f, 0xd8, 0x46, 0x71, 0xfe, 0x50, 0x28, 0xbd, 0xf9, 0xf6, 0xf7, 0xaf, 0x26, - 0xbe, 0x38, 0x4f, 0x4e, 0x9b, 0x2e, 0x9b, 0xb6, 0xd4, 0x4a, 0x8c, 0x4f, 0xf4, 0xa0, 0x95, 0xbf, - 0x49, 0x27, 0x34, 0x6c, 0x45, 0xa7, 0xcf, 0x27, 0xac, 0xb5, 0xf4, 0x4c, 0x3d, 0x2d, 0x2b, 0x6d, - 0x7d, 0xfd, 0x7d, 0x00, 0x00, 0x00, 0xff, 0xff, 0x32, 0xec, 0xb2, 0x85, 0x45, 0x0f, 0x00, 0x00, + 0x15, 0xb6, 0x24, 0x5b, 0xb2, 0x8e, 0x2c, 0x9b, 0xc2, 0x6e, 0x52, 0xc6, 0x4d, 0xd2, 0x0c, 0xa7, + 0x9b, 0xa4, 0xdb, 0x46, 0x6e, 0xb4, 0x19, 0xdb, 0x9b, 0xee, 0xa4, 0x23, 0x47, 0x4c, 0xac, 0xda, + 0xfa, 0x19, 0x48, 0xce, 0x36, 0xed, 0x74, 0x58, 0x98, 0x84, 0x69, 0xd6, 0x14, 0xc1, 0x80, 0xa0, + 0xbb, 0xca, 0x5b, 0xb4, 0x2f, 0xd1, 0xe9, 0x4c, 0x6f, 0x7b, 0xd9, 0x77, 0xe8, 0x55, 0xdf, 0xa4, + 0x77, 0xbd, 0xe8, 0x00, 0xfc, 0xd1, 0x8f, 0x9d, 0x78, 0x7c, 0x45, 0x9c, 0x73, 0xbe, 0x73, 0x00, + 0x1c, 0x9c, 0x3f, 0xc2, 0xbd, 0x33, 0x7f, 0x2a, 0xa8, 0xe7, 0xf8, 0x3b, 0x36, 0xe3, 0x74, 0x47, + 0x90, 0xe8, 0x22, 0x6a, 0x86, 0x9c, 0x09, 0x86, 0xea, 0x99, 0xa8, 0x29, 0x45, 0xdb, 0x0f, 0x17, + 0x91, 0x9e, 0x43, 0x03, 0xe1, 0x9d, 0x79, 0x94, 0x27, 0xf0, 0xed, 0x07, 0x4b, 0xf2, 0x40, 0x50, + 0x7e, 0x46, 0x6c, 0x9a, 0x8a, 0xef, 0x2f, 0x8a, 0x7d, 0x4f, 0x50, 0x4e, 0xfc, 0xe8, 0x7a, 0x69, + 0x44, 0xed, 0x98, 0x7b, 0x62, 0x9a, 0x4a, 0x1f, 0xba, 0x8c, 0xb9, 0x3e, 0xdd, 0x51, 0xd4, 0x69, + 0x7c, 0xb6, 0xe3, 0xc4, 0x9c, 0x08, 0x8f, 0x05, 0x99, 0xf6, 0xb2, 0x3c, 0x12, 0x3c, 0xb6, 0x45, + 0x22, 0x35, 0xfe, 0x5d, 0x84, 0x2a, 0xa6, 0x11, 0x8b, 0xb9, 0x4d, 0x23, 0x74, 0x00, 0xeb, 0x9c, + 0x7e, 0x88, 0x69, 0x24, 0x22, 0xbd, 0xf0, 0xa8, 0xf4, 0xb4, 0xd6, 0x7a, 0xdc, 0x5c, 0xb8, 0x68, + 0x33, 0xc7, 0xe6, 0x2b, 0x33, 0x10, 0x7c, 0x8a, 0x73, 0x3d, 0xf4, 0x0a, 0xca, 0xbe, 0x37, 0xf1, + 0x44, 0xa4, 0x17, 0x6f, 0x65, 0x21, 0xd5, 0xda, 0xfe, 0x23, 0xd4, 0x17, 0x04, 0xe8, 0x5b, 0x58, + 0x0d, 0xc8, 0x84, 0xea, 0x85, 0x47, 0x85, 0xa7, 0x9b, 0xad, 0xaf, 0x6e, 0x34, 0xd7, 0x27, 0x13, + 0x8a, 0x95, 0x0a, 0xfa, 0x12, 0xd6, 0x2e, 0x89, 0x1f, 0x53, 0xbd, 0xf8, 0xa8, 0xf0, 0xb4, 0x8a, + 0x13, 0xc2, 0xf8, 0x03, 0x6c, 0xcc, 0x63, 0x51, 0x0d, 0x2a, 0x27, 0xfd, 0xa3, 0xfe, 0xe0, 0xfb, + 0xbe, 0xb6, 0x82, 0x2a, 0x50, 0x7a, 0x3d, 0x3c, 0xd1, 0x0a, 0x72, 0xf1, 0x76, 0x78, 0xa2, 0x15, + 0x11, 0x40, 0xb9, 0x67, 0xf6, 0x06, 0xf8, 0xbd, 0x56, 0x92, 0xd0, 0xd1, 0x78, 0x80, 0xdb, 0x6f, + 0x4d, 0x6d, 0x15, 0xdd, 0x81, 0x86, 0x39, 0x3c, 0x34, 0x7b, 0x26, 0x6e, 0x1f, 0x5b, 0x19, 0x7b, + 0xcd, 0xf8, 0x4b, 0x01, 0x36, 0xdf, 0x0e, 0x4f, 0xda, 0xb6, 0x4d, 0x7d, 0xca, 0x89, 0x60, 0x1c, + 0xdd, 0x85, 0xb2, 0x43, 0x2f, 0x3d, 0x3b, 0xb9, 0x44, 0x15, 0xa7, 0x14, 0x7a, 0x0c, 0xf5, 0x38, + 0x08, 0x09, 0x17, 0x9e, 0x7c, 0x30, 0xea, 0xa8, 0x73, 0xae, 0x1f, 0xae, 0xe0, 0x45, 0x36, 0x7a, + 0x02, 0x9b, 0x39, 0x69, 0x45, 0xde, 0x47, 0xaa, 0x97, 0xa4, 0x1d, 0x09, 0xcc, 0xf9, 0x23, 0xef, + 0x23, 0x3d, 0xb8, 0x0b, 0x5f, 0x2e, 0x02, 0xad, 0xe4, 0xca, 0xbf, 0x87, 0x86, 0xf9, 0x83, 0xa0, + 0x81, 0x43, 0x9d, 0xd9, 0x6b, 0xbf, 0x81, 0x2d, 0x37, 0x8c, 0x2d, 0x32, 0x3b, 0xa8, 0x3a, 0x5e, + 0xad, 0xf5, 0x60, 0xc9, 0xc7, 0x8b, 0xb7, 0xc1, 0x9b, 0x6e, 0x18, 0xcf, 0xd1, 0xc6, 0x3f, 0x0a, + 0xb0, 0x85, 0xe3, 0x40, 0x78, 0x13, 0xda, 0xa3, 0x82, 0x38, 0x44, 0x10, 0xf4, 0x0a, 0x56, 0xc5, + 0x34, 0xcc, 0x1e, 0xed, 0xeb, 0xe5, 0x47, 0x5b, 0x44, 0x67, 0xf4, 0x78, 0x1a, 0x52, 0xac, 0xf4, + 0x90, 0x0e, 0x95, 0x4b, 0xca, 0x23, 0x8f, 0x05, 0xe9, 0xdb, 0x65, 0xa4, 0xf4, 0xe5, 0x99, 0x4f, + 0x2e, 0x19, 0x4f, 0x7c, 0x80, 0x53, 0xca, 0x78, 0x02, 0xb5, 0x39, 0x33, 0xa8, 0x0a, 0x6b, 0x83, + 0xf1, 0xa1, 0x89, 0xb5, 0x15, 0x54, 0x87, 0xea, 0x9b, 0xe3, 0xf7, 0x63, 0xd3, 0x1a, 0x75, 0x8e, + 0xb4, 0x82, 0xf1, 0xdf, 0x55, 0xd8, 0x18, 0x93, 0xe8, 0x22, 0x3f, 0xab, 0x01, 0x1b, 0x8e, 0x17, + 0xd9, 0xec, 0x92, 0x72, 0x72, 0xea, 0x27, 0x67, 0x5e, 0xc7, 0x0b, 0x3c, 0xb4, 0x0f, 0x15, 0x9e, + 0x58, 0x57, 0xe7, 0xa9, 0xb5, 0x1e, 0x7e, 0xfe, 0x4a, 0x38, 0x83, 0xa3, 0x6f, 0xa0, 0x22, 0xbf, + 0x2c, 0x16, 0xfa, 0xaa, 0xd2, 0xbc, 0xd7, 0x4c, 0x32, 0xb2, 0x99, 0x65, 0x64, 0xb3, 0x93, 0x66, + 0x2c, 0xce, 0x90, 0x68, 0x17, 0x2a, 0x9c, 0x0a, 0xee, 0xd1, 0x48, 0x5f, 0x53, 0x4a, 0xf7, 0xaf, + 0x84, 0xbd, 0xe0, 0xd3, 0x91, 0xe0, 0x44, 0x50, 0x77, 0x8a, 0x33, 0x30, 0xfa, 0x39, 0x34, 0xb2, + 0x63, 0x4f, 0xad, 0xcc, 0x81, 0x65, 0xe5, 0x27, 0x2d, 0x17, 0xbc, 0x4b, 0x3d, 0xb9, 0x0f, 0xba, + 0x43, 0x43, 0x4e, 0x6d, 0x22, 0xa8, 0x63, 0x51, 0xce, 0x19, 0xb7, 0x26, 0x34, 0x8a, 0x88, 0x4b, + 0xf5, 0x8a, 0xd2, 0xb9, 0x3b, 0x93, 0x9b, 0x52, 0xdc, 0x4b, 0xa4, 0x32, 0x6e, 0x55, 0x09, 0xe3, + 0x71, 0x28, 0x3c, 0xe9, 0xb2, 0xf5, 0x2c, 0x6e, 0x17, 0xd8, 0xe8, 0x19, 0x20, 0x9b, 0xd8, 0xe7, + 0xd4, 0x8a, 0x28, 0xf7, 0x88, 0xef, 0x7d, 0x54, 0xfe, 0xad, 0x2a, 0xff, 0x36, 0x94, 0x64, 0x34, + 0x27, 0x40, 0x5f, 0xc1, 0xa6, 0x4b, 0x03, 0x19, 0x55, 0x34, 0xb2, 0x1c, 0x6a, 0x5f, 0xe8, 0xa0, + 0xa0, 0xf5, 0x9c, 0xdb, 0xa1, 0xf6, 0x85, 0x2c, 0x08, 0x82, 0xb8, 0x91, 0x5e, 0x53, 0xf5, 0x65, + 0xb9, 0x20, 0xcc, 0x3f, 0x6d, 0x73, 0x4c, 0xdc, 0x28, 0x29, 0x2f, 0x4a, 0x05, 0x7d, 0x0d, 0x8d, + 0x90, 0x39, 0x96, 0xa0, 0x93, 0xd0, 0x27, 0x82, 0x5a, 0xaa, 0xb0, 0x6c, 0xa8, 0xbb, 0x6e, 0x85, + 0xcc, 0x19, 0xa7, 0x7c, 0x59, 0x16, 0xb6, 0xf7, 0xa0, 0x9a, 0xab, 0x23, 0x0d, 0x4a, 0x17, 0x74, + 0x9a, 0xa6, 0xaf, 0x5c, 0x5e, 0x5f, 0x5b, 0x5e, 0x16, 0xf7, 0x0b, 0x07, 0x77, 0xe0, 0x8b, 0x05, + 0x37, 0xa4, 0x39, 0xf8, 0x9f, 0xb5, 0x24, 0xee, 0xb2, 0x4d, 0xd0, 0xcf, 0xa0, 0xe8, 0x39, 0x69, + 0xca, 0xdd, 0x5b, 0xba, 0x45, 0x37, 0xef, 0x20, 0xb8, 0xe8, 0x39, 0x08, 0xa5, 0xe9, 0x94, 0xec, + 0x95, 0xa4, 0xc8, 0x1e, 0xac, 0x4f, 0xd2, 0x7b, 0xaa, 0x54, 0xa8, 0xb5, 0x7e, 0xfc, 0x19, 0x57, + 0xe0, 0x1c, 0x8c, 0x7e, 0x05, 0xd5, 0xbc, 0x01, 0xa5, 0x31, 0xb9, 0x9c, 0xf1, 0x32, 0x85, 0x9c, + 0x6e, 0x06, 0xc2, 0x33, 0x3c, 0xda, 0x81, 0xb2, 0x1d, 0x47, 0x82, 0x4d, 0xd2, 0xc0, 0xfc, 0xd1, + 0x95, 0x68, 0x1e, 0xa9, 0xfe, 0x82, 0x53, 0x18, 0xda, 0x87, 0xaa, 0xcd, 0x02, 0x41, 0xbc, 0x80, + 0x72, 0x15, 0x8a, 0xb5, 0x96, 0xbe, 0xb4, 0xdb, 0xeb, 0x4c, 0x7e, 0xb8, 0x82, 0x67, 0x60, 0xf4, + 0x4b, 0xa8, 0x5c, 0xec, 0x47, 0x56, 0xc8, 0x1c, 0xbd, 0xa1, 0xf4, 0xee, 0x2c, 0xe9, 0x1d, 0xed, + 0x47, 0x43, 0xe6, 0x1c, 0xae, 0xe0, 0xf2, 0x85, 0x5a, 0xa1, 0xc7, 0x50, 0x8a, 0x3e, 0xf8, 0x3a, + 0x52, 0x68, 0xb4, 0x84, 0x1e, 0x7d, 0xf0, 0x0f, 0x57, 0xb0, 0x04, 0xc8, 0x30, 0x90, 0xcd, 0xdc, + 0x92, 0x7e, 0xcc, 0xd3, 0x44, 0x86, 0xfc, 0x1a, 0xde, 0x92, 0x02, 0x79, 0xfd, 0x2c, 0x4b, 0xba, + 0xa0, 0x65, 0x1d, 0xd7, 0x92, 0x67, 0xa3, 0x3f, 0x08, 0x15, 0xee, 0x57, 0x4b, 0xc0, 0x28, 0x85, + 0xbd, 0x4e, 0x50, 0x78, 0x2b, 0x5a, 0x64, 0xa0, 0x01, 0x20, 0x9a, 0x56, 0x61, 0x8b, 0x67, 0x65, + 0x58, 0xa5, 0x43, 0xad, 0xf5, 0x68, 0xc9, 0xd8, 0x95, 0x72, 0x8d, 0x1b, 0xf4, 0x4a, 0x05, 0xff, + 0x35, 0x94, 0x6d, 0x16, 0x9c, 0x79, 0xae, 0xae, 0xa9, 0x5c, 0x78, 0x72, 0x4d, 0x00, 0x64, 0xe1, + 0x26, 0xbd, 0x7c, 0xe6, 0xb9, 0x69, 0xb3, 0x4d, 0xd4, 0xb6, 0xbf, 0x85, 0xda, 0x1c, 0xfb, 0x56, + 0x51, 0xbe, 0x0e, 0x65, 0x41, 0xb8, 0x4b, 0x85, 0xb1, 0x0b, 0xf5, 0xfc, 0x05, 0x87, 0x8c, 0x0b, + 0x99, 0xc7, 0xf9, 0x2b, 0x5a, 0x21, 0xe3, 0x42, 0x59, 0xac, 0xe3, 0xba, 0x3d, 0x0f, 0x33, 0xfe, + 0x57, 0x82, 0x6a, 0xae, 0x28, 0x77, 0xf2, 0x26, 0xb2, 0xf4, 0x24, 0xbb, 0x27, 0x84, 0xec, 0x03, + 0x36, 0x9b, 0x4c, 0x48, 0xe0, 0xa8, 0x71, 0xa2, 0x8a, 0x33, 0x52, 0xa6, 0x04, 0xe1, 0x6e, 0xa4, + 0x97, 0x14, 0x5b, 0xad, 0xd1, 0x2e, 0x54, 0x67, 0x7e, 0x5d, 0xbd, 0x36, 0xd6, 0x66, 0xfe, 0x9c, + 0x41, 0xd1, 0x33, 0x28, 0xd1, 0xe0, 0x52, 0x5f, 0x53, 0x4e, 0x5c, 0xce, 0xa2, 0x23, 0x3a, 0x7d, + 0x27, 0x6f, 0x3d, 0x24, 0x1e, 0xc7, 0x12, 0x87, 0xf6, 0x72, 0xb7, 0x97, 0x6f, 0xd4, 0x38, 0x28, + 0xea, 0x85, 0xcc, 0xdd, 0xa8, 0x05, 0x6b, 0xd2, 0x1d, 0x91, 0x5e, 0x51, 0x7a, 0xf7, 0x3f, 0x95, + 0x07, 0xd2, 0x3d, 0x38, 0x81, 0xa2, 0x36, 0xd4, 0x64, 0xd6, 0x5a, 0xe9, 0x8e, 0xd7, 0x47, 0x4b, + 0x87, 0x08, 0x72, 0xcc, 0x88, 0xe3, 0x05, 0x6e, 0xf2, 0x9e, 0x18, 0xa4, 0x52, 0xb2, 0x46, 0x5d, + 0xd8, 0x20, 0xdc, 0x3e, 0xf7, 0x04, 0xb5, 0x45, 0xcc, 0xa9, 0xaa, 0xaa, 0x57, 0x27, 0xa9, 0x7c, + 0xf7, 0x66, 0x7b, 0x0e, 0x8c, 0x17, 0x54, 0x8d, 0x2e, 0x6c, 0xcc, 0x4b, 0x17, 0x67, 0xa7, 0x2a, + 0xac, 0xb5, 0x7b, 0x9d, 0xdd, 0x17, 0x5a, 0x41, 0x2d, 0x71, 0x6f, 0xf7, 0x45, 0x32, 0x3f, 0xb5, + 0x71, 0xcf, 0x7a, 0xb7, 0xab, 0x95, 0xf2, 0xf5, 0x9e, 0xb6, 0x6a, 0xfc, 0xb3, 0x08, 0xd0, 0x1d, + 0x64, 0x3d, 0x0c, 0x1d, 0x41, 0xdd, 0x61, 0x7f, 0x0e, 0x7c, 0x46, 0x1c, 0x6b, 0xc2, 0x9c, 0x6c, + 0x74, 0x58, 0x1e, 0x1f, 0x67, 0x1a, 0xcd, 0x4e, 0x0a, 0xef, 0x31, 0x87, 0xe2, 0x0d, 0x67, 0x8e, + 0x42, 0x26, 0xd4, 0xe2, 0x70, 0x66, 0xaa, 0xa8, 0x4c, 0xfd, 0xf4, 0xd3, 0xa6, 0x4e, 0xc2, 0xdc, + 0x10, 0xc4, 0xf9, 0xda, 0x38, 0x86, 0x8d, 0xf9, 0x4d, 0x10, 0x82, 0xcd, 0xce, 0xe0, 0xfb, 0xfe, + 0xf1, 0xa0, 0xdd, 0xb1, 0xcc, 0xf6, 0x5b, 0x35, 0x5d, 0x7c, 0x01, 0x5b, 0x39, 0x6f, 0x34, 0xc6, + 0x66, 0xbb, 0xa7, 0x15, 0x12, 0xa6, 0xd5, 0x1f, 0x8c, 0xad, 0x4c, 0xa6, 0x15, 0x0d, 0x13, 0x60, + 0xb6, 0x8f, 0xb4, 0x75, 0x32, 0x54, 0x5a, 0x83, 0xbe, 0x65, 0xfe, 0xb6, 0x3b, 0xd6, 0x56, 0x90, + 0x06, 0x1b, 0x29, 0x2f, 0xb1, 0x5e, 0x40, 0x0d, 0xa8, 0xa7, 0x86, 0x12, 0x81, 0x56, 0x34, 0xfe, + 0x56, 0x84, 0xc6, 0x95, 0xf7, 0x96, 0x89, 0x42, 0x03, 0xd9, 0x45, 0x9d, 0x74, 0x7e, 0xc9, 0x48, + 0xf4, 0x00, 0xc0, 0x0b, 0xc2, 0x58, 0x58, 0x21, 0x11, 0xe7, 0x69, 0x1e, 0x57, 0x15, 0x67, 0x48, + 0xc4, 0x39, 0xfa, 0x09, 0xd4, 0x58, 0x2c, 0x72, 0x79, 0x32, 0x54, 0x41, 0xc2, 0x52, 0x80, 0x2e, + 0x94, 0xcf, 0x18, 0x9f, 0x90, 0x64, 0x7e, 0xd9, 0x6c, 0x3d, 0xbf, 0x29, 0xf6, 0x9a, 0xc7, 0xc9, + 0xff, 0x4b, 0x8f, 0x84, 0x6f, 0x94, 0x22, 0x4e, 0x0d, 0xa0, 0x97, 0x50, 0xf3, 0x98, 0x15, 0xa5, + 0x5e, 0x4f, 0x3b, 0xc8, 0xbd, 0x4f, 0x3e, 0x0b, 0x06, 0x8f, 0x65, 0x6b, 0xe3, 0x39, 0x68, 0xcb, + 0x76, 0xd1, 0x3a, 0xac, 0xfe, 0x66, 0x34, 0x90, 0xa1, 0xb7, 0x0e, 0xab, 0xef, 0xdb, 0xbd, 0xe3, + 0x24, 0xf2, 0x86, 0x78, 0x30, 0x1e, 0x68, 0x45, 0xe3, 0x5f, 0x05, 0x28, 0x27, 0x3d, 0x02, 0x7d, + 0x37, 0xd7, 0x2c, 0x0b, 0xd7, 0xa6, 0xd0, 0xd1, 0x7e, 0x34, 0x38, 0xfd, 0x13, 0xb5, 0xc5, 0x35, + 0x1d, 0xb3, 0x05, 0xeb, 0x72, 0x6c, 0x88, 0x42, 0x6a, 0xa7, 0xe3, 0xdf, 0x27, 0xdb, 0x5e, 0x25, + 0x64, 0xce, 0x28, 0xa4, 0xf6, 0x72, 0xde, 0x96, 0x6e, 0x9f, 0xb7, 0xc6, 0xdf, 0x8b, 0xd0, 0xb8, + 0x72, 0x2c, 0xd4, 0x81, 0xb2, 0x4f, 0x4e, 0xa9, 0x9f, 0xfd, 0xa2, 0xfd, 0xe2, 0xa6, 0x8b, 0x34, + 0x8f, 0x15, 0x3c, 0xfb, 0xcd, 0x52, 0x04, 0x1a, 0x41, 0x8d, 0x04, 0x01, 0x13, 0x6a, 0xf0, 0xcc, + 0xfe, 0xd5, 0x9e, 0xdf, 0x68, 0xaa, 0x3d, 0xd3, 0x49, 0xec, 0xcd, 0x5b, 0x91, 0xed, 0x64, 0x6e, + 0xaf, 0xdb, 0xb4, 0x93, 0xed, 0x57, 0xa0, 0x2d, 0xdb, 0xbe, 0x8d, 0xbe, 0xf1, 0xd7, 0x02, 0x94, + 0x46, 0x1f, 0x7c, 0x74, 0x1f, 0xaa, 0x91, 0x20, 0x82, 0x4e, 0x68, 0x20, 0x52, 0xcd, 0x19, 0x03, + 0xbd, 0x80, 0x8a, 0xe3, 0x11, 0x9f, 0xda, 0x22, 0xad, 0x09, 0xdb, 0x57, 0x87, 0x84, 0x66, 0x27, + 0x41, 0xe0, 0x0c, 0x6a, 0xec, 0x41, 0x25, 0xe5, 0xc9, 0x7f, 0x89, 0x93, 0x7e, 0xc7, 0x7c, 0xd3, + 0xed, 0x9b, 0x9d, 0x24, 0xec, 0xda, 0xfd, 0x51, 0x57, 0x2b, 0xc8, 0xd5, 0x61, 0xf7, 0x9d, 0xa9, + 0x15, 0x67, 0x7f, 0x1e, 0xa5, 0x83, 0xef, 0x7e, 0xf7, 0xd2, 0xf5, 0xc4, 0x79, 0x7c, 0xda, 0xb4, + 0xd9, 0x64, 0x47, 0xed, 0xc4, 0xb8, 0x9b, 0x2c, 0x76, 0xf2, 0xbf, 0x7a, 0x97, 0x06, 0x3b, 0xe1, + 0xe9, 0x33, 0x97, 0xed, 0x2c, 0xfc, 0xe8, 0x9f, 0x96, 0x55, 0x6c, 0x7d, 0xf3, 0xff, 0x00, 0x00, + 0x00, 0xff, 0xff, 0x61, 0xf6, 0x49, 0xdf, 0x87, 0x10, 0x00, 0x00, } diff --git a/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.validate.go b/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.validate.go index 57d157dfd05..7731c431251 100644 --- a/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.validate.go +++ b/flyteidl/gen/pb-go/flyteidl/core/tasks.pb.validate.go @@ -130,6 +130,160 @@ var _ interface { ErrorName() string } = ResourcesValidationError{} +// Validate checks the field values on GPUAccelerator with the rules defined in +// the proto definition for this message. If any rules are violated, an error +// is returned. +func (m *GPUAccelerator) Validate() error { + if m == nil { + return nil + } + + // no validation rules for Device + + switch m.PartitionSizeValue.(type) { + + case *GPUAccelerator_Unpartitioned: + // no validation rules for Unpartitioned + + case *GPUAccelerator_PartitionSize: + // no validation rules for PartitionSize + + } + + return nil +} + +// GPUAcceleratorValidationError is the validation error returned by +// GPUAccelerator.Validate if the designated constraints aren't met. +type GPUAcceleratorValidationError struct { + field string + reason string + cause error + key bool +} + +// Field function returns field value. +func (e GPUAcceleratorValidationError) Field() string { return e.field } + +// Reason function returns reason value. +func (e GPUAcceleratorValidationError) Reason() string { return e.reason } + +// Cause function returns cause value. +func (e GPUAcceleratorValidationError) Cause() error { return e.cause } + +// Key function returns key value. +func (e GPUAcceleratorValidationError) Key() bool { return e.key } + +// ErrorName returns error name. +func (e GPUAcceleratorValidationError) ErrorName() string { return "GPUAcceleratorValidationError" } + +// Error satisfies the builtin error interface +func (e GPUAcceleratorValidationError) Error() string { + cause := "" + if e.cause != nil { + cause = fmt.Sprintf(" | caused by: %v", e.cause) + } + + key := "" + if e.key { + key = "key for " + } + + return fmt.Sprintf( + "invalid %sGPUAccelerator.%s: %s%s", + key, + e.field, + e.reason, + cause) +} + +var _ error = GPUAcceleratorValidationError{} + +var _ interface { + Field() string + Reason() string + Key() bool + Cause() error + ErrorName() string +} = GPUAcceleratorValidationError{} + +// Validate checks the field values on ExtendedResources with the rules defined +// in the proto definition for this message. If any rules are violated, an +// error is returned. +func (m *ExtendedResources) Validate() error { + if m == nil { + return nil + } + + if v, ok := interface{}(m.GetGpuAccelerator()).(interface{ Validate() error }); ok { + if err := v.Validate(); err != nil { + return ExtendedResourcesValidationError{ + field: "GpuAccelerator", + reason: "embedded message failed validation", + cause: err, + } + } + } + + return nil +} + +// ExtendedResourcesValidationError is the validation error returned by +// ExtendedResources.Validate if the designated constraints aren't met. +type ExtendedResourcesValidationError struct { + field string + reason string + cause error + key bool +} + +// Field function returns field value. +func (e ExtendedResourcesValidationError) Field() string { return e.field } + +// Reason function returns reason value. +func (e ExtendedResourcesValidationError) Reason() string { return e.reason } + +// Cause function returns cause value. +func (e ExtendedResourcesValidationError) Cause() error { return e.cause } + +// Key function returns key value. +func (e ExtendedResourcesValidationError) Key() bool { return e.key } + +// ErrorName returns error name. +func (e ExtendedResourcesValidationError) ErrorName() string { + return "ExtendedResourcesValidationError" +} + +// Error satisfies the builtin error interface +func (e ExtendedResourcesValidationError) Error() string { + cause := "" + if e.cause != nil { + cause = fmt.Sprintf(" | caused by: %v", e.cause) + } + + key := "" + if e.key { + key = "key for " + } + + return fmt.Sprintf( + "invalid %sExtendedResources.%s: %s%s", + key, + e.field, + e.reason, + cause) +} + +var _ error = ExtendedResourcesValidationError{} + +var _ interface { + Field() string + Reason() string + Key() bool + Cause() error + ErrorName() string +} = ExtendedResourcesValidationError{} + // Validate checks the field values on RuntimeMetadata with the rules defined // in the proto definition for this message. If any rules are violated, an // error is returned. @@ -379,6 +533,16 @@ func (m *TaskTemplate) Validate() error { } } + if v, ok := interface{}(m.GetExtendedResources()).(interface{ Validate() error }); ok { + if err := v.Validate(); err != nil { + return TaskTemplateValidationError{ + field: "ExtendedResources", + reason: "embedded message failed validation", + cause: err, + } + } + } + // no validation rules for Config switch m.Target.(type) { diff --git a/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.go b/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.go index 4c9d57f38b8..8ab14963306 100644 --- a/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.go +++ b/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.go @@ -1282,10 +1282,13 @@ func (m *WorkflowTemplate) GetMetadataDefaults() *WorkflowMetadataDefaults { // Optional task node overrides that will be applied at task execution time. type TaskNodeOverrides struct { // A customizable interface to convey resources requested for a task container. - Resources *Resources `protobuf:"bytes,1,opt,name=resources,proto3" json:"resources,omitempty"` - XXX_NoUnkeyedLiteral struct{} `json:"-"` - XXX_unrecognized []byte `json:"-"` - XXX_sizecache int32 `json:"-"` + Resources *Resources `protobuf:"bytes,1,opt,name=resources,proto3" json:"resources,omitempty"` + // Overrides for all non-standard resources, not captured by + // v1.ResourceRequirements, to allocate to a task. + ExtendedResources *ExtendedResources `protobuf:"bytes,2,opt,name=extended_resources,json=extendedResources,proto3" json:"extended_resources,omitempty"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` } func (m *TaskNodeOverrides) Reset() { *m = TaskNodeOverrides{} } @@ -1320,6 +1323,13 @@ func (m *TaskNodeOverrides) GetResources() *Resources { return nil } +func (m *TaskNodeOverrides) GetExtendedResources() *ExtendedResources { + if m != nil { + return m.ExtendedResources + } + return nil +} + func init() { proto.RegisterEnum("flyteidl.core.WorkflowMetadata_OnFailurePolicy", WorkflowMetadata_OnFailurePolicy_name, WorkflowMetadata_OnFailurePolicy_value) proto.RegisterType((*IfBlock)(nil), "flyteidl.core.IfBlock") @@ -1345,96 +1355,98 @@ func init() { func init() { proto.RegisterFile("flyteidl/core/workflow.proto", fileDescriptor_fccede37486c456e) } var fileDescriptor_fccede37486c456e = []byte{ - // 1448 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x57, 0x4d, 0x73, 0xdb, 0xba, - 0x15, 0x95, 0x64, 0xd9, 0x16, 0xaf, 0x2d, 0x5b, 0x46, 0xdc, 0x56, 0xf9, 0xf6, 0x70, 0xd2, 0xc6, - 0xc9, 0xa4, 0x52, 0x26, 0x99, 0x38, 0x69, 0x9c, 0x66, 0x2a, 0xc5, 0x4a, 0xa4, 0xa9, 0x3f, 0x12, - 0x58, 0xfd, 0xdc, 0x70, 0x20, 0x12, 0x94, 0x31, 0xa6, 0x48, 0x05, 0x00, 0x9d, 0x68, 0xfa, 0x27, - 0xba, 0xeb, 0xb2, 0x33, 0x5d, 0xf4, 0x6f, 0x74, 0xd5, 0xed, 0x5b, 0xbd, 0x9f, 0xf3, 0x16, 0x6f, - 0x00, 0x12, 0x94, 0x45, 0x4b, 0x71, 0x76, 0x24, 0xee, 0xb9, 0xe0, 0xc1, 0xc5, 0x3d, 0xe7, 0x4a, - 0x70, 0xc7, 0x0f, 0x26, 0x92, 0x32, 0x2f, 0x68, 0xba, 0x11, 0xa7, 0xcd, 0x2f, 0x11, 0x3f, 0xf7, - 0x83, 0xe8, 0x4b, 0x63, 0xcc, 0x23, 0x19, 0xa1, 0xaa, 0x89, 0x36, 0x54, 0xf4, 0xd6, 0xdd, 0x59, - 0xb0, 0x1b, 0x85, 0x1e, 0x93, 0x2c, 0x0a, 0x13, 0x74, 0x3e, 0x4c, 0xbf, 0x52, 0x37, 0xbe, 0x14, - 0xbe, 0x37, 0x1b, 0x66, 0x1e, 0x0d, 0x25, 0xf3, 0x19, 0xe5, 0xf3, 0xd3, 0x59, 0x28, 0x29, 0xf7, - 0x89, 0x4b, 0xd3, 0x70, 0x8e, 0x69, 0xc0, 0x24, 0xe5, 0x24, 0x10, 0x69, 0xf4, 0xe6, 0x6c, 0x54, - 0x12, 0x71, 0xbe, 0x28, 0x34, 0x19, 0x53, 0x31, 0x7f, 0x4f, 0x41, 0xdd, 0x98, 0x33, 0x39, 0x31, - 0x84, 0x87, 0x51, 0x34, 0x0c, 0x68, 0x53, 0xbf, 0x0d, 0x62, 0xbf, 0xe9, 0xc5, 0x9c, 0x4c, 0x0f, - 0x64, 0xff, 0x03, 0x56, 0x7b, 0x7e, 0x3b, 0x88, 0xdc, 0x73, 0xf4, 0x16, 0xac, 0xac, 0x1a, 0xf5, - 0xe2, 0x4e, 0x71, 0x77, 0xed, 0xd9, 0x4e, 0x63, 0xa6, 0x78, 0x8d, 0x76, 0x14, 0x05, 0x94, 0x84, - 0x9d, 0xaf, 0x63, 0x4e, 0x85, 0x60, 0x51, 0x88, 0xa7, 0x29, 0xe8, 0x29, 0x58, 0xf2, 0x8c, 0x86, - 0x4e, 0x18, 0x79, 0xb4, 0x5e, 0xd2, 0xf9, 0x37, 0x72, 0xf9, 0xc7, 0x91, 0x47, 0x71, 0x45, 0xa1, - 0xd4, 0x93, 0xfd, 0x63, 0x11, 0xd6, 0x7a, 0x7e, 0x27, 0x10, 0x34, 0x61, 0xf0, 0x18, 0xca, 0x2e, - 0x11, 0x34, 0xfd, 0xf8, 0x2f, 0x73, 0xc9, 0x29, 0x4f, 0xac, 0x31, 0xe8, 0x09, 0x2c, 0x47, 0xf2, - 0x8c, 0xf2, 0x7a, 0x69, 0x67, 0xe9, 0x1b, 0xe0, 0x04, 0x84, 0x9e, 0x81, 0x45, 0x03, 0x41, 0x13, - 0x6e, 0x4b, 0x0b, 0xb9, 0x75, 0x0b, 0xb8, 0xa2, 0x70, 0xea, 0x59, 0x7d, 0x81, 0x72, 0x1e, 0xf1, - 0x7a, 0x59, 0xe3, 0xb7, 0x73, 0xf8, 0x8e, 0x8a, 0x75, 0x0b, 0x38, 0x01, 0xb5, 0x2d, 0x58, 0xf5, - 0xa8, 0x4f, 0xe2, 0x40, 0xda, 0x2d, 0x80, 0x36, 0x27, 0xa1, 0x7b, 0xa6, 0xb7, 0x79, 0x0e, 0xab, - 0xcc, 0x77, 0xd4, 0xae, 0xe9, 0xb9, 0x6e, 0x5d, 0xa1, 0x9a, 0x55, 0x00, 0xaf, 0x30, 0xfd, 0x62, - 0xff, 0xab, 0x08, 0x95, 0x3e, 0x11, 0xe7, 0x7a, 0x87, 0xb7, 0xb0, 0xce, 0xa9, 0x4f, 0x39, 0x0d, - 0x5d, 0xea, 0x30, 0x2f, 0xdd, 0xe6, 0x66, 0x7e, 0x9b, 0xac, 0x17, 0xbb, 0x05, 0xbc, 0x96, 0x25, - 0xf4, 0x3c, 0x75, 0xb1, 0xd1, 0x05, 0xe5, 0x9c, 0x79, 0x54, 0xa4, 0x17, 0x93, 0xbf, 0x58, 0xf3, - 0xad, 0x13, 0x83, 0xc3, 0xd3, 0x94, 0xf6, 0x1a, 0x58, 0xd9, 0x76, 0xf6, 0x7f, 0x8b, 0xb0, 0xfe, - 0x97, 0x54, 0x61, 0x9a, 0x5d, 0x1b, 0x36, 0x02, 0x12, 0x87, 0xee, 0xd9, 0x38, 0x20, 0xa1, 0xc3, - 0xa9, 0xff, 0x3d, 0xfc, 0xaa, 0xd3, 0x14, 0x4c, 0x7d, 0xd4, 0x81, 0x9a, 0x88, 0x07, 0x8e, 0x51, - 0xae, 0xde, 0xa5, 0x74, 0xfd, 0x2e, 0x1b, 0x22, 0x1e, 0x18, 0x2e, 0x98, 0xfa, 0xb3, 0x44, 0x9b, - 0x50, 0x6b, 0x8d, 0xc7, 0x3c, 0xba, 0xa0, 0xef, 0xb2, 0x16, 0xbd, 0x0d, 0x96, 0x60, 0xc3, 0x90, - 0x04, 0xa6, 0x8c, 0x16, 0xae, 0x24, 0x0b, 0x3d, 0xcf, 0xfe, 0x67, 0x11, 0x36, 0x4f, 0xf5, 0xcb, - 0xf7, 0x25, 0xa0, 0x06, 0x94, 0x95, 0x10, 0x53, 0xa6, 0xf9, 0x6b, 0x3d, 0x4c, 0xc4, 0xdd, 0x9f, - 0x8c, 0x29, 0xd6, 0x38, 0xf4, 0x14, 0xb6, 0xa3, 0x58, 0x8e, 0x63, 0xe9, 0x5c, 0x10, 0xce, 0xc8, - 0x20, 0xa0, 0x4e, 0x48, 0x46, 0x49, 0x3f, 0x5a, 0x18, 0x25, 0xb1, 0x3f, 0xa7, 0xa1, 0x63, 0x32, - 0xa2, 0xf6, 0x07, 0xd8, 0x38, 0x0d, 0x28, 0x1d, 0x4f, 0x09, 0xbd, 0x80, 0x8a, 0x51, 0x70, 0x56, - 0xe7, 0x44, 0xe2, 0x0d, 0x23, 0xf1, 0xc6, 0x41, 0x0a, 0xc0, 0x19, 0xd4, 0xfe, 0x7f, 0x11, 0x2a, - 0x1f, 0x88, 0x4c, 0x1a, 0x7b, 0x1f, 0x56, 0x49, 0x52, 0x99, 0x74, 0x8b, 0xfb, 0x39, 0xea, 0xf9, - 0xba, 0x75, 0x0b, 0xd8, 0x64, 0xa0, 0x57, 0xb0, 0x92, 0x14, 0x20, 0x3d, 0xf6, 0xbd, 0x5c, 0x6e, - 0xae, 0x82, 0xdd, 0x02, 0x4e, 0xf1, 0xe8, 0x05, 0x2c, 0x0b, 0x75, 0x98, 0x54, 0x7f, 0x77, 0xf3, - 0x89, 0x33, 0x07, 0x55, 0xc2, 0xd2, 0x68, 0x75, 0xa9, 0x99, 0xc7, 0xd8, 0xff, 0x2b, 0x82, 0xd5, - 0xe2, 0x9c, 0x4c, 0xf4, 0x41, 0x1e, 0x42, 0x59, 0x0b, 0xba, 0xb8, 0xd8, 0x6c, 0x34, 0x00, 0xed, - 0xc0, 0xda, 0x98, 0x70, 0x12, 0x04, 0x34, 0x60, 0x62, 0xa4, 0x99, 0x57, 0xf1, 0xe5, 0x25, 0xf4, - 0x6b, 0xa8, 0x8e, 0x58, 0xe8, 0x88, 0xd8, 0x75, 0xa9, 0x10, 0x54, 0x68, 0x92, 0xd5, 0x6e, 0x01, - 0xaf, 0x8f, 0x58, 0x78, 0x6a, 0x56, 0xd1, 0x13, 0xd8, 0xba, 0x04, 0x73, 0x74, 0x75, 0xb5, 0x3f, - 0x94, 0xba, 0x05, 0xbc, 0x39, 0x85, 0x62, 0x15, 0x68, 0x23, 0xd5, 0xd6, 0x09, 0xd2, 0xe5, 0xaa, - 0x1d, 0x18, 0xb1, 0x7f, 0x28, 0xc2, 0xba, 0x62, 0x76, 0x44, 0x25, 0xf1, 0x88, 0x24, 0x08, 0x41, - 0x59, 0x77, 0x41, 0xd2, 0x5d, 0xfa, 0x59, 0x79, 0x86, 0x64, 0x23, 0x1a, 0xc5, 0x32, 0x35, 0x9f, - 0x6f, 0x5c, 0xb2, 0x41, 0xa2, 0x3d, 0x58, 0xe5, 0x54, 0x72, 0x46, 0x45, 0x7d, 0x59, 0x27, 0xdd, - 0xc9, 0x15, 0x04, 0x53, 0xc9, 0x27, 0xa7, 0x92, 0x13, 0x49, 0x87, 0x13, 0x6c, 0xc0, 0xe8, 0x37, - 0x50, 0xd5, 0x73, 0x8a, 0xc7, 0x63, 0xc9, 0x06, 0x01, 0xad, 0xaf, 0xec, 0x14, 0x77, 0x2b, 0x4a, - 0xa4, 0x33, 0xcb, 0xed, 0x5f, 0xc0, 0x8d, 0x99, 0x05, 0xe7, 0x82, 0x04, 0xb1, 0xd2, 0xd9, 0x72, - 0x2b, 0x60, 0x44, 0xa0, 0x1a, 0x2c, 0x5d, 0x10, 0x9e, 0x9e, 0x43, 0x3d, 0xa2, 0x6d, 0x58, 0x26, - 0x2a, 0xa4, 0x0b, 0x6e, 0xe1, 0xe4, 0xc5, 0xfe, 0x77, 0x19, 0xca, 0xfa, 0xfa, 0x36, 0xa0, 0x94, - 0xa9, 0xaa, 0xc4, 0x3c, 0xf4, 0x12, 0x2a, 0xa3, 0xb4, 0x2a, 0x69, 0x73, 0xdd, 0x9e, 0x73, 0xa5, - 0xa6, 0x70, 0x38, 0x03, 0xa3, 0x06, 0xac, 0xb0, 0x70, 0x1c, 0x4b, 0x75, 0x6b, 0xf3, 0x86, 0x41, - 0x9b, 0x85, 0x1e, 0x0b, 0x87, 0x38, 0x45, 0xa1, 0xc7, 0xb0, 0x15, 0x8f, 0x85, 0xe4, 0x94, 0x8c, - 0xf4, 0x44, 0x70, 0x98, 0x27, 0xea, 0xe5, 0x9d, 0xa5, 0x5d, 0x0b, 0x6f, 0x9a, 0x80, 0xfa, 0x54, - 0xcf, 0x13, 0x68, 0x1f, 0x36, 0x52, 0xd1, 0x6a, 0xf6, 0xba, 0xb8, 0x4b, 0x73, 0xc6, 0x81, 0xae, - 0x01, 0xae, 0x26, 0xd8, 0x56, 0x02, 0x45, 0x7b, 0x60, 0xa9, 0x29, 0x9e, 0x8c, 0x9d, 0x15, 0x7d, - 0xa4, 0x5f, 0x2d, 0x70, 0x5e, 0x35, 0x7a, 0xa4, 0x71, 0xfc, 0x36, 0x54, 0x33, 0x2f, 0xd4, 0xb9, - 0xab, 0x73, 0xcb, 0x71, 0xd9, 0x87, 0x55, 0xab, 0x7e, 0xb9, 0xec, 0xcb, 0x6f, 0x60, 0x6d, 0xa0, - 0xa7, 0x50, 0xb2, 0x43, 0x65, 0xae, 0x9d, 0x4e, 0xe7, 0x54, 0xb7, 0x80, 0x61, 0x30, 0x9d, 0x5a, - 0x7b, 0x60, 0x0d, 0x89, 0x4c, 0x07, 0xa6, 0x35, 0x97, 0xb9, 0xf1, 0x13, 0xc5, 0x7c, 0x68, 0xbc, - 0xe5, 0x77, 0x00, 0x44, 0xe9, 0x33, 0x49, 0x04, 0x9d, 0x58, 0xcf, 0x97, 0xca, 0x08, 0xb8, 0x5b, - 0xc0, 0x16, 0x31, 0x2f, 0xed, 0x0a, 0xac, 0x48, 0xc2, 0x87, 0x54, 0xda, 0x3f, 0x95, 0xa0, 0x66, - 0xce, 0x96, 0xe9, 0xe4, 0x08, 0xd0, 0xe7, 0x98, 0x04, 0x4c, 0x4e, 0x9c, 0xc8, 0x77, 0x04, 0xe5, - 0x17, 0xcc, 0x5d, 0x64, 0x60, 0x9f, 0x12, 0xe0, 0x89, 0x7f, 0x9a, 0xc0, 0x70, 0xed, 0x73, 0x6e, - 0x05, 0x1d, 0x03, 0x44, 0xa1, 0xe3, 0x13, 0x16, 0xc4, 0x3c, 0xb1, 0xf0, 0x8d, 0x67, 0xcd, 0x05, - 0xf5, 0x35, 0x1c, 0x1a, 0x27, 0xe1, 0xfb, 0x24, 0xe1, 0x63, 0x14, 0x30, 0x77, 0x82, 0xad, 0xc8, - 0x2c, 0xa0, 0xdf, 0x43, 0x59, 0x92, 0xa1, 0xe9, 0xc0, 0x47, 0xd7, 0xed, 0xd4, 0x27, 0x43, 0xd1, - 0x09, 0x25, 0x9f, 0x60, 0x9d, 0x76, 0xeb, 0x25, 0x58, 0xd9, 0x92, 0x52, 0xd2, 0x39, 0x9d, 0x18, - 0x25, 0x9d, 0xd3, 0x89, 0x52, 0x92, 0x56, 0x9b, 0x51, 0x92, 0x7e, 0x79, 0x5d, 0x7a, 0x55, 0xb4, - 0x3f, 0xc1, 0x66, 0x8e, 0x15, 0xda, 0x86, 0xda, 0xfb, 0x56, 0xef, 0xd0, 0xe9, 0x1d, 0x1d, 0x75, - 0x0e, 0x7a, 0xad, 0x7e, 0xe7, 0xf0, 0x6f, 0xb5, 0x02, 0xda, 0x85, 0x07, 0x7a, 0xb5, 0xf5, 0xbe, - 0xdf, 0xc1, 0x4e, 0xe7, 0xaf, 0x9d, 0x77, 0x7f, 0xea, 0xb7, 0xda, 0x87, 0x1d, 0xe7, 0xf8, 0xe4, - 0xa0, 0x73, 0xea, 0xbc, 0x3b, 0x39, 0xfa, 0x78, 0xd8, 0xe9, 0x77, 0x6a, 0x45, 0xfb, 0x0f, 0x50, - 0xcf, 0xf3, 0x3d, 0x48, 0x7e, 0xda, 0x08, 0xf4, 0x20, 0x6f, 0x16, 0x8a, 0x64, 0x25, 0x67, 0x15, - 0xf6, 0x7f, 0x96, 0xa6, 0x17, 0xd8, 0xa7, 0xa3, 0x71, 0x40, 0x24, 0x45, 0x8f, 0x32, 0xb9, 0x7f, - 0x6b, 0xac, 0x6b, 0x27, 0xd8, 0xbf, 0xe2, 0x04, 0xf7, 0xaf, 0x29, 0xe8, 0x25, 0x37, 0xd8, 0x07, - 0x2b, 0xfb, 0xdd, 0xbd, 0x60, 0xd6, 0xa8, 0xa1, 0xec, 0xf5, 0x0c, 0x08, 0x4f, 0xf1, 0xe8, 0x11, - 0x2c, 0xab, 0xce, 0x4d, 0xec, 0x60, 0xc1, 0x4c, 0x49, 0x10, 0xe8, 0x29, 0xac, 0x26, 0x6a, 0x37, - 0x96, 0xb0, 0xc8, 0x76, 0x0c, 0x0c, 0xed, 0xc1, 0x7a, 0xda, 0x70, 0x97, 0x1d, 0x61, 0xee, 0x37, - 0xd6, 0x52, 0xa0, 0x16, 0x55, 0x1f, 0xb6, 0xcc, 0xe9, 0x9c, 0xf4, 0x47, 0xa6, 0x48, 0x2d, 0xe1, - 0xe1, 0x35, 0x75, 0x31, 0x17, 0x87, 0x6b, 0xa3, 0xdc, 0x8a, 0xfd, 0x47, 0xd8, 0xba, 0xf2, 0xb3, - 0x4f, 0xe9, 0x9e, 0x53, 0x11, 0xc5, 0xdc, 0xa5, 0x22, 0xbd, 0xab, 0xfa, 0x95, 0x31, 0x92, 0xc6, - 0xf1, 0x14, 0xda, 0x7e, 0xf3, 0xf7, 0xd7, 0x43, 0x26, 0xcf, 0xe2, 0x41, 0xc3, 0x8d, 0x46, 0x4d, - 0x9d, 0x10, 0xf1, 0x61, 0xf2, 0xd0, 0xcc, 0xfe, 0xa1, 0x0c, 0x69, 0xd8, 0x1c, 0x0f, 0x7e, 0x3b, - 0x8c, 0x9a, 0x33, 0x7f, 0x5a, 0x06, 0x2b, 0x7a, 0xac, 0x3d, 0xff, 0x39, 0x00, 0x00, 0xff, 0xff, - 0x7d, 0xc4, 0xce, 0x2c, 0xca, 0x0d, 0x00, 0x00, + // 1473 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x57, 0x4d, 0x6f, 0xdb, 0xcc, + 0x11, 0x96, 0x64, 0xd9, 0x16, 0xc7, 0x96, 0x2d, 0x6f, 0xdc, 0x56, 0xf9, 0x36, 0x88, 0xb4, 0x71, + 0x82, 0x54, 0x0a, 0x12, 0xc4, 0x49, 0xe3, 0x34, 0xa8, 0x14, 0x2b, 0x91, 0x00, 0x7f, 0x24, 0x6b, + 0xf5, 0xf3, 0x42, 0xac, 0xc8, 0xa1, 0xbc, 0x30, 0x45, 0x2a, 0xcb, 0xa5, 0x13, 0xa1, 0x7f, 0xa2, + 0xb7, 0x5e, 0x0a, 0x14, 0xe8, 0xa1, 0x7f, 0xa3, 0xa7, 0x5e, 0xdf, 0xd3, 0xfb, 0x73, 0xde, 0xc3, + 0x8b, 0x5d, 0x7e, 0xc8, 0xa2, 0xa5, 0x38, 0x37, 0xee, 0xce, 0x33, 0xc3, 0xd9, 0xd9, 0x79, 0x9e, + 0x21, 0xe1, 0x8e, 0xeb, 0x4d, 0x24, 0x72, 0xc7, 0x6b, 0xda, 0x81, 0xc0, 0xe6, 0x97, 0x40, 0x9c, + 0xbb, 0x5e, 0xf0, 0xa5, 0x31, 0x16, 0x81, 0x0c, 0x48, 0x35, 0xb5, 0x36, 0x94, 0xf5, 0xd6, 0xdd, + 0x59, 0xb0, 0x1d, 0xf8, 0x0e, 0x97, 0x3c, 0xf0, 0x63, 0x74, 0xde, 0x8c, 0x5f, 0xd1, 0x8e, 0x2e, + 0x99, 0xef, 0xcd, 0x9a, 0xb9, 0x83, 0xbe, 0xe4, 0x2e, 0x47, 0x31, 0xdf, 0x9d, 0xfb, 0x12, 0x85, + 0xcb, 0x6c, 0x4c, 0xcc, 0xb9, 0x4c, 0x3d, 0x2e, 0x51, 0x30, 0x2f, 0x4c, 0xac, 0x37, 0x67, 0xad, + 0x92, 0x85, 0xe7, 0x8b, 0x4c, 0x93, 0x31, 0x86, 0xf3, 0x63, 0x86, 0x68, 0x47, 0x82, 0xcb, 0x49, + 0x9a, 0xf0, 0x30, 0x08, 0x86, 0x1e, 0x36, 0xf5, 0x6a, 0x10, 0xb9, 0x4d, 0x27, 0x12, 0x6c, 0x7a, + 0x20, 0xf3, 0xef, 0xb0, 0xda, 0x73, 0xdb, 0x5e, 0x60, 0x9f, 0x93, 0xb7, 0x60, 0x64, 0xd5, 0xa8, + 0x17, 0x77, 0x8a, 0xbb, 0x6b, 0xcf, 0x76, 0x1a, 0x33, 0xc5, 0x6b, 0xb4, 0x83, 0xc0, 0x43, 0xe6, + 0x77, 0xbe, 0x8e, 0x05, 0x86, 0x21, 0x0f, 0x7c, 0x3a, 0x75, 0x21, 0x4f, 0xc1, 0x90, 0x67, 0xe8, + 0x5b, 0x7e, 0xe0, 0x60, 0xbd, 0xa4, 0xfd, 0x6f, 0xe4, 0xfc, 0x8f, 0x03, 0x07, 0x69, 0x45, 0xa1, + 0xd4, 0x93, 0xf9, 0x63, 0x11, 0xd6, 0x7a, 0x6e, 0xc7, 0x0b, 0x31, 0xce, 0xe0, 0x31, 0x94, 0x6d, + 0x16, 0x62, 0xf2, 0xf2, 0x5f, 0xe6, 0x9c, 0x93, 0x3c, 0xa9, 0xc6, 0x90, 0x27, 0xb0, 0x1c, 0xc8, + 0x33, 0x14, 0xf5, 0xd2, 0xce, 0xd2, 0x37, 0xc0, 0x31, 0x88, 0x3c, 0x03, 0x03, 0xbd, 0x10, 0xe3, + 0xdc, 0x96, 0x16, 0xe6, 0xd6, 0x2d, 0xd0, 0x8a, 0xc2, 0xa9, 0x67, 0xf5, 0x06, 0x14, 0x22, 0x10, + 0xf5, 0xb2, 0xc6, 0x6f, 0xe7, 0xf0, 0x1d, 0x65, 0xeb, 0x16, 0x68, 0x0c, 0x6a, 0x1b, 0xb0, 0xea, + 0xa0, 0xcb, 0x22, 0x4f, 0x9a, 0x2d, 0x80, 0xb6, 0x60, 0xbe, 0x7d, 0xa6, 0xc3, 0x3c, 0x87, 0x55, + 0xee, 0x5a, 0x2a, 0x6a, 0x72, 0xae, 0x5b, 0x57, 0x52, 0xcd, 0x2a, 0x40, 0x57, 0xb8, 0x5e, 0x98, + 0xff, 0x2c, 0x42, 0xa5, 0xcf, 0xc2, 0x73, 0x1d, 0xe1, 0x2d, 0xac, 0x0b, 0x74, 0x51, 0xa0, 0x6f, + 0xa3, 0xc5, 0x9d, 0x24, 0xcc, 0xcd, 0x7c, 0x98, 0xac, 0x17, 0xbb, 0x05, 0xba, 0x96, 0x39, 0xf4, + 0x1c, 0x75, 0xb1, 0xc1, 0x05, 0x0a, 0xc1, 0x1d, 0x0c, 0x93, 0x8b, 0xc9, 0x5f, 0x6c, 0xfa, 0xae, + 0x93, 0x14, 0x47, 0xa7, 0x2e, 0xed, 0x35, 0x30, 0xb2, 0x70, 0xe6, 0x7f, 0x8b, 0xb0, 0xfe, 0xe7, + 0x84, 0x61, 0x3a, 0xbb, 0x36, 0x6c, 0x78, 0x2c, 0xf2, 0xed, 0xb3, 0xb1, 0xc7, 0x7c, 0x4b, 0xa0, + 0xfb, 0x3d, 0xf9, 0x55, 0xa7, 0x2e, 0x14, 0x5d, 0xd2, 0x81, 0x5a, 0x18, 0x0d, 0xac, 0x94, 0xb9, + 0x3a, 0x4a, 0xe9, 0xfa, 0x28, 0x1b, 0x61, 0x34, 0x48, 0x73, 0xa1, 0xe8, 0xce, 0x26, 0xda, 0x84, + 0x5a, 0x6b, 0x3c, 0x16, 0xc1, 0x05, 0xbe, 0xcb, 0x5a, 0xf4, 0x36, 0x18, 0x21, 0x1f, 0xfa, 0xcc, + 0x4b, 0xcb, 0x68, 0xd0, 0x4a, 0xbc, 0xd1, 0x73, 0xcc, 0x7f, 0x14, 0x61, 0xf3, 0x54, 0x2f, 0xbe, + 0xcf, 0x81, 0x34, 0xa0, 0xac, 0x88, 0x98, 0x64, 0x9a, 0xbf, 0xd6, 0xc3, 0x98, 0xdc, 0xfd, 0xc9, + 0x18, 0xa9, 0xc6, 0x91, 0xa7, 0xb0, 0x1d, 0x44, 0x72, 0x1c, 0x49, 0xeb, 0x82, 0x09, 0xce, 0x06, + 0x1e, 0x5a, 0x3e, 0x1b, 0xc5, 0xfd, 0x68, 0x50, 0x12, 0xdb, 0xfe, 0x94, 0x98, 0x8e, 0xd9, 0x08, + 0xcd, 0x0f, 0xb0, 0x71, 0xea, 0x21, 0x8e, 0xa7, 0x09, 0xbd, 0x80, 0x4a, 0xca, 0xe0, 0xac, 0xce, + 0x31, 0xc5, 0x1b, 0x29, 0xc5, 0x1b, 0x07, 0x09, 0x80, 0x66, 0x50, 0xf3, 0xff, 0x45, 0xa8, 0x7c, + 0x60, 0x32, 0x6e, 0xec, 0x7d, 0x58, 0x65, 0x71, 0x65, 0x92, 0x10, 0xf7, 0x73, 0xa9, 0xe7, 0xeb, + 0xd6, 0x2d, 0xd0, 0xd4, 0x83, 0xbc, 0x82, 0x95, 0xb8, 0x00, 0xc9, 0xb1, 0xef, 0xe5, 0x7c, 0x73, + 0x15, 0xec, 0x16, 0x68, 0x82, 0x27, 0x2f, 0x60, 0x39, 0x54, 0x87, 0x49, 0xf8, 0x77, 0x37, 0xef, + 0x38, 0x73, 0x50, 0x45, 0x2c, 0x8d, 0x56, 0x97, 0x9a, 0x69, 0x8c, 0xf9, 0xbf, 0x22, 0x18, 0x2d, + 0x21, 0xd8, 0x44, 0x1f, 0xe4, 0x21, 0x94, 0x35, 0xa1, 0x8b, 0x8b, 0xc5, 0x46, 0x03, 0xc8, 0x0e, + 0xac, 0x8d, 0x99, 0x60, 0x9e, 0x87, 0x1e, 0x0f, 0x47, 0x3a, 0xf3, 0x2a, 0xbd, 0xbc, 0x45, 0x7e, + 0x0d, 0xd5, 0x11, 0xf7, 0xad, 0x30, 0xb2, 0x6d, 0x0c, 0x43, 0x0c, 0x75, 0x92, 0xd5, 0x6e, 0x81, + 0xae, 0x8f, 0xb8, 0x7f, 0x9a, 0xee, 0x92, 0x27, 0xb0, 0x75, 0x09, 0x66, 0xe9, 0xea, 0x6a, 0x7d, + 0x28, 0x75, 0x0b, 0x74, 0x73, 0x0a, 0xa5, 0xca, 0xd0, 0x26, 0xaa, 0xad, 0x63, 0xa4, 0x2d, 0x54, + 0x3b, 0x70, 0x66, 0xfe, 0x50, 0x84, 0x75, 0x95, 0xd9, 0x11, 0x4a, 0xe6, 0x30, 0xc9, 0x08, 0x81, + 0xb2, 0xee, 0x82, 0xb8, 0xbb, 0xf4, 0xb3, 0xd2, 0x0c, 0xc9, 0x47, 0x18, 0x44, 0x32, 0x11, 0x9f, + 0x6f, 0x5c, 0x72, 0x8a, 0x24, 0x7b, 0xb0, 0x2a, 0x50, 0x0a, 0x8e, 0x61, 0x7d, 0x59, 0x3b, 0xdd, + 0xc9, 0x15, 0x84, 0xa2, 0x14, 0x93, 0x53, 0x29, 0x98, 0xc4, 0xe1, 0x84, 0xa6, 0x60, 0xf2, 0x1b, + 0xa8, 0xea, 0x39, 0x25, 0xa2, 0xb1, 0xe4, 0x03, 0x0f, 0xeb, 0x2b, 0x3b, 0xc5, 0xdd, 0x8a, 0x22, + 0xe9, 0xcc, 0x76, 0xfb, 0x17, 0x70, 0x63, 0x66, 0xc3, 0xba, 0x60, 0x5e, 0xa4, 0x78, 0xb6, 0xdc, + 0xf2, 0x38, 0x0b, 0x49, 0x0d, 0x96, 0x2e, 0x98, 0x48, 0xce, 0xa1, 0x1e, 0xc9, 0x36, 0x2c, 0x33, + 0x65, 0xd2, 0x05, 0x37, 0x68, 0xbc, 0x30, 0xff, 0x5d, 0x86, 0xb2, 0xbe, 0xbe, 0x0d, 0x28, 0x65, + 0xac, 0x2a, 0x71, 0x87, 0xbc, 0x84, 0xca, 0x28, 0xa9, 0x4a, 0xd2, 0x5c, 0xb7, 0xe7, 0x5c, 0x69, + 0x5a, 0x38, 0x9a, 0x81, 0x49, 0x03, 0x56, 0xb8, 0x3f, 0x8e, 0xa4, 0xba, 0xb5, 0x79, 0xc3, 0xa0, + 0xcd, 0x7d, 0x87, 0xfb, 0x43, 0x9a, 0xa0, 0xc8, 0x63, 0xd8, 0x8a, 0xc6, 0xa1, 0x14, 0xc8, 0x46, + 0x7a, 0x22, 0x58, 0xdc, 0x09, 0xeb, 0xe5, 0x9d, 0xa5, 0x5d, 0x83, 0x6e, 0xa6, 0x06, 0xf5, 0xaa, + 0x9e, 0x13, 0x92, 0x7d, 0xd8, 0x48, 0x48, 0xab, 0xb3, 0xd7, 0xc5, 0x5d, 0x9a, 0x33, 0x0e, 0x74, + 0x0d, 0x68, 0x35, 0xc6, 0xb6, 0x62, 0x28, 0xd9, 0x03, 0x43, 0x4d, 0xf1, 0x78, 0xec, 0xac, 0xe8, + 0x23, 0xfd, 0x6a, 0x81, 0xf2, 0xaa, 0xd1, 0x23, 0x53, 0xc5, 0x6f, 0x43, 0x35, 0xd3, 0x42, 0xed, + 0xbb, 0x3a, 0xb7, 0x1c, 0x97, 0x75, 0x58, 0xb5, 0xea, 0x97, 0xcb, 0xba, 0xfc, 0x06, 0xd6, 0x06, + 0x7a, 0x0a, 0xc5, 0x11, 0x2a, 0x73, 0xe5, 0x74, 0x3a, 0xa7, 0xba, 0x05, 0x0a, 0x83, 0xe9, 0xd4, + 0xda, 0x03, 0x63, 0xc8, 0x64, 0x32, 0x30, 0x8d, 0xb9, 0x99, 0xa7, 0x7a, 0xa2, 0x32, 0x1f, 0xa6, + 0xda, 0xf2, 0x3b, 0x00, 0xa6, 0xf8, 0x19, 0x3b, 0x82, 0x76, 0xac, 0xe7, 0x4b, 0x95, 0x12, 0xb8, + 0x5b, 0xa0, 0x06, 0x4b, 0x17, 0xed, 0x0a, 0xac, 0x48, 0x26, 0x86, 0x28, 0xcd, 0x9f, 0x4a, 0x50, + 0x4b, 0xcf, 0x96, 0xf1, 0xe4, 0x08, 0xc8, 0xe7, 0x88, 0x79, 0x5c, 0x4e, 0xac, 0xc0, 0xb5, 0x42, + 0x14, 0x17, 0xdc, 0x5e, 0x24, 0x60, 0x9f, 0x62, 0xe0, 0x89, 0x7b, 0x1a, 0xc3, 0x68, 0xed, 0x73, + 0x6e, 0x87, 0x1c, 0x03, 0x04, 0xbe, 0xe5, 0x32, 0xee, 0x45, 0x22, 0x96, 0xf0, 0x8d, 0x67, 0xcd, + 0x05, 0xf5, 0x4d, 0x73, 0x68, 0x9c, 0xf8, 0xef, 0x63, 0x87, 0x8f, 0x81, 0xc7, 0xed, 0x09, 0x35, + 0x82, 0x74, 0x83, 0xfc, 0x1e, 0xca, 0x92, 0x0d, 0xd3, 0x0e, 0x7c, 0x74, 0x5d, 0xa4, 0x3e, 0x1b, + 0x86, 0x1d, 0x5f, 0x8a, 0x09, 0xd5, 0x6e, 0xb7, 0x5e, 0x82, 0x91, 0x6d, 0x29, 0x26, 0x9d, 0xe3, + 0x24, 0x65, 0xd2, 0x39, 0x4e, 0x14, 0x93, 0x34, 0xdb, 0x52, 0x26, 0xe9, 0xc5, 0xeb, 0xd2, 0xab, + 0xa2, 0xf9, 0x09, 0x36, 0x73, 0x59, 0x91, 0x6d, 0xa8, 0xbd, 0x6f, 0xf5, 0x0e, 0xad, 0xde, 0xd1, + 0x51, 0xe7, 0xa0, 0xd7, 0xea, 0x77, 0x0e, 0xff, 0x5a, 0x2b, 0x90, 0x5d, 0x78, 0xa0, 0x77, 0x5b, + 0xef, 0xfb, 0x1d, 0x6a, 0x75, 0xfe, 0xd2, 0x79, 0xf7, 0xc7, 0x7e, 0xab, 0x7d, 0xd8, 0xb1, 0x8e, + 0x4f, 0x0e, 0x3a, 0xa7, 0xd6, 0xbb, 0x93, 0xa3, 0x8f, 0x87, 0x9d, 0x7e, 0xa7, 0x56, 0x34, 0xff, + 0x00, 0xf5, 0x7c, 0xbe, 0x07, 0xf1, 0xa7, 0x4d, 0x48, 0x1e, 0xe4, 0xc5, 0x42, 0x25, 0x59, 0xc9, + 0x49, 0x85, 0xf9, 0x9f, 0xa5, 0xe9, 0x05, 0xf6, 0x71, 0x34, 0xf6, 0x98, 0x44, 0xf2, 0x28, 0xa3, + 0xfb, 0xb7, 0xc6, 0xba, 0x56, 0x82, 0xfd, 0x2b, 0x4a, 0x70, 0xff, 0x9a, 0x82, 0x5e, 0x52, 0x83, + 0x7d, 0x30, 0xb2, 0xef, 0xee, 0x05, 0xb3, 0x46, 0x0d, 0x65, 0xa7, 0x97, 0x82, 0xe8, 0x14, 0x4f, + 0x1e, 0xc1, 0xb2, 0xea, 0xdc, 0x58, 0x0e, 0x16, 0xcc, 0x94, 0x18, 0x41, 0x9e, 0xc2, 0x6a, 0xcc, + 0xf6, 0x54, 0x12, 0x16, 0xc9, 0x4e, 0x0a, 0x23, 0x7b, 0xb0, 0x9e, 0x34, 0xdc, 0x65, 0x45, 0x98, + 0xfb, 0x8e, 0xb5, 0x04, 0xa8, 0x49, 0xd5, 0x87, 0xad, 0xf4, 0x74, 0x56, 0xf2, 0x91, 0x19, 0x26, + 0x92, 0xf0, 0xf0, 0x9a, 0xba, 0xa4, 0x17, 0x47, 0x6b, 0xa3, 0xdc, 0x8e, 0xf9, 0xaf, 0x22, 0x6c, + 0x5d, 0xf9, 0xee, 0x53, 0xc4, 0x17, 0x18, 0x06, 0x91, 0xb0, 0x31, 0x4c, 0x2e, 0xab, 0x7e, 0x65, + 0x8e, 0x24, 0x76, 0x3a, 0x85, 0x92, 0x13, 0x20, 0xf8, 0x55, 0xa2, 0xef, 0xa0, 0x63, 0x4d, 0x03, + 0xcc, 0xff, 0xda, 0xec, 0x24, 0xc0, 0x69, 0xa0, 0x2d, 0xcc, 0x6f, 0xb5, 0xdf, 0xfc, 0xed, 0xf5, + 0x90, 0xcb, 0xb3, 0x68, 0xd0, 0xb0, 0x83, 0x51, 0x53, 0x07, 0x08, 0xc4, 0x30, 0x7e, 0x68, 0x66, + 0xff, 0x3c, 0x43, 0xf4, 0x9b, 0xe3, 0xc1, 0x6f, 0x87, 0x41, 0x73, 0xe6, 0x37, 0x68, 0xb0, 0xa2, + 0x07, 0xe5, 0xf3, 0x9f, 0x03, 0x00, 0x00, 0xff, 0xff, 0x84, 0xd9, 0xfd, 0xdc, 0x1c, 0x0e, 0x00, + 0x00, } diff --git a/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.validate.go b/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.validate.go index cf7a679a1f5..7102112d216 100644 --- a/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.validate.go +++ b/flyteidl/gen/pb-go/flyteidl/core/workflow.pb.validate.go @@ -1549,6 +1549,16 @@ func (m *TaskNodeOverrides) Validate() error { } } + if v, ok := interface{}(m.GetExtendedResources()).(interface{ Validate() error }); ok { + if err := v.Validate(); err != nil { + return TaskNodeOverridesValidationError{ + field: "ExtendedResources", + reason: "embedded message failed validation", + cause: err, + } + } + } + return nil } diff --git a/flyteidl/gen/pb-go/flyteidl/service/admin.swagger.json b/flyteidl/gen/pb-go/flyteidl/service/admin.swagger.json index 8c1ebc04d67..655512db165 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/admin.swagger.json +++ b/flyteidl/gen/pb-go/flyteidl/service/admin.swagger.json @@ -6031,6 +6031,34 @@ }, "description": "Represents the error message from the execution." }, + "coreExtendedResources": { + "type": "object", + "properties": { + "gpu_accelerator": { + "$ref": "#/definitions/coreGPUAccelerator", + "description": "GPU accelerator to select for task. Contains information about device type, and\nfor multi-instance GPUs, the partition size to use." + } + }, + "description": "Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to\nallocate to a task." + }, + "coreGPUAccelerator": { + "type": "object", + "properties": { + "device": { + "type": "string", + "description": "This can be any arbitrary string, and should be informed by the labels or taints\nassociated with the nodes in question. Default cloud provider labels typically\nuse the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc." + }, + "unpartitioned": { + "type": "boolean", + "format": "boolean" + }, + "partition_size": { + "type": "string", + "description": "Like `device`, this can be any arbitrary string, and should be informed by\nthe labels or taints associated with the nodes in question. Default cloud\nprovider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc." + } + }, + "description": "Metadata associated with the GPU accelerator to allocate to a task. Contains\ninformation about device type, and for multi-instance GPUs, the partition size to\nuse." + }, "coreGateNode": { "type": "object", "properties": { @@ -7014,6 +7042,10 @@ "resources": { "$ref": "#/definitions/coreResources", "description": "A customizable interface to convey resources requested for a task container." + }, + "extended_resources": { + "$ref": "#/definitions/coreExtendedResources", + "description": "Overrides for all non-standard resources, not captured by\nv1.ResourceRequirements, to allocate to a task." } }, "description": "Optional task node overrides that will be applied at task execution time." @@ -7059,6 +7091,10 @@ "$ref": "#/definitions/coreSecurityContext", "description": "security_context encapsulates security attributes requested to run this task." }, + "extended_resources": { + "$ref": "#/definitions/coreExtendedResources", + "description": "Encapsulates all non-standard resources, not captured by\nv1.ResourceRequirements, to allocate to a task." + }, "config": { "type": "object", "additionalProperties": { diff --git a/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json b/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json index 2d1d5ea07bd..4149506f171 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json +++ b/flyteidl/gen/pb-go/flyteidl/service/agent.swagger.json @@ -369,6 +369,34 @@ }, "description": "Represents an error thrown from a node." }, + "coreExtendedResources": { + "type": "object", + "properties": { + "gpu_accelerator": { + "$ref": "#/definitions/coreGPUAccelerator", + "description": "GPU accelerator to select for task. Contains information about device type, and\nfor multi-instance GPUs, the partition size to use." + } + }, + "description": "Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to\nallocate to a task." + }, + "coreGPUAccelerator": { + "type": "object", + "properties": { + "device": { + "type": "string", + "description": "This can be any arbitrary string, and should be informed by the labels or taints\nassociated with the nodes in question. Default cloud provider labels typically\nuse the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc." + }, + "unpartitioned": { + "type": "boolean", + "format": "boolean" + }, + "partition_size": { + "type": "string", + "description": "Like `device`, this can be any arbitrary string, and should be informed by\nthe labels or taints associated with the nodes in question. Default cloud\nprovider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc." + } + }, + "description": "Metadata associated with the GPU accelerator to allocate to a task. Contains\ninformation about device type, and for multi-instance GPUs, the partition size to\nuse." + }, "coreIOStrategy": { "type": "object", "properties": { @@ -1020,6 +1048,10 @@ "$ref": "#/definitions/coreSecurityContext", "description": "security_context encapsulates security attributes requested to run this task." }, + "extended_resources": { + "$ref": "#/definitions/coreExtendedResources", + "description": "Encapsulates all non-standard resources, not captured by\nv1.ResourceRequirements, to allocate to a task." + }, "config": { "type": "object", "additionalProperties": { diff --git a/flyteidl/gen/pb-go/flyteidl/service/external_plugin_service.swagger.json b/flyteidl/gen/pb-go/flyteidl/service/external_plugin_service.swagger.json index bb0e406339c..ae3af727ba1 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/external_plugin_service.swagger.json +++ b/flyteidl/gen/pb-go/flyteidl/service/external_plugin_service.swagger.json @@ -328,6 +328,34 @@ }, "description": "Represents an error thrown from a node." }, + "coreExtendedResources": { + "type": "object", + "properties": { + "gpu_accelerator": { + "$ref": "#/definitions/coreGPUAccelerator", + "description": "GPU accelerator to select for task. Contains information about device type, and\nfor multi-instance GPUs, the partition size to use." + } + }, + "description": "Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to\nallocate to a task." + }, + "coreGPUAccelerator": { + "type": "object", + "properties": { + "device": { + "type": "string", + "description": "This can be any arbitrary string, and should be informed by the labels or taints\nassociated with the nodes in question. Default cloud provider labels typically\nuse the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc." + }, + "unpartitioned": { + "type": "boolean", + "format": "boolean" + }, + "partition_size": { + "type": "string", + "description": "Like `device`, this can be any arbitrary string, and should be informed by\nthe labels or taints associated with the nodes in question. Default cloud\nprovider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc." + } + }, + "description": "Metadata associated with the GPU accelerator to allocate to a task. Contains\ninformation about device type, and for multi-instance GPUs, the partition size to\nuse." + }, "coreIOStrategy": { "type": "object", "properties": { @@ -951,6 +979,10 @@ "$ref": "#/definitions/coreSecurityContext", "description": "security_context encapsulates security attributes requested to run this task." }, + "extended_resources": { + "$ref": "#/definitions/coreExtendedResources", + "description": "Encapsulates all non-standard resources, not captured by\nv1.ResourceRequirements, to allocate to a task." + }, "config": { "type": "object", "additionalProperties": { diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/README.md b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/README.md index 14173a7992f..863164c8d5d 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/README.md +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/README.md @@ -240,7 +240,9 @@ Class | Method | HTTP request | Description - [CoreEnumType](docs/CoreEnumType.md) - [CoreError](docs/CoreError.md) - [CoreExecutionError](docs/CoreExecutionError.md) + - [CoreExtendedResources](docs/CoreExtendedResources.md) - [CoreGateNode](docs/CoreGateNode.md) + - [CoreGpuAccelerator](docs/CoreGpuAccelerator.md) - [CoreIdentifier](docs/CoreIdentifier.md) - [CoreIdentity](docs/CoreIdentity.md) - [CoreIfBlock](docs/CoreIfBlock.md) diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/api/swagger.yaml b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/api/swagger.yaml index d38070fe5f7..53e03d1e83f 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/api/swagger.yaml +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/api/swagger.yaml @@ -9241,6 +9241,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -10381,6 +10386,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -11520,6 +11530,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -13140,6 +13155,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -14280,6 +14300,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -15419,6 +15444,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -15635,6 +15665,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -15898,6 +15933,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -17567,6 +17607,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -18707,6 +18752,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -19846,6 +19896,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -21114,6 +21169,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -21393,6 +21453,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -22085,6 +22150,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -22358,6 +22428,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -24141,6 +24216,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -25281,6 +25361,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -26420,6 +26505,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -28040,6 +28130,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -29180,6 +29275,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -30319,6 +30419,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -30535,6 +30640,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -30798,6 +30908,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -32467,6 +32582,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -33607,6 +33727,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -34746,6 +34871,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -36651,6 +36781,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -37791,6 +37926,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -38930,6 +39070,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -40550,6 +40695,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -41690,6 +41840,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -42829,6 +42984,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -43045,6 +43205,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -43308,6 +43473,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -44977,6 +45147,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -46117,6 +46292,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -47256,6 +47436,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -49108,6 +49293,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -50248,6 +50438,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -51387,6 +51582,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -53007,6 +53207,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -54147,6 +54352,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -55286,6 +55496,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -55502,6 +55717,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -55765,6 +55985,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -57434,6 +57659,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -58574,6 +58804,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -59713,6 +59948,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -61344,6 +61584,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -62484,6 +62729,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -63623,6 +63873,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -65243,6 +65498,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -66383,6 +66643,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -67522,6 +67787,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -67738,6 +68008,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -68001,6 +68276,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -69670,6 +69950,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -70810,6 +71095,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -71949,6 +72239,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -73946,6 +74241,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -75627,6 +75927,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -76767,6 +77072,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -77906,6 +78216,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -79553,6 +79868,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -80693,6 +81013,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -81832,6 +82157,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -83452,6 +83782,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -84592,6 +84927,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -85731,6 +86071,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -85947,6 +86292,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -86210,6 +86560,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -87879,6 +88234,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -89019,6 +89379,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -90158,6 +90523,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -90554,6 +90924,45 @@ definitions: kind: {} message: "message" error_uri: "error_uri" + coreExtendedResources: + type: "object" + properties: + gpu_accelerator: + description: "GPU accelerator to select for task. Contains information about\ + \ device type, and\nfor multi-instance GPUs, the partition size to use." + $ref: "#/definitions/coreGPUAccelerator" + description: "Encapsulates all non-standard resources, not captured by v1.ResourceRequirements,\ + \ to\nallocate to a task." + example: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" + coreGPUAccelerator: + type: "object" + properties: + device: + type: "string" + description: "This can be any arbitrary string, and should be informed by\ + \ the labels or taints\nassociated with the nodes in question. Default cloud\ + \ provider labels typically\nuse the following values: `nvidia-tesla-t4`,\ + \ `nvidia-tesla-a100`, etc." + unpartitioned: + type: "boolean" + format: "boolean" + partition_size: + type: "string" + description: "Like `device`, this can be any arbitrary string, and should\ + \ be informed by\nthe labels or taints associated with the nodes in question.\ + \ Default cloud\nprovider labels typically use the following values: `1g.5gb`,\ + \ `2g.10gb`, etc." + description: "Metadata associated with the GPU accelerator to allocate to a task.\ + \ Contains\ninformation about device type, and for multi-instance GPUs, the\ + \ partition size to\nuse." + example: + partition_size: "partition_size" + unpartitioned: true + device: "device" coreGateNode: type: "object" properties: @@ -93141,6 +93550,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -94583,6 +94997,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -94601,9 +95020,18 @@ definitions: description: "A customizable interface to convey resources requested for a\ \ task container." $ref: "#/definitions/coreResources" + extended_resources: + description: "Overrides for all non-standard resources, not captured by\n\ + v1.ResourceRequirements, to allocate to a task." + $ref: "#/definitions/coreExtendedResources" description: "Optional task node overrides that will be applied at task execution\ \ time." example: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -94655,6 +95083,10 @@ definitions: description: "security_context encapsulates security attributes requested\ \ to run this task." $ref: "#/definitions/coreSecurityContext" + extended_resources: + description: "Encapsulates all non-standard resources, not captured by\nv1.ResourceRequirements,\ + \ to allocate to a task." + $ref: "#/definitions/coreExtendedResources" config: type: "object" title: "Metadata about the custom defined for this task. This is extensible\ @@ -94724,6 +95156,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -96956,6 +97393,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -98096,6 +98538,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -99235,6 +99682,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -101155,6 +101607,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -102295,6 +102752,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -103434,6 +103896,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -105054,6 +105521,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -106194,6 +106666,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -107333,6 +107810,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -107549,6 +108031,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -107812,6 +108299,11 @@ definitions: tags: key: "tags" task_type_version: 7 + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" custom: fields: key: @@ -109481,6 +109973,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -110621,6 +111118,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} @@ -111760,6 +112262,11 @@ definitions: project: "project" version: "version" overrides: + extended_resources: + gpu_accelerator: + partition_size: "partition_size" + unpartitioned: true + device: "device" resources: requests: - name: {} diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_extended_resources.go b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_extended_resources.go new file mode 100644 index 00000000000..bb02119b08e --- /dev/null +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_extended_resources.go @@ -0,0 +1,16 @@ +/* + * flyteidl/service/admin.proto + * + * No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) + * + * API version: version not set + * Generated by: Swagger Codegen (https://github.com/swagger-api/swagger-codegen.git) + */ + +package flyteadmin + +// Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. +type CoreExtendedResources struct { + // GPU accelerator to select for task. Contains information about device type, and for multi-instance GPUs, the partition size to use. + GpuAccelerator *CoreGpuAccelerator `json:"gpu_accelerator,omitempty"` +} diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_gpu_accelerator.go b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_gpu_accelerator.go new file mode 100644 index 00000000000..6be738d59f5 --- /dev/null +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_gpu_accelerator.go @@ -0,0 +1,19 @@ +/* + * flyteidl/service/admin.proto + * + * No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) + * + * API version: version not set + * Generated by: Swagger Codegen (https://github.com/swagger-api/swagger-codegen.git) + */ + +package flyteadmin + +// Metadata associated with the GPU accelerator to allocate to a task. Contains information about device type, and for multi-instance GPUs, the partition size to use. +type CoreGpuAccelerator struct { + // This can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. + Device string `json:"device,omitempty"` + Unpartitioned bool `json:"unpartitioned,omitempty"` + // Like `device`, this can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc. + PartitionSize string `json:"partition_size,omitempty"` +} diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_node_overrides.go b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_node_overrides.go index 241ad411974..7717000d4e1 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_node_overrides.go +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_node_overrides.go @@ -13,4 +13,6 @@ package flyteadmin type CoreTaskNodeOverrides struct { // A customizable interface to convey resources requested for a task container. Resources *CoreResources `json:"resources,omitempty"` + // Overrides for all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. + ExtendedResources *CoreExtendedResources `json:"extended_resources,omitempty"` } diff --git a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_template.go b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_template.go index 163a0eb7c8a..64a4a984a8d 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_template.go +++ b/flyteidl/gen/pb-go/flyteidl/service/flyteadmin/model_core_task_template.go @@ -28,5 +28,7 @@ type CoreTaskTemplate struct { TaskTypeVersion int32 `json:"task_type_version,omitempty"` // security_context encapsulates security attributes requested to run this task. SecurityContext *CoreSecurityContext `json:"security_context,omitempty"` + // Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. + ExtendedResources *CoreExtendedResources `json:"extended_resources,omitempty"` Config map[string]string `json:"config,omitempty"` } diff --git a/flyteidl/gen/pb-go/flyteidl/service/openapi.go b/flyteidl/gen/pb-go/flyteidl/service/openapi.go index 944147b4531..a19d4ff07ba 100644 --- a/flyteidl/gen/pb-go/flyteidl/service/openapi.go +++ b/flyteidl/gen/pb-go/flyteidl/service/openapi.go @@ -78,7 +78,7 @@ func (fi bindataFileInfo) Sys() interface{} { return nil } -var _adminSwaggerJson = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xfd\xf9\x73\x2b\xb7\x95\x37\x8c\xff\x3e\x7f\x05\x9e\x3b\x4f\x95\xed\x44\x8b\x93\xcc\xe4\x9b\xd2\xd4\x53\xdf\x97\x96\x78\xaf\xf5\x58\x5b\xb4\xd8\xe3\x77\x98\xa2\xc1\x6e\x90\x44\xd4\x0d\x74\x00\xb4\x74\xe9\x54\xfe\xf7\xb7\x70\xb0\x34\x7a\x23\x9b\x8b\x24\xea\xba\x67\xaa\x62\x5d\x76\x37\xd6\x83\x83\xb3\x7e\xce\x3f\xff\x0d\xa1\x0f\xf2\x19\xcf\x66\x44\x7c\x38\x41\x1f\xfe\x78\xf4\xed\x87\x03\xfd\x1b\x65\x53\xfe\xe1\x04\xe9\xe7\x08\x7d\x50\x54\x25\x44\x3f\x9f\x26\x0b\x45\x68\x9c\x1c\x4b\x22\x9e\x68\x44\x8e\x71\x9c\x52\x76\x94\x09\xae\x38\x7c\x88\xd0\x87\x27\x22\x24\xe5\x4c\xbf\x6e\xff\x44\x8c\x2b\x24\x89\xfa\xf0\x6f\x08\xfd\x0b\x9a\x97\xd1\x9c\xa4\x44\x7e\x38\x41\xff\x63\x3e\x9a\x2b\x95\xb9\x06\xf4\xdf\x52\xbf\xfb\x37\x78\x37\xe2\x4c\xe6\xa5\x97\x71\x96\x25\x34\xc2\x8a\x72\x76\xfc\x77\xc9\x59\xf1\x6e\x26\x78\x9c\x47\x1d\xdf\xc5\x6a\x2e\x8b\x39\x1e\xe3\x8c\x1e\x3f\xfd\xe1\x18\x47\x8a\x3e\x91\x71\x82\x73\x16\xcd\xc7\x59\x82\x99\x3c\xfe\x27\x8d\xf5\x1c\xff\x4e\x22\xf5\x2f\xf8\x47\xcc\x53\x4c\x99\xf9\x9b\xe1\x94\xfc\xcb\xb7\x83\xd0\x87\x19\x51\xc1\x3f\xf5\x6c\xf3\x34\xc5\x62\xa1\x57\xe4\x23\x51\xd1\x1c\xa9\x39\x41\xa6\x1f\xe4\x96\x88\x4f\x11\x46\x27\x82\x4c\x4f\x7e\x11\x64\x3a\x76\x0b\x7d\x64\x16\xf8\x02\x46\x73\x93\x60\xf6\xcb\x91\x5d\x26\x68\x99\x67\x44\xc0\xdc\xce\x63\xdd\xfa\x27\xa2\x06\xd0\x6c\xf1\x7e\xf8\xb6\x20\x32\xe3\x4c\x12\x59\x1a\x1e\x42\x1f\xfe\xf8\xed\xb7\x95\x9f\x10\xfa\x10\x13\x19\x09\x9a\x29\xbb\x97\x03\x24\xf3\x28\x22\x52\x4e\xf3\x04\xb9\x96\xc2\xc1\x98\xa9\xea\x8d\xc5\xb5\xc6\x10\xfa\xf0\xbf\x05\x99\xea\x76\xfe\xfd\x38\x26\x53\xca\xa8\x6e\x57\x1a\xfa\x09\x46\x5b\xfa\xea\x5f\xff\xd6\xf4\xf7\xbf\x82\x19\x65\x58\xe0\x94\x28\x22\x8a\x1d\x37\xff\x57\x99\x8b\xde\x23\xdd\x79\xb1\x8f\xd5\x81\x57\x66\x7b\x85\x53\xa2\xf7\x44\xef\x94\xfd\x02\xfe\x16\x44\xf2\x5c\x44\x04\x4d\x48\xc2\xd9\x4c\x22\xc5\x6b\x6b\x40\xa1\x05\x4d\x5e\xd5\x27\x82\xfc\x23\xa7\x82\xe8\xbd\x52\x22\x27\x95\xa7\x6a\x91\xc1\x20\xa5\x12\x94\xcd\xc2\xa5\xf8\xd7\x41\xa7\xa9\x19\xaa\x5c\x63\x66\xe6\x83\xd6\x89\x8d\xd8\xc0\xbd\x12\x61\x86\x26\x04\xe9\xb3\x48\x63\x22\x48\x8c\xb0\x44\x18\xc9\x7c\x22\x89\x42\xcf\x54\xcd\x29\xd3\xff\xce\x48\x44\xa7\x34\x72\x6b\xb6\x3f\x6b\x03\x7f\x2e\x5f\x99\x07\x49\x84\x1e\xf8\x13\x8d\x49\x8c\x9e\x70\x92\x13\x34\xe5\xa2\xb4\x3c\x47\x23\x76\x3f\xd7\xeb\x90\x4e\x28\x83\x93\xa7\xd7\xd2\x51\xc8\xef\xdd\x72\xfd\x1e\xe9\xfe\x50\xce\xe8\x3f\x72\x92\x2c\x10\x8d\x09\x53\x74\x4a\x89\xac\xb6\xf6\x7b\x0e\xfd\xe3\x04\x1d\x22\xbd\xce\x44\x28\x58\x6f\xce\x14\xf9\xac\x24\x3a\x44\x09\x7d\x24\xe8\xab\x0b\x2a\x15\x1a\xdc\x9c\x7f\x75\x80\xbe\x32\xe7\x05\x01\x6f\xfa\xea\x15\x56\xd8\xff\xfd\xb7\xe0\xe8\x29\x3c\xab\x1e\xba\x0f\x03\x7d\x9a\xef\xcc\xd5\x50\xb4\xf0\xb7\x7f\x0b\xdb\xb1\xfb\xb5\x9c\xdf\x16\xcc\xd6\x72\xda\xae\xfc\x15\x96\xa9\xcc\x5a\xa5\xde\xa1\x6d\x39\xab\x6e\xb7\xca\x5a\xe5\xfb\xe2\xad\x7a\x0a\x2f\xcd\x5f\xb7\x61\xae\x58\x01\xd5\x63\xca\xcc\x21\xf1\x67\x46\x48\x7d\x4e\x1c\xf5\xee\x09\x4b\xd9\x86\xd7\x06\x33\x0b\xd8\xad\xe3\xa2\xc1\xaa\xec\xe1\xbc\x13\x9a\xd2\x55\xfb\x7b\xce\x62\x2d\x72\x59\x66\xc7\xf2\x74\x42\x84\x5e\x06\xc7\xf6\x60\xb6\x13\xcd\x06\x55\x2e\x18\x89\x3b\x4c\xf3\x1f\x39\x11\x8b\x25\xf3\x9c\xe2\x44\xb6\x4d\x94\x32\x45\xb4\x7c\x5b\x79\x3c\xe5\x22\xc5\xca\xbe\xf0\xe7\xff\x58\x77\x21\x14\x7f\x24\xab\xf6\xff\xdc\xec\x66\x84\x25\x90\x41\x9a\x27\x8a\x66\x09\x41\x19\x9e\x11\x69\x57\x24\x4f\x94\x3c\x80\xd7\xb4\x4c\x4d\xc4\xa1\xbf\x81\xa0\x07\x77\xf3\xe6\x12\x7e\x41\x53\x2f\x40\x32\xf2\x59\x41\x4b\x23\x06\x77\x2f\x2c\x51\x78\xa3\xbc\xc0\x52\x6e\x46\x33\x92\x0b\x35\x9e\x2c\x8e\x1e\x49\xad\xdf\x56\xca\xc1\x0c\x61\xa5\x04\x9d\xe4\x8a\xe8\x79\xeb\x36\xdc\xdd\x09\xec\xd1\x5c\xd0\x5d\x58\xc3\xdb\x4d\x38\xa6\x82\x44\x30\xb7\x75\x0e\x8c\xff\x4a\xcf\x5b\xeb\x2f\x0b\x33\xfb\x47\xb2\x00\x79\xa4\x61\x05\xfc\x96\x8f\xd8\x88\xa1\x43\x74\x36\xbc\x3b\x1d\x5e\x9d\x9d\x5f\x7d\x3a\x41\xdf\x2d\x50\x4c\xa6\x38\x4f\xd4\x01\x9a\x52\x92\xc4\x12\x61\x41\xa0\x49\x12\x6b\x99\x43\x0f\x86\xb0\x98\xb2\x19\xe2\x22\x26\xe2\xe5\x96\xb1\xf2\x94\xb0\x3c\xad\xdc\x2b\xf0\x7b\x31\xfa\xca\x17\x5a\xc4\xf0\x8f\x4a\x4f\xfe\x56\x5b\x60\x98\xb1\xee\x3b\x68\xed\xd5\x84\x9a\x68\x4e\x93\x58\x10\x76\xac\xb0\x7c\x1c\x93\xcf\x24\xca\xcd\x9d\xfc\xcf\xf2\x0f\x63\x2d\x99\xf2\x98\x94\x7f\x29\xfd\xa3\x10\x85\xd6\xfe\xd4\x6b\xa9\x6b\x7f\x09\x3a\x6d\xb7\xef\xe0\x17\x1a\x37\xbe\x0d\xbf\xac\x98\x83\x7b\x67\xc9\x60\xdd\x2b\xad\xa3\x72\x2f\x58\x89\xaf\xf1\x1d\x41\x94\x58\x8c\xb1\x52\x24\xcd\xd4\x9a\xfa\x3a\x46\x89\x96\x2b\x97\xc9\x91\x57\x3c\x26\x43\xd7\xdf\x2f\xc8\x88\xb3\x24\x46\x93\x85\xe5\x5a\x53\x22\x08\x8b\x48\x7b\x0b\xf7\x58\x3e\x16\x2d\xac\x12\x46\x4b\xfd\xc9\x8f\x5c\xe8\xcf\xdf\x83\x40\x5a\x1a\xf8\x6b\xc8\xa4\x9b\x9e\xb8\x2f\xce\x42\xb0\x21\xff\xe8\xed\x09\xdb\xaf\x64\x57\xeb\x03\x17\x48\x2e\xa4\x22\xe9\x4a\x3b\xc4\xfb\x59\x08\x7b\x41\xec\xeb\x80\x2b\x77\xd4\x6f\xe0\xd4\x97\x6f\xdc\xfe\x78\xaf\xb1\x64\xbb\xb2\x22\xee\xfb\x3c\x9d\x0f\x67\xf9\x54\xef\xdc\xf6\x05\x4e\x8c\x77\x31\xcd\x92\x2c\xb8\xeb\x41\xbe\x90\xb9\xa1\x75\xaf\xdc\x6a\x8f\x61\x00\x2b\x14\xcd\xb2\x1d\xda\x9f\x3f\xfd\x69\x68\xa1\x31\xe6\x38\x35\xa7\x32\x30\x56\xa1\x88\x0b\x23\x0b\xc6\xf6\xbc\x1b\x5d\x73\x70\x3f\xb8\x1b\xde\x9f\xa0\x01\x8a\xb1\xc2\xfa\x80\x0b\x92\x09\x22\x09\x53\xa0\xc7\xeb\xef\xd5\x02\xa5\x3c\x26\x89\xd1\x38\x3f\x6a\xc9\x17\x9d\x61\x85\x4f\xb1\xc2\x09\x9f\x1d\xa1\x01\xfc\x53\x7f\x4c\x25\xc2\x89\xe4\x08\x3b\xb2\x22\xb1\x6b\x02\xb3\xd8\xb1\x16\x8c\x22\x9e\x66\x34\xf1\x36\x78\x6f\x5c\xa1\x2c\xa6\x4f\x34\xce\x71\x82\xf8\x44\x73\x15\xad\x21\x0f\x9f\x08\x53\x39\x4e\x92\x05\xc2\x49\x82\x6c\xb7\xee\x05\x24\xe7\x3c\x4f\x62\xdd\xae\x1b\xa5\xa4\x29\x4d\xb0\xd0\x2a\xb8\x19\xed\xb5\x6d\x0b\xdd\xcf\x89\x1f\x2b\x8c\x4b\xaf\x66\x8a\x1f\x89\x44\x54\xa1\x8c\x4b\x49\x27\x49\x71\xe6\x1f\xce\x11\x8c\xfb\xf4\xe2\x1c\xf4\xf9\x48\x21\x6e\x78\xa8\xeb\xdc\xda\x6f\x5c\x8f\x29\x66\x8c\x40\xc7\x5c\xcd\x89\xb0\xdd\xdb\x97\xdf\x5a\x35\x7f\xb8\xba\xbb\x19\x9e\x9e\x7f\x3c\x1f\x9e\xd5\x75\xf3\xfb\xc1\xdd\x0f\xf5\x5f\x7f\xba\xbe\xfd\xe1\xe3\xc5\xf5\x4f\xf5\x27\x17\x83\x87\xab\xd3\xef\xc7\x37\x17\x83\xab\xfa\x43\x4b\x56\x9d\xd5\xfc\x70\x64\x6b\x9e\xad\xde\xa6\xf9\x52\x36\xcd\x83\x2f\xd7\xa8\x39\xa5\x09\xe8\xa0\x9d\x0d\x9a\xde\x86\x60\xbf\x44\x19\x96\xd2\x48\x46\x66\x04\x47\x23\x76\xc9\x85\x66\x60\x53\xae\x79\x84\x96\x9e\x94\xc8\x23\x45\xd9\xcc\x7f\x74\x82\x46\xf9\xb7\xdf\xfe\x29\xba\xa0\xec\x11\xfe\x22\xfb\xb8\x38\xbd\xc5\xb7\xb7\xf8\xfe\xb6\x2c\xbe\x5a\xf4\x39\x0e\x0d\xbd\xbb\x0d\x19\xd2\xc2\x05\xcb\x72\x05\xa2\x04\xcf\x95\xfe\x53\x77\x09\xe4\xb1\x24\x70\xa8\x9b\x41\xf1\x13\x51\xfe\x45\x2d\xda\xbc\x07\x3b\xe2\x4f\x5c\x3c\x4e\x13\xfe\xec\x07\xfe\x89\x28\x3d\xf6\x5b\xdb\x4b\x1f\x4a\xd4\x87\x12\xbd\x6d\x28\xd1\x5e\x19\xf3\x5e\x9e\xf9\x95\x2d\x7f\x86\x03\xb6\x78\xb2\x5a\x1d\x55\x2d\x7e\xa8\xc0\xcd\xf4\x2a\x5c\xb3\xec\xcc\x59\xc1\x39\x4b\x2f\xbf\x17\xee\x59\x1a\xf4\xeb\x73\xce\xdf\x84\xbf\xa5\x77\xa7\x6c\xb8\x50\xef\x92\xc1\x76\xbc\x3b\x5e\xcd\x19\xf2\xf2\x0c\xbf\x16\xdb\xb0\x4e\x30\xc3\x1a\xd1\x0b\x9d\xc3\x15\x56\xc4\x27\x34\x06\x24\x34\x45\x20\xd4\x43\x0e\x1a\x63\x0c\xb6\x0b\x2a\xd8\xf4\x6e\xea\x1e\x26\xf0\x89\xa8\xd2\xcb\xef\xe5\x6e\x2a\x0d\xfa\xf5\xef\xa6\xdf\x68\x74\x40\x1f\x0e\xf0\x82\x4b\xf7\xa5\xdf\x68\xfb\xeb\xf0\xff\x0d\x78\xf8\x7b\x97\xfe\x5a\x6b\xf4\x65\xf9\xf0\xbf\x54\xa7\xfd\xfb\xf4\xd2\xf7\x6e\xf9\xde\x2d\xff\x16\xfe\x93\xf7\xe7\x96\x7f\x59\xf5\xb4\x38\x5e\x63\x47\x0b\x56\x5f\x0b\x0e\xe5\xbf\x3a\x38\x69\xe0\x2f\xa7\xf2\xad\x1b\x34\xde\xaa\xc3\x9d\x15\xe3\x1b\xc2\x11\xfa\xc5\x12\xd2\x0a\x75\xae\xf6\xdd\x7b\x50\xe7\xea\x83\x7e\x79\x1d\xee\xcd\x98\xef\x0b\x5d\x9e\xef\x84\x0d\xac\x7f\x5b\x7e\xc1\x32\x79\x2f\x8b\xbf\x7c\x36\xfe\xde\x4c\xe8\xfd\xc8\xde\x6f\x70\xf1\x76\xbc\x75\x77\x9e\x93\xd5\x70\xcd\x06\xb7\xd3\xaa\x0c\xab\xea\xd7\x94\xc8\x3f\xbe\xcb\xfb\xf6\x35\x92\xac\xfa\x0b\xb7\xbf\x70\x6d\x53\xfd\x85\xfb\x05\x5f\xb8\x7b\x07\x7f\xb3\x37\x11\xa0\x7d\x10\x79\x0f\x8c\xd1\xc7\x90\xef\x70\x71\xfa\x18\xf2\x3e\x86\xfc\x37\x16\x43\xbe\x8d\xf6\xb4\x29\x16\xe5\x5b\xe8\x51\xbd\x1a\xd5\xab\x51\xe1\xef\xbd\x1a\xd5\xab\x51\xbd\x1a\xf5\x85\xa3\x88\xf6\x3a\x54\xf7\x85\xe8\x75\xa8\xce\x4b\xd5\xeb\x50\x4b\x16\xa7\xd7\xa1\x7a\x1d\xea\xb7\xa5\x43\x91\x27\xc2\x94\x84\x64\xb4\x50\xa3\xf8\x90\x71\xd9\xae\x09\x85\xdc\xa1\x41\x0b\x82\x36\xcb\x49\x61\x10\xb8\xf4\x0b\x9a\x63\x89\x78\x14\xe5\xa2\x72\x06\xaa\x7a\xd0\xa9\x20\x58\x11\x68\x41\x7f\xf8\x1e\xf4\x9f\xfa\x74\x5f\x2b\x06\x7f\xc2\xe3\x1a\xb5\x9b\x83\xd0\xf4\x64\xb9\x3c\xb2\xb3\xa9\xff\x23\x27\xdd\xd4\xbf\x17\x24\x6a\x85\xe5\xe3\x8e\x89\xba\x94\x6b\xb1\x11\x51\x43\x0b\xef\x85\xa8\xeb\xd3\xfd\xcd\x10\x75\xd3\xd4\xf7\x81\xa8\x9f\x6d\x1e\xff\x8e\x09\xbb\x06\x0f\xb0\x11\x71\xfb\x56\xde\x0b\x81\x37\x4f\xfb\x37\x43\xe4\x6d\xd3\x7f\x5b\x42\xf7\x29\x92\x9d\x49\xfc\x5e\xd0\xd9\x4c\xab\x19\xa0\xe1\x69\x52\x5c\x5d\x23\xa8\x48\x0a\x5c\x49\xd6\xfe\xd5\xf7\x40\xd2\x7e\xb0\x66\xec\xbf\x19\x5a\xae\xcd\x7b\x4f\x88\xf8\x58\x90\x88\x3f\x41\xbd\xb0\x6e\xc4\x7c\x4b\x80\x82\x81\x5f\x67\x82\x3c\x51\x9e\xcb\x64\x71\x28\x72\x86\x1c\xf3\x47\xbe\x79\x63\xad\x7e\xa6\x49\x82\x38\xd3\xfa\x97\xc2\x42\xb9\xc7\x5a\xff\x16\x3c\x85\x53\x91\x60\xa9\xd0\x23\xe3\xcf\x0c\x4d\x31\x4d\x72\x41\x50\xc6\x29\x53\x47\x23\x76\xce\xd0\xad\x19\x23\xe4\x0d\x1c\xa0\x5c\xea\xb3\x14\x61\xc6\xb8\x42\xd1\x1c\xb3\x19\x41\x98\x2d\x6c\x02\x6e\x41\x19\x88\x0b\x94\x67\x31\xd6\x8a\xef\x9c\x54\xa3\xf4\xfc\x18\xc1\x7c\x47\x25\xa2\x12\x91\xcf\x4a\x90\x94\x24\x0b\xdd\x87\xa6\x7d\xc5\x91\x5d\x1f\x33\x54\x9b\xce\x47\x84\xe0\x42\x42\xc6\xc1\x64\xf1\x2b\x66\x8a\x32\x82\x40\x51\x92\xc6\x34\x77\x88\x2e\xb8\x04\xb3\xcd\x0f\x7f\x91\x28\x4a\x72\xa9\x88\x38\x40\x93\x7c\x26\xb5\xa6\x98\x25\x58\x4d\xb9\x48\xf5\x08\x29\x93\x0a\x4f\x68\x42\xd5\xe2\x00\xa5\x38\x9a\x9b\xb6\x60\x0d\xe4\xc1\x88\xc5\xfc\x99\x49\x25\x08\xf6\xbd\xbb\x87\xe8\xeb\xf0\x99\x21\x00\xf9\xcd\x01\xa4\x1d\xd2\x54\xab\xbb\xc1\xf0\x8b\x1d\x37\x7b\xa2\x1b\x21\x31\x9a\x90\x08\xe7\xd2\x7a\x18\x94\x58\x20\xf2\x79\x8e\x73\x09\x7b\xa7\xa7\x67\x73\x36\x22\x9e\x66\x09\x51\x04\xd1\x29\x52\x82\x92\x18\xe1\x19\xa6\x7a\xe9\xee\xc8\x12\x10\x74\x4f\xf4\x76\x03\x2d\xd5\xff\x02\xea\x77\xca\x05\x41\x31\x51\x98\x26\x4b\xbd\x4e\xf6\xdb\x9e\xcb\xbd\x27\x2e\x57\xde\xf0\xbd\x60\x73\x06\xc4\x7f\x07\x97\x36\xb3\xa6\xfb\x08\x27\x5b\xde\xdf\xb7\x76\x50\x3d\x6d\xbf\x2f\xda\x36\xbb\xb6\x3f\xc4\xfd\x6a\x31\xd8\xdd\x2b\x5a\x14\xd5\x2c\xde\x15\x4d\xbf\x46\x58\x40\xef\x70\xee\x1d\xce\xad\x2b\xf3\x3e\x1d\xce\x7b\xe3\x31\xea\x7d\xce\x2f\xe4\x73\xa6\xb2\x77\x3a\xf7\x4e\xe7\xae\x0b\xd4\x3b\x9d\x7b\xa7\xf3\xfb\x75\x3a\xbf\x24\xee\xf3\x4e\xd1\x9d\xdf\x95\x68\xdd\x8b\xd5\xbd\x58\xdd\x43\x38\xfb\xa9\xed\x8a\x85\xb9\xaf\x3f\xc4\x24\x21\x8a\xb4\xdb\xb3\x88\x48\xb5\xb6\x60\xae\x67\xca\xb4\x1c\x37\x13\x44\xca\x6d\x19\x92\x6f\xf8\x7d\xb2\x25\x3f\xfc\x1e\x6a\xbe\xe7\x53\x3d\x9f\xda\x64\x6a\xfb\x63\x9a\x0d\x0e\xf3\x6b\xd9\x66\x3d\xff\xcd\xf2\x76\xe9\xef\xc1\xb8\x21\x0b\xbf\xa8\xa1\x70\x2d\xb5\x2b\xee\x0f\xb7\xa5\xf3\x2d\xf9\xb1\xe9\xeb\x7d\x32\x63\x33\xf6\x9e\x13\xf7\x9c\xb8\xe7\xc4\xef\x9b\x13\xbb\x93\xfc\xa6\x2e\x32\xe3\xa7\x1b\x67\x09\x66\x63\x1a\xcb\xe3\x7f\x16\xba\xfc\x4b\x79\xc8\xf4\x81\x8a\x4d\x8a\xa9\x4f\xe9\x14\xbf\xe8\x4f\x92\xc2\x60\xee\x31\x33\x57\x38\xd1\x8c\x8d\xfd\x26\xc1\xec\x3c\x7e\x17\x7e\xb4\xc6\xd9\xbf\x86\x4f\x6d\x1b\x3e\x8e\x15\x78\x3a\x30\x65\xc6\x84\x57\xe4\xd5\x96\x0c\x94\xfb\x71\xc2\xb7\xe1\xea\xc1\xc4\x02\xc6\xee\xf8\x75\xb0\x28\xfb\x37\xed\xde\xaf\xd3\xe7\x12\xf6\x9e\x8b\x8e\x13\xee\x3d\x17\xfb\xeb\xb9\x78\x2b\x77\xe4\x2b\x1f\xcf\xd7\x12\xeb\xba\x07\xe1\x9b\x68\x35\x08\x6a\xcd\xb3\x84\xe3\x78\x99\x2b\xa6\x10\xbc\x42\x70\x94\x95\x91\xf8\xc5\x67\xef\x41\x58\x2b\x46\xfb\x1b\x8b\xe4\xab\x4f\x7c\x5f\xb4\x94\x57\x0c\xe5\x6b\x26\xf1\x35\x54\x92\xf7\x81\x9f\x5a\x8c\xb7\x0f\xed\xeb\x2d\x4a\x6f\x6f\x51\xea\x43\xfb\x7a\x15\x70\xcf\x54\xc0\x3e\xb4\xaf\x0f\xed\xeb\x15\xe4\xe5\xd3\xee\x15\xe4\x2f\x22\xb4\xaf\x93\xa8\xfd\x82\xd8\x9b\xdb\x0b\xdd\xbd\xcc\xed\xde\xeb\x65\xee\x5e\xe6\xfe\x42\x65\xee\xfd\x58\xe1\x5e\xe0\xee\x05\xee\x5e\xe0\xee\x05\xee\x5e\xe0\xde\xf9\x32\xf6\x02\xf7\x6b\x16\xe8\x6c\x96\xba\x57\x24\xd9\xbc\x57\x5f\x4e\x2f\x6e\xf7\xe2\xf6\x7e\x8b\xdb\x7b\x33\xa1\xf7\x53\xe6\xb1\xdb\x7c\xfa\x22\xe5\x7d\x91\xf2\xbe\x48\xf9\x8b\x17\x29\x77\x5f\x77\xc8\xf8\xb0\x87\x4b\x61\x95\x4b\x03\xf8\x28\xc8\x8c\x4a\x05\xec\xbf\x8b\xbc\xb2\x3a\xd1\xe3\xbd\xca\x29\x7d\xaa\x87\x7f\xda\x4b\x2d\xbd\xd4\xf2\x1b\x95\x5a\xf6\x28\x16\x6c\x2f\x32\x56\x52\xac\xa2\x39\x9e\x24\x64\xec\x8d\x3e\xb2\xab\x1e\x7c\x41\xa5\x92\x28\xca\xa5\xe2\x69\xfb\xe5\x72\xe9\x7a\x18\xf8\x0e\x4e\x39\x9b\xd2\x59\x6e\xee\x16\x03\xce\x19\x9c\xe8\x42\x12\x5c\x64\x64\x95\xa7\xaa\xa1\xf5\x77\x71\x2d\x35\x0f\xfd\xb5\x6e\xa7\x75\x04\xf7\xc2\xc8\x67\xa5\x6e\x2d\x6b\x8d\x6f\x87\x77\xd7\x0f\xb7\xa7\xc3\x13\x34\xc8\xb2\x84\x1a\xbb\xbb\x21\x05\xfa\xab\x9e\x14\x52\x58\x3e\x16\x7b\x29\x0c\x99\x1b\x0c\x5b\x30\xf4\x6b\xd9\x18\x1d\xa2\xd3\x8b\x87\xbb\xfb\xe1\x6d\x4b\x83\x96\x50\x20\x6f\x95\xa4\x59\x82\x15\x89\xd1\x63\x3e\x21\x82\x11\x2d\xed\x58\xa4\xdb\xc2\xfc\x6f\x1a\x1d\xfe\xf7\xf0\xf4\xe1\xfe\xfc\xfa\x6a\xfc\xd7\x87\xe1\xc3\xf0\x04\x39\x8a\xd3\xcd\xea\x71\xe9\x51\xc4\x0b\x86\x53\xad\x81\xe8\x1f\x8a\x4c\xd9\x7f\xe4\x24\x27\x08\x4b\x49\x67\x2c\x25\x80\x08\x5c\x6a\xd1\x0d\xf8\x62\xf0\xdd\xf0\xa2\xdc\xf2\x9c\x84\xf0\xbb\x28\xc1\x13\x92\x58\x7f\x04\x98\xd8\x35\xa1\x07\x50\xc5\xc6\x51\x91\x9b\x55\xfd\xeb\xc3\xe0\xe2\xfc\xfe\xe7\xf1\xf5\xc7\xf1\xdd\xf0\xf6\xc7\xf3\xd3\xe1\xd8\x4a\x95\xa7\x03\xdd\x6f\xa9\x27\x2b\x7c\xa2\x7f\xe4\x38\xd1\xda\x09\x9f\x3a\x3c\x5e\xf4\x3c\x27\x0c\xe5\x0c\x28\xce\xa8\x3c\x5a\x0f\xf2\x9d\xea\x53\x66\x66\x74\x73\xf1\xf0\xe9\xfc\x6a\x7c\xfd\xe3\xf0\xf6\xf6\xfc\x6c\x78\x82\xee\x48\x02\x4a\x81\x5b\x74\xd8\xc5\x2c\xc9\x67\x94\x21\x9a\x66\x09\xd1\xab\x81\x6d\x36\xf1\x1c\x3f\x51\x2e\xec\xd1\x9d\xd1\x27\xc2\xcc\x3a\xc2\x99\x85\xf6\x9d\xf0\x3d\x0e\x96\xee\xfa\xea\xe3\xf9\xa7\x13\x34\x88\x63\x3f\x07\x09\x6d\x94\x28\xc7\xc1\x3a\x1f\x96\x87\xad\x99\x03\x74\x6f\x88\x88\x3f\x11\x21\x68\x4c\x2a\x74\x34\xb8\xbb\x3b\xff\x74\x75\x39\xbc\xba\x87\x15\x53\x82\x27\x12\xcd\xf9\x33\x98\xb2\x61\x86\x60\xe1\x7e\xc2\x34\x81\xce\xdc\x66\x71\x86\x9e\xe7\x14\xdc\x1f\x00\xcc\xec\x7b\x36\xfa\x99\xc8\xd9\x9b\x5b\x67\x4b\x07\xaf\xae\xb6\x54\x4f\x52\xfd\x8d\xca\xb1\x58\xf6\x42\x89\xca\xeb\x2f\xae\xa2\xd6\xfa\x17\x15\x72\x6b\x57\xd6\x6a\xf4\xd2\x3e\xd3\x62\xaf\x3b\xeb\x6a\xe5\x35\x7c\xbd\x6b\x96\x28\x41\x23\xf9\xb2\x50\x4f\x22\x67\x8a\xa6\x04\xd9\xce\xec\xe1\xdc\x21\xfc\xd3\xa5\x69\xf8\x3d\x5c\xb0\xb5\x52\x0e\x9f\x88\xb2\xc3\xef\x55\xc0\x5e\x05\xdc\x0f\x15\xf0\xbd\x65\xfb\xc7\x24\xab\x77\x58\x99\x18\xbc\x63\xbc\x5e\xb5\x20\x0d\x63\x4f\xb4\x16\xd5\x84\x3c\x91\x04\xa4\x3c\x25\xb0\x56\x1a\xad\xec\x32\x11\x04\x3f\x6a\x81\x2f\xe6\xcf\xa1\xe4\xd2\x80\xdc\x8f\x76\x73\x0b\x77\x09\xe2\xf8\xd3\x1f\x5f\xef\xb2\xd0\xcb\x1d\xbf\x46\x09\xef\x5b\x08\x92\x59\x8a\x11\x18\x24\xd8\xff\x62\x2d\xc1\x2b\x6e\x8b\xe0\x8b\xf7\x70\x51\x84\xc3\xdd\x23\xad\xeb\x36\x54\x82\x1d\x0b\x4d\x89\xc2\x31\x56\x58\x1f\x9a\x19\x51\x47\xe8\x9a\xc1\xb3\x7b\x2c\x1f\x0f\x90\xbb\xf2\x34\x5b\x29\xac\x0c\xaf\x90\x5a\xff\x4e\x0c\xf8\xeb\xf3\xf1\xfe\xfa\xee\xaf\xef\xe6\x95\xe9\xc3\x3c\x5b\x56\x78\x57\x17\xe3\x5a\x3e\xaf\xdd\xdd\x5f\xa6\xc5\xf7\x7b\x85\xbd\xae\x93\x6b\xa7\x17\x9a\xa9\x9c\xd5\xdf\x56\xe6\xff\xfa\xdb\xaa\xbf\xad\xfa\xdb\x6a\x0f\x56\xf8\xcd\x1d\x86\x0d\xdc\xfd\x4d\x3d\x86\xab\xb4\xd3\x8d\x21\xef\x0a\x6d\x74\x1d\xd0\xbb\x5f\xba\x62\xdb\x15\xdf\xd0\xf7\xe1\x23\x0c\x26\xf9\x1a\x69\x6d\x3b\xbd\xcc\x4d\xbe\x48\xaf\x9f\xbe\xe0\x8d\xdf\x23\x10\xee\x14\x81\x70\x3f\xe6\xfa\x22\x29\x70\x6f\x63\x31\x7d\xfb\xb4\xb7\x1e\x6a\xb0\x4f\xec\xea\x13\xbb\xe0\xf7\x1e\x6a\x70\x77\xd4\xfa\xb2\xd2\x35\x8f\xc9\xb8\x12\x25\xe0\xff\x39\xae\x7a\x7e\x4a\x4f\x42\x37\x50\xe9\x41\x91\xe9\x06\xad\xd3\x78\x97\x45\xa4\xae\x78\x4c\x3a\x47\x12\x94\x5e\xde\x73\x19\xdc\xcd\xd3\xc8\xe2\xa5\x81\xbf\xb0\x24\xde\xb2\xe5\x5f\xa2\x61\xa7\x81\x80\x7b\x2b\xcf\xca\x85\xfa\x52\xe3\x0b\x0a\x0e\xf5\x8e\x3c\x15\xdd\xd8\xb8\x8b\x69\x1c\xb7\x30\xf3\xe6\xe7\x9e\xa5\x37\x3f\x7e\x19\xcc\xa0\xee\x1c\x1d\xcc\x2a\xe1\xdb\xef\xc3\xae\x12\x8e\xf8\x35\x2c\x2b\x4b\xf7\xfe\x8b\xe3\xea\xcb\x28\xb9\xe7\xed\x1d\x97\xeb\x4b\xe5\xf0\x3d\xc4\xcf\x32\x5b\x47\x8f\xa1\xd3\x9b\x5a\xf6\x67\xc2\xbd\xa9\xe5\x5d\x9b\x5a\x8c\x8b\x76\x9c\x61\x41\x98\x6a\x10\xa9\xab\xd7\x09\xbc\x1e\x62\x2e\x38\xa9\x03\x1a\x40\x5a\xa2\x45\xf6\x42\xf6\x57\xd5\x97\x65\x7b\xb1\x82\x41\x90\x09\x79\xfc\xcf\xe2\x6f\x2f\xac\x97\x2a\x40\x2c\x89\x4e\x32\x58\xff\x52\xdf\xd1\xb9\x0d\x54\xda\x3e\x57\x12\xab\x92\x28\x08\x41\xd4\x2b\xe3\x99\x6e\xcc\xdb\xef\x2b\x45\xb2\x36\xe8\xd7\x8d\x6d\xaa\x6f\x7c\xb7\x03\xe4\x76\x86\x9a\x74\xbf\x20\xa7\x4c\x4b\xa3\x7c\x5a\x5c\x0c\x12\x3d\xd3\x24\x01\x44\x11\xc8\x78\x6c\xbb\x01\x7e\x73\x11\x0f\xad\x3b\xff\xa6\x71\x0f\x4d\xdc\xa1\x89\x25\x74\xb1\xa7\xee\x2a\x67\xda\x11\x1b\xa4\xb3\x82\x36\xb4\xc2\x00\xfb\x65\x70\x82\x4f\x44\xbd\x16\x1b\xd8\xf4\xec\x2f\x3d\xf7\x82\x4c\x89\x20\x2c\x22\x7b\xe8\x6d\x5f\x27\x0c\xe4\x27\x33\x49\x1b\x03\xe2\xa1\x04\xc2\xa9\x2a\x6e\xf5\xb4\x92\xa8\xdb\x67\x92\xf7\x99\xe4\x7d\x26\x79\xf5\xa8\xf7\x99\xe4\x7d\x26\x79\x63\x0e\x44\x4c\x12\xa2\x48\xab\x54\x71\x06\x8f\xdf\x4a\xaa\x30\xbd\x7f\x19\x82\x85\x99\x4b\x2f\x5b\xfc\x66\x34\x0b\xb7\xe1\x7b\xa1\x59\x98\xb3\xb6\xca\xfc\x50\xfa\xb1\x21\xc4\xfa\xd5\x4d\x12\x9b\x30\x8d\x92\x5d\xe2\x0c\x5e\x7f\x97\xac\xa3\x3a\xf4\xde\x46\x81\x82\xad\x7b\x39\x4e\x52\x3b\x02\xdd\x26\x6e\x3d\x86\xef\x77\xde\xfb\xc2\x41\xdb\xe8\x7e\x5f\xf9\xe8\xc6\x49\x29\xfb\x62\xb1\xf9\x82\x78\x64\x6f\xbd\x79\xe3\x5c\x89\x1a\x33\x7c\xb7\xd3\xed\x8d\x55\xbd\xb1\xaa\x37\x56\xf5\xc6\xaa\xde\x58\x85\x7a\x63\xd5\xda\xc6\xaa\x2f\x48\xa6\xea\x0d\x57\xbd\x58\xb5\xbb\xe9\xee\xab\x96\xb9\x4f\xd6\xba\xce\x28\xe9\x45\x0e\xd5\xca\xc8\x7b\x3b\xed\x5f\x56\x84\xdc\xdf\xb8\x11\xbc\x1f\x7e\x25\x5f\x9a\x25\x6d\x13\x58\xec\x76\xf4\x8b\x8d\x2b\xee\x4b\x87\x36\xae\x55\x1f\xf6\xbc\x64\x71\xfa\xb0\xe7\x3e\xec\x79\xef\xc2\x9e\x77\xae\xac\x64\x5c\x2e\x03\x24\x32\xa5\xb3\x96\xe6\x3f\xbb\x3b\x1b\x12\x8d\x80\x14\x0c\xca\x71\x4c\xb2\x84\x2f\xc0\x92\xb2\xe4\x3a\x77\x5d\xdc\xd4\x24\xea\x7d\xbf\xd1\xdd\xc8\x5f\x4b\xe7\xd8\x17\x99\xb4\x98\xf7\x5e\x48\xa1\xc7\xff\xac\xa4\xf3\x77\xc2\xcb\x64\x88\x7c\xa6\x12\x6e\xa5\xd5\x84\x3d\x62\xcd\x4f\x82\xd2\x85\xf6\x1e\x9c\xe4\x2a\xc8\xdd\x93\x5a\xb0\xca\x88\x50\x8b\xe0\x4d\x92\x66\x6a\xf1\x5f\x23\x46\x95\xf7\xb0\xd1\x19\xe3\xc2\x70\x35\xfd\xf1\x1c\xb3\x38\x21\x42\x5f\xaa\xae\x9d\x08\x33\xc6\x15\x88\x1b\x30\x83\x18\x3d\x51\x6c\x84\x93\xc1\xcd\x79\x67\x3f\xf3\x3b\x3a\x5d\xaf\x5d\xac\x6e\xc5\x5d\xf7\x29\xe1\x13\xa8\x60\x99\x97\x75\x7a\xdd\x40\xef\x19\x2d\xed\xdc\x5b\x31\x04\x85\xe5\x63\x15\x38\xa4\x9c\x85\x3e\x5e\x0a\x25\xb2\xe2\xdd\x12\xc6\xfc\xf2\x57\x2b\x70\x23\xe5\x67\x16\x80\x04\x1e\xc3\x90\xab\xe3\x70\x3f\x86\x1d\xba\xdf\x8a\x96\xdd\x2f\xae\x74\x37\xfc\x28\x88\x12\x8b\x31\x56\x4a\x33\x99\x5d\x62\x9c\xdc\x63\xf9\xd8\x19\xe3\xa4\xf4\xf2\x9e\xb3\x9c\x12\xc6\x49\x79\xe0\x2f\xce\x72\x3a\x52\xe7\x0a\xce\xf4\xfe\xf2\xe3\xbb\x9e\xb5\x35\x26\xfe\x5b\xc9\x95\xef\xc6\x7b\x56\x99\x69\xdf\x63\xde\xfc\x32\x66\xba\x37\x23\xac\xf0\xf3\x2f\xf1\xe4\x96\x6f\xa7\xfe\x88\x2e\x5b\xa3\x2f\xae\x10\x6e\x45\xe8\x58\x31\xb7\x77\x52\x10\xb7\x2a\x37\xed\x7a\x54\x2f\x63\xe6\x0e\x76\x63\x9d\x18\xa0\xf3\x32\x5a\xb9\x3f\x43\x2e\x2a\xa8\x28\x3d\x3b\x87\x44\x6b\x2a\xc3\x84\xf8\x88\x0b\x23\x79\xc5\xf6\xcc\x1a\xb3\x9d\x01\xf3\x3d\x41\x03\x14\xdb\xda\xfc\x82\x64\x82\x48\xc2\x94\x51\xb5\x4d\xbd\x2b\x57\xde\x9f\x32\x6b\x21\x3a\xc3\x0a\x9f\x62\x85\x13\x3e\x3b\x42\x03\x5f\xd8\x9f\x4a\x84\x13\xc9\x11\x76\x84\x43\x62\xd7\x04\x66\xb1\x63\x0f\x18\x45\x3c\xcd\x68\xe2\x91\xda\xbd\x15\x9f\xb2\x98\x3e\xd1\x38\xc7\x89\x47\xc6\x1e\xb1\xe1\x13\x61\x2a\x07\x15\x0e\x27\x09\xb2\xdd\xba\x17\x02\xfd\xdc\x8d\x52\xd2\x94\x26\x58\x20\xc5\xed\x68\xaf\x6d\x5b\xe8\x7e\x4e\xfc\x58\x1d\x0a\x38\x4a\xf1\x23\x91\x88\x2a\x94\x71\x29\xe9\x24\x29\x8e\xf1\xc3\x39\x82\x71\x9f\x5e\x9c\x83\x69\x34\x52\x88\x1b\x3e\xe8\x3a\xb7\x7e\x02\xd7\x63\x8a\x19\x23\xd0\x31\x57\x73\x22\x6c\xf7\xf6\xe5\xb7\xb6\x72\xbe\x35\x46\x74\xbb\xc5\x34\x1c\xd9\xdb\x29\x9d\x9d\x35\xce\xae\xea\x66\x37\x5d\xb3\x5d\xd1\x7c\x01\x2f\x6d\x77\x6d\xf0\x82\xca\xb2\x3a\xf8\x2e\x5c\xb6\xa5\x11\xbf\x06\x3e\xda\x6f\x54\x11\xec\xb5\xc0\x17\x59\xb7\x2f\x55\x05\xdc\x73\xfd\xaf\x47\x76\xeb\x51\xec\xfb\x00\x8c\x1d\x2e\x4e\x1f\x80\xd1\x07\x60\x7c\xb1\x01\x18\xed\xda\x04\x8d\xb7\x4e\xd7\x5b\xb3\x82\x94\x37\x0a\x88\x5f\x40\x94\xc2\xf2\xb1\x6b\x4d\x29\x2d\x2a\x9f\xc7\xef\x42\xaa\x6f\x9c\xf0\x6b\x48\xf7\x7d\x9d\xa2\x9d\xd6\x29\xda\xbb\x69\xf7\x82\x5f\x2f\xf8\xf5\xb2\x4d\xc7\x09\xf7\xb2\xcd\xfe\xca\x36\x6f\xa5\xb0\x7c\x49\x10\xba\x5a\x78\x2a\x65\xc6\x2c\x0d\xb0\x35\x70\x34\xe0\x1e\xc8\xb3\x84\xe3\x78\x55\x10\xce\x2f\xa8\x90\x6b\x96\x88\x66\xa6\x5d\xfd\xc1\x9e\x4b\x66\xb5\xf8\x1b\x33\xf2\xdf\x42\x4c\x6d\xeb\xd4\xdf\x34\xac\x16\xe8\x17\x82\xc9\x4a\x41\x69\x2f\xa5\x85\x54\x69\xba\x93\xc2\x21\xff\xb8\xe7\x54\xed\xb7\xf4\x35\xd4\x8b\x2f\xd8\x41\xd0\x3b\x01\x7e\x9b\x85\xcf\xf7\x46\x6a\xed\x55\xbb\x3e\xab\xb2\x37\xea\xf7\x8a\x6f\xaf\xf8\xee\x7c\x19\xf7\x49\xf1\x7d\x43\x89\xda\xa4\x89\xbc\x48\x19\xc3\xcd\x64\xeb\x5e\xb4\xee\x45\xeb\x5e\xb4\xfe\x62\x45\xeb\xfd\x58\xe1\x5e\xae\xee\xe5\xea\x5e\xae\xee\xe5\xea\x5e\xae\xde\xf9\x32\xf6\x72\x75\x45\xae\x86\xbf\x5c\x9a\xf4\xba\x42\x76\x67\xe1\xba\x43\x4e\xf4\x7b\x91\xac\x7b\xa9\xba\x97\xaa\xf7\x5b\xaa\xde\x9b\x09\x7d\x79\x89\x90\x7d\x2a\x61\x9f\x4a\xd8\xa7\x12\xbe\x45\x2a\xa1\xe3\x25\xcb\x24\x94\xba\x60\xf1\x63\x8d\x03\xed\xad\x6c\x51\x8c\x76\xd3\xf0\x8e\x5d\x2d\xb5\x03\x9a\xdf\xa4\xd2\x54\xe9\x37\xd7\xd0\x1e\xd5\x9f\x3a\x70\xd2\x82\x66\x14\x6e\x7c\xab\x11\xc2\x7e\xb2\x6f\xbe\x2f\x30\xf0\xfa\xa8\xfb\xfa\x53\x28\xd8\xb5\xbe\xfe\xd4\x0b\xce\xdb\x1d\xae\x15\x33\x77\x34\x6a\x6c\xbc\xef\x74\xda\x6f\x0e\x2e\xd7\x7e\xd2\xdf\x34\x5c\xae\xf1\x26\xa9\x25\xef\x1c\xff\xb3\xf1\xa2\x78\x83\xb2\x5b\x6b\xdf\x0e\x9f\x88\xfa\x52\xae\x86\xbe\xec\x56\x5f\x1f\x62\x47\xd3\xdd\x88\xf5\xbf\xdb\xd9\xf6\x45\xc6\xfa\x22\x63\x7d\x91\xb1\xbe\xc8\x58\x5f\x64\x0c\xfd\xc6\x8b\x8c\xad\x2d\x3e\x9a\x71\x7c\x29\x12\x64\x5f\x64\xac\x17\x22\x77\x37\xdd\xdf\x96\x10\xb9\x87\x16\x84\xbd\xa8\xa6\xe6\x2d\x08\x6f\x8e\xfb\xe1\x46\xd2\x15\xfb\xc3\x2d\x68\x8f\xff\x61\xff\xaf\xc7\xff\xe8\xf1\x3f\x5a\x66\xdd\x07\xb3\xf6\xf8\x1f\xa8\x0f\xd7\xec\xc3\x35\xf7\x39\x5c\xb3\xc3\x36\xf6\xf8\x1f\x1d\xc5\xb9\x17\xc2\x00\x71\x32\xd7\x5a\x38\x20\x3f\xd5\x15\x8d\xbd\x95\xd2\xdc\x58\x7f\x3b\x38\x20\x8d\xd3\xde\x0b\x95\xe4\x15\x71\x40\x9a\xe8\xba\xb3\x02\xf2\x3e\xf0\x40\xdc\x68\xfb\xc4\xc5\x3e\xc4\x7a\xff\x43\xac\xf7\x2e\x71\x71\x6f\x24\xd9\x5e\xdd\xeb\x73\x17\xfb\xdc\xc5\x5e\x19\xee\x95\xe1\x9d\x2f\xe3\x3e\x29\xc3\x6f\x2c\x61\xbf\x20\x2e\xc8\x76\xb2\x76\x2f\x6a\x9b\xf7\x7a\x51\xbb\x17\xb5\xbf\x50\x51\x7b\x3f\x56\xb8\x97\xb3\x7b\x39\xbb\x97\xb3\x7b\x39\xbb\x97\xb3\x77\xbe\x8c\xbd\x9c\xfd\x6a\x38\x21\x4d\xc2\x76\xc7\x7c\x9b\xf7\x24\x69\xf7\x52\x76\x2f\x65\xef\xb7\x94\xbd\x37\x13\xea\x31\x43\x7a\xcc\x90\x1e\x33\xa4\xc7\x0c\xd9\x48\xbe\xf9\x37\x7b\x2c\x3f\x04\x37\xb1\xbf\xb2\x3f\x7c\x97\xf0\xc9\xfd\x22\x23\xfa\xbf\x67\x34\x25\x4c\x82\x34\x4a\xd5\x22\x94\x67\x5a\x56\xbe\xbe\xe6\x1f\xee\xce\xaf\x3e\x5d\x84\xd9\x34\x1f\x2e\x1f\x2e\xee\xcf\x6f\x06\xb7\x7e\x5d\xfc\xac\xc2\xb5\xb0\xdf\x95\x44\x32\x4b\xf2\xb7\x44\xeb\x9e\x70\x6a\xee\x14\x56\xb9\xdc\x6c\x64\xb7\xc3\xbb\xe1\xed\x8f\x90\x0d\x34\x3e\x3b\xbf\x1b\x7c\x77\x51\x22\x88\xd2\xf3\xc1\xe9\x5f\x1f\xce\x6f\xdb\x9f\x0f\xff\xfb\xfc\xee\xfe\xae\xed\xe9\xed\xf0\x62\x38\xb8\x6b\xff\xfa\xe3\xe0\xfc\xe2\xe1\x76\xb8\x74\x3d\x96\x8e\x76\xb9\x12\x22\x61\x91\x20\xce\x1f\x45\x96\x6b\x88\x62\x0d\x91\x17\x1f\x1d\x3b\x6c\xea\xeb\x04\x3d\x58\x9d\x9e\xda\xc6\x0d\x83\x0d\x1a\x32\xca\x48\x4c\x25\x9e\x24\x24\xae\xb5\xe4\xd6\xb0\xad\x25\x5c\x1a\xd4\xb3\xd6\x9e\xbd\xc8\xa9\x79\x5e\x64\x78\x01\x82\x1c\x45\x45\x58\xdc\xd0\x87\xd9\x87\xd6\x1e\x98\xe6\x5d\xf4\x89\x94\x7a\x8a\x72\x21\x08\x53\xc9\x02\x91\xcf\x54\x2a\x59\x6b\xd4\x6d\x5f\x5b\xb3\xf6\x4e\xf5\x0d\xce\xb1\x44\x13\x42\x58\x79\xfc\x82\x24\x04\xcb\x86\x31\xdb\xdd\xef\xb6\x2c\x7e\xaf\xac\x35\xc6\x5c\x46\x53\x4c\x93\x5c\x90\xca\x69\xe1\x69\x86\x05\x95\x9c\x0d\x3f\xeb\xbb\x4c\x1f\xe4\x6b\xf8\x9c\x8b\xcd\x4e\xcc\xf0\xaf\x21\x05\x5f\x95\xff\xf9\xe9\xbe\xfc\xaf\xd2\x99\xbf\xb8\x2f\xff\x6b\x39\xad\x07\x0d\x57\x29\xfb\x10\x7d\xba\x3f\x41\x9f\x20\xc4\x49\xa0\xfb\x39\x36\x14\x7b\x71\x7f\x82\x2e\x88\x94\xf0\x4b\xf1\xb1\xa2\x2a\x81\xb9\x7d\x47\x19\x16\x0b\xe4\xa6\x6f\x12\x5d\x71\x34\x47\xc4\x2f\x4d\x75\xf1\xd8\xdf\x73\x06\xaa\x7b\xb1\x7a\x17\x7c\x46\x23\x9c\x6c\xb7\x88\x83\xab\x12\x1f\xb8\xbe\x5d\xba\x14\xe1\xdb\xf5\xb5\x18\x5c\x9d\x41\x12\xa9\x1b\x6a\xc3\xcc\xaf\x88\xd4\x44\x12\x71\x16\x5b\x2f\x8d\xbe\xfd\x17\x81\x50\xff\x77\x0e\x89\xb8\xb9\xa4\x6c\xa6\x5b\x44\xc7\xe8\xfa\x76\xc4\xae\x45\x6c\x0c\xa1\x44\x4b\xc3\x86\xe6\xa8\x44\x8c\x2b\x44\xd3\x8c\x0b\x85\x99\xd2\x8a\x00\x88\x01\x76\x45\x0c\x07\x38\xe5\x69\x9a\x2b\xac\x0f\x5a\x6d\x51\x99\x31\x87\xdc\x11\x75\x1e\x83\x6b\xa5\x61\x0d\x8d\x9c\x50\xcc\x25\x13\xba\x7d\x2d\xa3\x94\x75\x68\x1a\xd7\x54\x59\xd7\x04\x16\x02\x97\xa5\x89\x0f\x54\x91\xb4\xfa\x7e\xc7\x20\xcf\x7f\x35\x1a\x08\x4e\x4d\x52\x05\x11\x03\x11\xcd\xa9\x22\x91\xd2\x47\x70\x23\x9a\x78\xb8\xfa\xe1\xea\xfa\xa7\x50\x82\xf8\x30\xb8\x3c\xfb\xf3\x7f\x94\x7e\xb8\xbd\xac\xfd\x30\xfe\xf1\xcf\xb5\x5f\xfe\x7f\x4b\xe9\xa9\xda\x53\x4d\xcf\x0f\xe6\x72\x08\x22\x35\xd8\x84\xdd\x54\x11\x4d\xf1\x8c\x20\x99\x67\x9a\x02\xe4\x51\x79\x7f\xb5\x48\x79\xc1\x71\x4c\xd9\xcc\x64\x80\x5e\x50\x45\x04\x4e\x2e\x71\xf6\xd1\xd9\xaf\x37\x58\x9d\xff\x7b\x57\xca\xd7\xfd\xf0\xf3\xe0\x32\xcc\xf8\xfd\x70\x73\x7b\x7d\x7f\xbd\x74\xd6\xa5\x16\xea\xc7\x48\x3f\x3e\x81\xff\x45\xc7\x48\xb7\xee\x25\xdf\x94\x28\xac\x35\x02\xf4\xb5\x49\x9a\xf3\x89\x34\x94\x25\x70\x6a\x32\x41\x53\x0a\x57\x8a\xb1\xe0\x7d\x63\x84\x6b\xaf\x3d\xf8\x73\x63\x3e\x00\x6d\xd9\x5d\xca\x2c\xc6\x22\x46\x7f\x97\xd5\xf4\x71\x30\x1c\x9b\x1f\x48\x8c\x0e\xd1\x5c\xa9\x4c\x9e\x1c\x1f\x3f\x3f\x3f\x1f\xe9\xb7\x8f\xb8\x98\x1d\xeb\x3f\x0e\x09\x3b\x9a\xab\x34\x31\xe9\xf2\x7a\x15\x4e\xd0\x8d\xe0\xfa\x0a\x01\x05\x9d\x08\x8a\x13\xfa\x2b\x89\xd1\xc4\xf0\x3f\x3e\x45\xbf\x44\x5c\x90\xa3\x62\x63\xac\x51\xc9\xde\x23\xd6\xf0\x74\xac\x5f\x6a\x60\x26\xd5\xfd\x44\x31\x89\x68\x6c\xc5\x0c\xc2\x22\x0e\x96\x47\xe3\xab\xd0\xed\xb9\x4c\x43\xad\xd1\x64\xb9\x2a\x96\x33\x50\x56\x70\x4c\x82\x6c\x77\xc5\xcb\x04\xa7\x15\x9f\x73\xa3\xb6\xe6\x5a\x45\xd7\x77\x2b\x86\x5b\xd5\xbd\x9a\xe9\x09\x47\x3c\x41\x93\x7c\x3a\x25\x22\x74\x48\x1f\x68\x6d\x86\x4a\x24\x48\xc4\xd3\x14\x24\x06\xfd\x55\x2e\x0d\x55\xc3\x8a\xd9\xd1\x1e\x8d\x18\xec\xbf\x56\x73\x80\x02\x62\x0e\xac\x8e\x11\x12\x23\xcc\x16\xa6\x9b\x49\x3e\x0d\xdb\x37\x30\x14\x38\x46\x54\x8d\xd8\x20\x49\x90\x20\x29\x57\x24\xc8\xa1\x04\xe7\x59\x79\xc1\x81\x45\x0a\x92\x25\x38\x22\xb1\xa1\x87\x84\x47\x38\x41\x53\x9a\x10\xb9\x90\x8a\xa4\x61\x03\x5f\x83\xad\x46\xaf\x19\x95\x28\xe6\xcf\x2c\xe1\xd8\xce\xa3\xfa\xd9\x37\xe5\xd3\x38\x74\x10\x01\x43\x21\xb8\x80\xff\xf9\x81\xb2\x78\x67\x1c\xea\xe1\x6e\x78\x1b\xfe\xfb\xee\xe7\xbb\xfb\xe1\xe5\x7a\xdc\xc7\x53\x16\x0c\x0f\x74\xf8\x13\x74\x67\x16\x81\x0b\x2d\x11\x89\x96\x49\x5d\x5a\x52\x2a\x7e\xe0\xf1\x86\xdc\xf7\x72\x70\xf5\x30\x28\x71\x94\xbb\xd3\xef\x87\x67\x0f\x15\x7d\xc0\xce\xaf\x24\xc3\x1b\xf5\x2f\xfc\xed\xf4\xfb\xf3\x8b\xb3\x71\x83\xc2\xf8\xe1\x76\x78\x7a\xfd\xe3\xf0\xb6\xd0\xed\x1a\x97\xa8\x32\x98\x2a\xb3\xba\x37\x4c\x69\xce\x63\x34\x59\x34\x03\x42\x68\xc9\x39\x01\x5f\x6c\x01\x89\x62\x5a\x3d\x01\xde\xe4\xb0\x39\x8a\x2f\x52\x1e\x93\x03\xfb\x0e\x20\x69\x18\xe3\x8a\x91\x98\x9b\x1b\xd6\xbd\x63\x16\x18\x2a\x0c\xc8\x85\x5f\xb8\x13\x34\x40\x52\xbf\x98\xeb\x43\x2d\xe8\x6c\x06\x86\xc3\xca\x50\x4d\x6b\xf6\x53\x58\x5e\xf8\xce\xec\x7f\x26\x38\x9c\x73\xdd\xad\xb5\x38\x7b\xab\x84\xf9\x10\x50\x57\xca\x2d\x0a\x0c\x06\x87\x86\xa1\xb9\xcd\xd2\x8b\xd0\xba\x5e\xe6\x3c\x1a\x7b\x91\x3e\x5c\xc0\xb6\xa4\xb1\x77\x66\x82\x3c\x51\x9e\x07\x9f\x5a\x60\x8f\xd2\x8e\x37\x36\x5f\x2c\x00\x2c\x9b\x31\x8a\x54\x9a\xf1\xe4\xd1\xd8\x82\x66\x61\x4f\xd0\xc2\x54\xf0\xb4\xa1\x8d\xf2\x31\x39\xbf\xbe\x53\x02\x2b\x32\x5b\x9c\x59\x96\xb1\xf9\xf1\x38\xbb\xfe\xe9\xea\xe2\x7a\x70\x36\x1e\x0e\x3e\x95\x4f\xbc\x7f\x72\x77\x7f\x3b\x1c\x5c\x96\x1f\x8d\xaf\xae\xef\xc7\xee\x8d\xa5\x24\xdf\xd2\x41\xfd\x9e\x2e\xbf\x78\x82\x34\xcb\x05\xd6\xe8\x00\xef\x02\xfe\x38\x21\x53\x2e\x0c\x9f\x4f\x5d\xe8\x82\x15\x61\xdc\xda\x5a\x5d\xac\x32\x8b\x13\xb0\x8c\x35\x35\x69\xac\xde\x4a\x10\x9c\xc2\x3d\x81\x19\x1a\xb2\xf8\xf0\x7a\x7a\x78\x67\x7e\x4c\xb1\x78\x24\xc2\x7f\xfa\x2c\xa8\x52\x84\x95\x54\x3a\xec\x86\xec\x95\xc4\xa2\x83\x23\x74\xab\xf9\xbe\x7e\xdf\x5f\x6a\x9a\xd8\x63\xa2\x30\x4d\xa4\x1d\x6c\x69\x5d\x4f\xd0\x05\x16\xb3\xc2\x0e\xf7\x35\x9f\x4e\x4d\x63\xdf\x98\x61\xe8\x3b\xac\x34\x8b\x06\xde\xab\x49\xc3\xdd\x8b\xd0\x9f\x7d\xd9\xcb\xc3\x75\xaa\x7a\xc8\xb6\xa3\xa9\x87\x1b\x58\x71\xa3\xb1\x97\x74\x43\xfb\xa4\x81\xd6\x60\xe2\xe6\xf1\xf2\x4b\xa6\xb9\xed\x3a\x39\x95\x5f\x6c\x20\x27\x93\x4b\xa5\x77\x7e\xaa\xb5\xcd\x06\x5a\x22\x9f\xa9\x35\x18\x84\xe3\xae\x90\x50\xd1\x0c\x98\x57\x71\x96\x11\x2c\x64\xd3\x6e\x97\xc5\xc0\x96\xbd\x37\x3d\x85\x7d\xd8\x4d\x76\xfd\x1c\x20\xce\xc0\xe0\xe0\x85\x88\x0a\x45\x76\xa0\x01\xd3\x56\x8d\x02\x6e\x00\x6d\xe9\xda\x22\x1b\x5d\x52\xa9\x95\x46\xf3\xe3\x77\x16\x72\x69\x33\x82\xf8\x38\x38\xbf\xa8\x08\x17\xe3\xb3\xe1\xc7\xc1\xc3\xc5\x72\x33\x61\xe9\xbb\xea\x16\xa3\x43\xa4\x9f\x97\xfd\xe6\x74\x6a\xee\x0c\x07\x1c\x65\x54\x5a\xc2\xc0\x68\x65\xa1\x6a\x8c\xbd\x3a\x26\x59\xc2\x17\x29\x61\x60\xe2\x29\xdd\x84\x7a\x3d\xa7\x98\xda\xab\x25\x18\x2c\x58\x71\xac\xd9\x0d\xae\xb1\x43\x87\x56\x45\x62\x7f\xf3\x96\xc1\xaa\x2a\xac\xfb\xc6\x78\xcf\xec\x7f\xee\x14\x56\x1b\x9e\xb1\xc1\xe9\xfd\xf9\x8f\xc3\xb2\x7e\x78\xfa\xfd\xf9\x8f\x4d\x52\xcd\xf8\xd3\xf0\x6a\x78\x3b\xb8\x5f\x21\x9c\x54\x9a\x6c\x12\x4e\xa4\x1e\x70\xd5\x7b\x4a\xa5\x8f\x08\x8a\x0c\xe4\x15\xa2\x4a\xa2\x27\x2a\xe9\x84\x02\x40\x98\xf5\x44\x3e\x9c\x03\x67\x7d\xc2\x09\x8d\xa9\x5a\x38\xf1\xc5\xf4\x5b\xde\x47\xcd\x49\x6d\xfb\xc6\xec\x10\xfa\x27\xc1\xca\x67\x36\xc7\x4d\xfa\x04\x81\x6e\xfb\x04\x4a\x5b\xf0\x19\xd3\x82\x34\x9b\x11\x61\x86\x03\xde\x97\x70\x2c\xc1\x73\x3d\xaa\x50\x58\x29\x56\xcd\x0b\xad\x33\xc2\x88\x00\x10\x38\xdf\x89\x11\xa4\x04\x61\x5f\x69\x99\x2b\x4b\x68\x44\x55\xb2\x40\x11\xd8\xb0\xc0\x9c\x99\x62\x86\x67\x56\x38\x00\x35\xa7\x42\x12\x7f\x35\x28\x6a\xd7\x53\x6b\xda\xbf\xa7\x64\xc3\x63\xf6\x70\x75\x36\xfc\x78\x7e\x55\x26\x81\xef\xcf\x3f\x95\x44\xd8\xcb\xe1\xd9\xf9\x43\xe9\x36\xd7\x92\xec\x72\xb9\xbe\xda\x6c\xc3\x51\xf4\x2f\x9d\xa0\x33\xf3\xe9\x89\x5e\xdc\x06\x88\x38\xaf\xfc\x56\xd6\xe1\xd6\x85\xe4\xb9\x3f\x86\x4c\x89\x46\xbf\x44\x57\x13\x92\xf5\x41\x96\x6c\x48\xcd\xa1\x0a\xb5\xbe\xaf\xaa\x4e\xe5\xea\x94\xdd\x8b\x10\x74\x79\x54\x58\x96\xc2\x18\x06\x30\x1a\xb4\x19\xb1\x1a\xdc\x5a\x05\xc3\xfe\x11\x5c\xd4\x69\x2e\x95\x71\x25\x02\x71\xa2\xc7\xbf\x48\xbd\xa0\xe0\x6a\x3c\x42\x77\x84\x8c\x98\xb3\x1e\xcc\xa8\x9a\xe7\x93\xa3\x88\xa7\xc7\x05\x3e\xe1\x31\xce\x68\x8a\xb5\x24\x4d\xc4\xe2\x78\x92\xf0\xc9\x71\x8a\xa5\x22\xe2\x38\x7b\x9c\x41\x04\x8c\x73\xa7\x1e\xfb\x66\x67\xfc\xdf\x2f\xfe\xf4\xed\xe1\xc5\x5f\xbe\xfd\x50\xb7\x90\xb5\xed\xff\x90\x45\x38\x93\x79\x62\x23\xe6\x44\xb8\x36\xee\xc8\xe7\x64\xd5\x7e\x5f\x95\xb7\x6b\x3b\xfd\xf5\xf4\xe6\xa1\x64\xb1\x2e\xff\xf3\x72\x78\x79\x7d\xfb\x73\x89\x53\xde\x5f\xdf\x0e\x3e\x95\x18\xea\xf0\xe6\xfb\xe1\xe5\xf0\x76\x70\x31\x76\x0f\xb7\xb1\xbd\xfd\xc0\xf8\x33\x2b\x2f\x8d\x74\x1c\xb0\xd6\xd3\x09\xfa\xc8\x05\xfa\xc1\xef\xe4\xe1\x04\x4b\xb8\x62\xdc\x9d\x25\x0f\x50\xc6\x63\x60\xbc\x88\x64\x73\x92\x12\x81\x13\x6b\x33\x90\x8a\x0b\x3c\x33\x37\xbd\x8c\x04\x56\xd1\x1c\xc9\x0c\x47\xe4\x00\x45\x40\x0d\xb3\x03\xd8\x14\x50\xb5\xf8\xac\x6a\xe7\xbb\xcd\x99\xa2\x29\x71\x2a\xb8\xfd\xe7\xbd\xd9\x8c\x0d\x36\xe7\xfa\xfe\xfb\xb2\xb0\xf7\xf1\xe2\xe7\xfb\xe1\xf8\xee\xec\x87\xa5\xeb\x69\x3e\x2b\x8d\xec\x0e\x02\x90\x4e\x79\x92\xa7\x2c\xfc\x7b\xf3\xb1\x9d\x5f\xdd\x0f\x3f\x55\x47\x77\x3d\xb8\x2f\x53\xc6\x6d\x39\xc0\xed\xc3\x77\xd7\xd7\x17\xc3\x92\x4b\xf8\xc3\xd9\xe0\x7e\x78\x7f\x7e\x59\xa2\x9f\xb3\x87\x5b\x83\x46\xb8\x6c\x9a\x6e\x04\x0d\x13\xd5\xd3\x0a\xa7\xb9\x6b\x56\xd8\x89\x13\x0d\x6c\x40\xb9\x39\xcb\x87\x01\xdc\x8e\x09\x07\x03\xab\xce\xa1\x37\xa9\x46\x66\xa4\x8d\xec\x50\x95\xb7\x09\xb5\xb3\xe3\xa5\x1b\xbd\x8c\x2b\xdf\xfb\x21\x18\x28\x50\xa3\x6c\xe3\x24\xe1\xcf\x26\x94\x37\xa5\xfa\x56\xb6\xc0\x68\xfa\x15\x59\x78\x08\x8f\x1a\x38\x5e\x79\x5b\x48\x24\x88\xba\xe4\x39\x53\x9b\x93\xdc\xe0\xaa\xc4\x77\x86\x57\x3f\x8e\x7f\x1c\x94\x29\xf0\xfc\x62\x39\xab\x09\x9b\x68\xb8\x8a\x07\x57\x3f\xfb\x4b\x18\x02\xbe\x0f\xbc\x86\x6a\x64\xd7\x28\xa1\x5a\xec\x8d\xb0\xd6\x5e\x13\x90\x68\x10\xa1\x60\x72\x48\xf5\xe4\x20\xc0\x34\x33\xfe\x24\xc3\x9f\xcc\x20\x4f\xdc\x1f\x95\xf6\x24\xac\x0b\x58\x53\x5d\x3c\x3d\xb4\x63\xb5\x6a\x86\x08\x7b\xa2\x82\x03\x9e\x2d\x7a\xc2\x82\x6a\x69\xdc\xb4\xac\xe7\x7a\x02\xff\xbb\x5e\x9b\x60\x18\xad\x30\xae\x3b\x2e\xd4\x99\x0f\xe4\xdd\xcc\x1a\xd2\x14\xd0\x5a\x0f\x65\x6d\x36\x74\xd4\xbf\x6d\xd8\x9c\x2d\x03\x7e\xcb\x13\xfe\x47\x72\x46\x71\xa2\x19\xc0\xee\xe4\xc5\xc1\xd5\xdd\x79\x59\x7e\x2c\xab\x19\x01\x5f\xde\x58\x5e\x04\x43\xa5\x19\xb9\x53\x26\xee\xfe\x7a\x61\xb4\x0b\x00\x3d\x36\xe7\x36\x50\x2c\x40\x00\x72\x28\x28\x19\x16\xb2\xf2\x85\x44\x00\x84\x56\x04\x5c\xe9\x3b\x0b\xc2\x99\x9e\x38\x8d\x47\x8c\x7c\xce\x08\x93\x10\x1c\x60\xee\xb3\xc2\xd7\x2e\x8f\xd0\xf9\x14\x58\x82\x7e\x9d\xa1\x9c\x59\x07\x98\xbe\x70\xcd\x20\x0f\xb4\x28\x6b\x87\xe0\x35\x44\x30\xbc\x30\xe2\x82\xa5\x8a\xc1\x8f\xd8\x4f\xde\x89\x06\x8f\xa6\x5c\x33\x20\xbd\x8b\xb6\xbd\x13\x84\x99\xa4\x07\x48\x2b\x2c\xd5\x3d\x85\xd4\x01\xad\x50\xda\x10\x2e\xcd\x69\xec\x9f\xaf\x7f\x0d\xd4\xe2\x84\xc3\xcb\xa0\xf9\x2e\xa8\x5c\x05\x2d\xa2\x71\x62\x3c\x26\xe3\xee\x77\x42\xc4\x05\xb1\x7e\x96\xb5\xaf\x81\x55\x8c\xfd\x1e\xcb\xc7\x9a\xef\xe1\x9c\x49\x85\x59\x44\x4e\x13\x2c\x37\x0c\x42\x72\x36\x8e\x83\xb2\xc4\x71\x7b\xfb\x70\x73\x7f\xfe\xdd\x0a\x2e\x5f\xfd\xb8\x1e\x06\x14\x25\xb9\x73\xcf\x4d\x04\xc7\x31\xd2\xec\x73\xc6\x8d\x2b\xd0\x0a\xfe\x05\xf4\xb7\xc9\xeb\xf1\x01\x95\x25\xd8\xf1\x22\x1d\xc1\xda\x39\x42\x57\x02\xb5\x0b\x81\x22\xbd\x12\x28\x30\x79\xb8\xad\x06\xcf\xa2\x29\x48\x62\xad\x5b\x59\x82\xd5\x94\x8b\xd4\x70\xf9\xd2\xa4\x4d\xe3\xcb\x1b\xa5\x4c\x11\x21\xf2\x4c\x51\x87\xe5\x5e\x95\x52\xa1\xc2\x3b\x9f\x5d\x12\x29\xf1\x8c\x6c\xe3\x80\x6e\x52\x1e\xee\x7e\x0c\xff\x09\x0e\xe6\x2e\xb2\x7f\x69\x84\x2e\xf2\xdd\xd1\xd3\x35\xfb\x68\x02\x79\x6e\x78\x42\xa3\x0d\x03\xee\x3e\x0e\xce\x2f\xc6\xe7\x97\x5a\x89\x1f\xdc\x0f\x2f\x4a\xa2\x04\x3c\x1b\x7c\xbc\x1f\xde\x5a\x10\xeb\xc1\x77\x17\xc3\xf1\xd5\xf5\xd9\xf0\x6e\x7c\x7a\x7d\x79\x73\x31\x5c\x11\x99\xd3\xda\x78\xdd\xba\x5a\x7d\xf5\xa4\xf6\x0b\xec\xb0\xe6\x65\xa1\xbd\x0c\xb2\xc6\x30\x4d\xc0\x09\xce\x8d\x33\x1c\x23\xc6\x63\x02\x3f\x4b\x67\x9d\xf1\xc8\xd1\xe8\x5c\x7d\x95\x24\x08\xe7\x8a\xa7\x18\xbc\x36\xc9\x62\xc4\xf0\x44\xb3\x56\x9c\x24\x41\x78\x97\xc8\x19\xd3\x2c\x56\x37\x66\x20\xda\xa3\x84\x68\x76\x9e\x05\xc9\x7e\xd6\x6f\x30\xa5\x0c\x22\x6d\x53\x2c\x1e\x8d\x9b\xa9\xe8\xb2\x38\x14\x12\x61\x39\x62\x7a\x5c\xc4\x1a\x86\xba\xac\xf0\x49\xa7\xb7\x5a\x57\x27\xc5\x8f\x44\xaf\x4a\x9a\x47\x73\x94\x09\x3e\x13\x44\x4a\x6b\x5b\x8e\x30\x33\x01\x08\xf6\x75\x7d\x0d\x8d\x18\xe3\x7a\x29\x9c\x09\x3b\x26\x19\x61\x31\x61\x11\x35\x69\x7d\xe0\xbb\xf7\xa6\xcd\x99\xc0\xd9\x1c\x49\x0e\x4e\x6f\x58\x76\xb0\x5f\x99\x8f\xdc\x4d\x66\x66\x6c\x1e\x87\x16\x68\x91\x6b\x3e\x71\x0d\x72\xa2\x59\x65\xf8\xd8\x5d\x86\xce\xed\x62\xec\x80\x69\x96\x10\x65\xc0\xfa\x61\xc9\x61\x33\xf4\x5a\x97\xf6\x43\x6f\x53\xd3\x26\xe8\x0b\xdb\x8d\x19\x4b\x3b\xa2\xa3\x06\xcb\xb6\x3d\x52\xe8\x7b\xcc\xe2\x44\xb7\xe2\x7c\x18\xe5\xb3\x08\xa9\x28\x03\x4d\x35\xee\x34\x6e\x73\x8b\x46\x38\x97\xdb\x5c\xa3\x95\x5c\x4c\x63\x15\x3c\x2c\x82\x42\x80\xbc\x6d\x22\x26\xac\x6e\xa6\x59\x24\x4e\xb8\x5d\x25\xf3\x7a\x6e\xea\x3f\x21\x18\x4d\xcb\x35\x9b\x09\xca\x22\x9a\xe1\x64\x23\xdd\xaf\x12\x8c\x6f\x63\xdc\xbf\xa6\x53\x4d\x3e\xdf\xd4\xdc\xb6\x8a\x88\x14\x12\x94\xed\x30\xfd\x16\xae\x61\x49\xb2\x59\x0d\x44\x16\xd1\x24\x58\xf0\xdc\xf8\xe3\x60\x5d\x48\xdc\x70\x54\x8f\x9a\xb6\x5b\x9f\x0c\x5c\x0e\x80\xde\x60\xb3\x4d\xe4\x4f\xdb\xfa\x55\x5a\xb1\xbd\x9b\x60\x3c\x9c\xdc\x34\xb7\xd9\xb4\x03\xc1\xc3\x7f\x2d\xa3\x9d\x4b\x9c\x69\x9a\xb1\xb0\xfd\xb8\x98\xa3\x55\x92\x6c\x55\x30\x17\x3f\x13\xf8\xce\x7d\x5e\x48\xf7\xdd\x28\x96\xd0\x06\x40\xd5\x3b\x29\xc5\x10\x04\x39\xe6\x96\xc6\xa7\xb9\x96\x65\x11\x86\x28\x04\xf4\x35\x39\x9a\x1d\x21\x57\x84\xe1\x00\x0d\x6e\x6e\x86\x57\x67\x07\x88\xa8\xe8\x1b\x17\xb3\x68\x03\x96\x46\x4c\x71\x2b\xad\x2c\x5c\x01\x8d\x94\x88\x19\x29\xcd\xd9\x45\x37\x41\xa8\xf2\x8c\x4a\x65\xc3\x67\x35\x5f\x09\x4a\x9d\xd0\xb4\x2a\x66\x1b\x0a\xc9\xd5\x7c\x1b\xd2\xc0\x52\xe6\xa9\xd6\x65\xc7\x14\xa7\x63\xc1\x93\x6d\x98\xc2\x19\x4c\x05\xd4\x65\x9f\x9e\x4f\x71\x8a\x74\xb3\x36\x14\xc4\xbb\x1c\xbd\x48\xa7\x05\x23\xcd\x97\xf5\xbd\x19\xdc\x5b\xce\xfb\x60\xe3\xd1\xa8\x0b\x81\x80\xf4\xfd\x16\x56\x51\x98\x8d\xc7\xd6\x52\x3f\xc6\x51\xa4\x55\xee\x1d\x4f\x2a\xa8\x9f\xe3\x5c\x02\xb6\xa3\x17\x9b\xe6\x2a\x3a\x77\xc3\xcc\x34\x07\x83\x60\x60\x7d\xe5\x4a\x1e\xd1\xa2\xfd\x86\x7e\x27\x8b\x5a\xaf\xae\xc2\xcd\x83\xf4\x26\x15\x73\x09\x4b\x02\x3b\x29\x4d\x85\x1c\x35\x27\x0b\x34\xc7\x4f\xa4\xd4\xa5\x4b\x88\xd1\x0d\x2f\x78\x2e\x9a\x18\xdd\x88\x9d\x91\x4c\x10\x2d\xe9\x57\x1d\x28\x9e\xa6\x6f\xcb\x94\xd8\xd3\x75\x4f\xd7\xef\x9e\xae\x4f\x4d\xa1\xa4\x81\x2f\x8c\xb5\x95\x00\x67\x1a\x1b\x67\x9c\x27\xe3\x0e\x36\x91\xee\x2b\x5e\xf2\x84\x55\xca\x46\x01\x24\x00\xcf\x41\x3e\x2a\x5d\x9b\x5c\xdf\x75\x41\x8a\xad\x1d\xde\x92\x65\x70\x2e\xb3\xa0\x5e\xce\x36\xe7\xbd\xa9\x95\x65\x2d\xa1\x17\x17\x73\x4e\x8d\x7c\xe3\xdd\x65\x61\xfd\xd3\xd2\x61\x72\xa2\x08\x65\xb5\x6a\x6c\x86\x9e\xf5\x02\x1b\xb9\xe3\x1f\x39\x57\x58\x7e\x73\x34\x62\x5a\x88\x7a\x24\x0b\x63\x6e\xd5\x62\xca\xef\xb4\x2c\x7e\x28\x09\x93\x10\xee\xfd\x3b\xe3\x9e\xd3\x24\xee\xcc\xd5\x46\x35\x35\x45\xe0\x20\xe8\xda\xf7\x02\x21\xba\xb6\x51\x2b\x25\x15\x01\xd0\x20\xe7\x9b\xb9\xd8\x67\x66\xf8\x33\xa2\x20\xc5\x5a\x51\x05\x3a\x53\x6c\xaa\xcc\xd5\x86\xbe\xd2\x74\x65\xa8\x42\x70\xf0\x93\xc4\xf9\x76\x8c\x5f\xd6\xdb\x58\xc9\x19\xbd\xb6\x70\x67\x63\xde\x8f\x9d\xdd\x28\x12\xbc\x56\xba\x0d\x4b\x64\x76\x7a\x62\xd8\x81\xf3\x5f\x13\x76\xf4\x4c\x1f\x69\x46\x62\x8a\x21\x02\x5e\xff\xeb\x58\xcf\xeb\xdf\x4f\x6f\xaf\xaf\xc6\x45\x26\xcf\x7f\x8d\xd8\x20\x91\xdc\x67\x29\x20\xc6\x99\x0f\xb7\xcf\x04\x71\x22\xa1\x9d\x0b\x58\x5d\x0b\x33\xe2\x88\xb5\x8d\x20\xe6\x91\x3c\xc2\xcf\xf2\x08\xa7\xf8\x57\xce\xc0\x95\x3e\x80\x3f\x4f\x13\x9e\xc7\x3f\x61\x15\xcd\x8f\xe1\x5c\xab\x63\xf2\x44\x98\x32\x6e\x2a\xbd\x5c\x31\x24\xef\x4a\x88\xd6\xff\x77\x3d\xe6\x22\xa9\x48\x6a\x4d\x36\x22\x99\x42\xff\x8f\x20\x13\xce\x55\xf3\x25\xc5\xa7\x53\x49\xd6\xba\x90\x0a\x25\xed\xee\x1a\xfd\xe5\xcf\xdf\xfe\x41\x93\xd0\x26\x6b\x7c\x7e\x77\x3d\xd6\xdf\xff\xfb\x99\xfd\x5e\xae\xc1\xee\xae\xb3\x82\xb5\x39\xe2\x31\x81\xf3\x39\x83\xdb\x4f\x80\xf3\x02\xd8\x1b\x90\x43\xb1\x8f\x4d\xdc\xed\xac\xd4\xfa\x76\x2a\xdb\x46\x8b\x09\x2a\x76\x30\x47\x74\x88\x18\x47\xa9\x89\x35\xc5\x0c\xfd\xc7\x0f\xdf\x35\x6f\x60\x2e\xe8\x46\x1d\x52\x0b\xd7\x10\x74\x29\xe9\xaf\x44\x22\x4d\x35\x9a\x8a\x79\xaa\xbb\x16\x44\xce\x79\x12\xa3\x67\x02\x6a\x92\x8d\x03\xf5\x5a\xb9\x20\x23\x16\x36\x01\x21\x87\x08\x27\x8a\xcf\x08\xdc\xd5\x4e\x51\x53\x44\x68\x51\xc5\x64\x69\x28\x2e\xc8\x81\x81\xfa\xba\xfb\x93\x8b\xad\x86\x69\xc2\x23\x97\xd4\x62\x4d\x72\xf1\xa4\x79\xe6\xd3\xaa\xe9\x15\xb5\xdb\xf0\xab\x9b\x6c\xcd\xb6\xcd\x4b\x63\x93\x50\xac\x0d\xab\xba\x33\xcd\x83\xa1\x11\x67\xe3\x84\xb2\xc7\x8d\x36\xe3\xda\x89\x72\xba\x05\xbb\x66\xba\x45\x6f\xe7\x36\x16\x90\x35\xce\xc7\xc7\x3c\x49\x4c\x6a\x4b\xb8\x3d\x20\x77\x99\x75\x03\x61\x20\x33\x39\xa0\x24\xb6\x7e\x2f\xab\x09\x0b\xc2\x20\xe0\x6d\xc4\x26\x0b\xeb\xb3\x95\x07\x48\xe6\xd1\xdc\x65\xe6\x45\x9c\x49\x2d\x46\x73\x81\x22\x9e\xa6\xa6\xb8\x29\x23\x48\x71\x9e\x48\x1b\xed\xce\x0e\x15\x8e\xd4\x88\x15\xfd\xad\x38\x79\xa6\x02\xd2\x76\xa9\x7b\xdd\x5d\x3a\x45\xa5\xa5\xa5\x02\x37\x8d\x43\xcc\x06\x30\x82\x19\x4f\x54\x80\xfe\xc0\xeb\x67\xc9\x6c\x58\x8b\x66\x20\xe7\x5c\xa8\x71\xdc\xc8\x73\x56\x12\x4d\x95\x11\x32\x72\x98\x40\xd0\x30\x7f\xd2\xc2\x3f\x79\xf6\xc6\xd7\x65\x43\xd0\x54\xbd\x6c\x04\xdd\x8e\xd1\xd2\x91\xad\x4b\x82\x2d\x6b\x65\x10\x3c\xa2\x72\x4c\xf8\xaa\x31\xde\xc1\x57\xa7\xfa\xa3\xa5\x8b\x57\x3d\x77\x4e\x08\xe2\x71\x01\x36\x67\xee\x75\x9b\x11\xb2\x6c\x4d\x2d\x74\xc2\xcb\x65\x8e\x2e\x9b\xca\x43\xd9\x92\xab\xc7\x02\x26\x7b\x49\x40\xd6\xc4\x62\x42\x95\xc0\xa2\x84\x14\xe2\xf5\x41\x49\xb0\x80\xf8\xac\x11\x33\xb8\x71\x46\x53\x88\x51\x4c\x25\x24\x88\xc0\x5d\x1a\x38\xc3\x50\x37\x25\xb0\x72\xb4\x8b\x3c\x47\x13\x7f\x0e\x81\x65\x05\x69\x38\x66\xa7\x3b\xf2\xf8\x58\x5a\x3f\xe3\x51\x5e\x08\x72\x11\x48\xb8\x16\x53\x07\x51\x26\xe9\x6c\xae\x10\x65\xd6\xee\x88\x93\x19\x17\x54\xcd\x53\x79\x80\x26\xb9\xd4\x5a\xa8\x09\x56\x33\xf1\x28\x44\x45\x9d\xb8\xd0\xb6\x49\xc4\x71\xa5\xc1\xba\x8a\xb2\x01\x69\x74\x3b\x94\xc3\xca\x5d\xb1\x82\x70\x06\x1e\x67\xb0\xda\x06\x85\xba\x8d\x06\x9e\x12\x99\x38\x40\xee\x90\x9d\xa0\x0a\x48\xdb\x39\x00\x54\xc8\x9d\x79\x29\x5e\xa3\x10\x17\x32\xc9\xa0\x82\xb8\xd8\x6d\x90\xbc\xca\xc8\x94\x06\xbd\xc9\x3b\x9d\xd2\x4c\x35\x06\x6e\xd5\x5d\x45\xb7\x01\xe6\x4f\xb7\xc5\x86\x64\x2c\xa0\x66\x40\x6a\x1b\xb1\x3b\x42\xda\x81\xdc\x6a\x7b\x6f\x4a\xe3\xc2\x14\x6c\xa2\xc7\x72\x92\xdf\xc6\x89\x7d\x36\xbc\x3b\xbd\x3d\xbf\x31\x90\x13\xd7\xb7\x97\x83\xfb\x71\x83\x5f\xbb\xe1\xad\xcb\xc1\xed\x0f\x67\xab\x5f\xfb\xfe\xbe\x9c\x95\xdd\xf0\xca\xed\xdd\xf2\x64\x8e\x0e\x43\x6c\x48\x0a\x6b\xec\xe7\x04\x65\x0b\x35\xe7\xcc\x87\x28\xc4\x25\xde\x74\x88\x4c\x46\xb0\x82\x10\x22\x21\x55\x83\xe3\xf0\x1e\xe2\x72\x56\x4b\x98\xe5\xcd\x32\x30\x6c\x3b\x15\x8d\xd6\x38\x91\x9f\x12\x3e\x01\xbf\x75\x5e\x2a\x71\xbb\x24\x02\x7d\xcb\x78\x9f\x33\x2a\xb3\x04\x2f\x6a\x3d\xac\xba\x72\xae\x70\x4a\x20\xe2\xb8\xc0\x8f\x73\xc9\x22\x7a\x67\x20\x81\xc9\xdf\xeb\x74\x0a\x99\x4c\x8a\x62\x45\xd0\x84\xa8\x67\xc8\x9b\x73\xbf\x7a\x5b\xaa\x0b\x18\x91\x47\x23\x06\xe6\x9c\x91\x5e\xe4\x38\x87\x68\xbf\xd1\x87\x03\x34\xfa\x10\x93\x27\x92\xf0\x4c\xef\xbc\xfe\xa1\xe5\x92\x19\xa6\x98\x26\x57\x5c\x79\xcb\xdc\x36\xfb\x29\x48\x44\x33\x90\xcc\xc7\x44\xb7\xfb\x7a\x82\x47\x89\x92\x1d\x3b\x83\x31\x20\x1c\xc7\x5a\xc9\x06\x56\xe6\x86\x57\x84\x00\xb1\x60\xea\xa5\x5a\x99\xeb\x88\x14\xde\xfc\x6d\x7a\x0c\xdb\x2c\x9b\x3d\x1b\x77\x80\x3d\xbd\xa0\x4b\x76\xdb\x8b\x5c\x6b\x25\x3f\x90\x05\xa4\x60\xdc\x60\x2a\x36\x74\xcd\x36\xc5\xbc\xbe\x88\x93\x76\xd8\xd0\xd1\x1e\xb9\x6b\x9b\xd7\x61\x3b\xc7\xad\x8f\xd5\x7b\x2d\x2d\xd5\xc5\x72\xf9\x8e\x3b\xaa\xad\x0f\x6d\x4a\x6a\x6b\x08\x03\xaa\x2a\x5e\x19\x89\xd6\xd0\xb8\xfc\x00\xef\xf4\x77\x2b\x35\x15\x2f\xae\x45\x61\x4d\x7f\xd8\x05\x9b\x1c\x5f\xcd\xc7\x27\x2b\x47\x1c\x25\x5c\x96\xb1\x72\x3a\x0f\xfa\xd4\x7e\xba\x6c\xdc\xc3\x90\x7c\xb5\x5c\xb8\x56\x40\x43\xc3\xc2\x57\xc0\x20\xcd\x3d\xa3\xac\x87\xcc\xbe\x7d\x80\x28\x44\x5b\x82\x42\x96\x14\xc8\x01\x2c\x46\x85\x1b\x64\xc4\x8a\x98\x15\x89\x9e\x49\x02\x61\x6e\x11\x4f\x33\x30\xf1\xdb\xe1\xda\x96\x48\x6c\x22\x86\x0f\x10\xcf\x95\x6e\xcc\xe4\xe4\x38\x23\xae\x4d\xf8\x29\xdc\x1e\xc6\xf7\x66\x83\xdf\x3d\xb0\xb4\xa1\x75\x73\x97\x52\x86\x3e\x11\x05\xad\x00\x70\x7f\x38\x41\xd0\x13\xaa\x21\x94\xcd\x6b\xbf\xc5\x89\xb2\x33\x59\x63\xe7\x0b\xe0\x94\xef\x12\x3e\x59\x6e\x24\x80\xc6\xd1\xc3\xed\xb9\xb3\x48\x16\xf1\x53\x01\x7a\x71\xc9\xa3\x38\xbc\xb9\x1d\x9e\x0e\xee\x87\x67\x47\xe8\x41\x12\xbd\x3c\x7e\xba\x90\x5f\xed\x55\x12\x33\x72\x8b\xc4\xc2\xa4\x22\xb8\xcd\x10\x42\x84\x28\x65\x41\xaf\x60\x1c\x65\x98\x96\xe5\x84\x0d\x20\x29\xd4\x1a\xea\x00\x58\xa8\x3a\x4f\x1b\x99\xb7\xea\x04\x42\x9c\xd4\xf8\xfd\x44\xa9\x99\xf1\xa6\xf5\xc8\xbc\x55\xe4\x53\x8e\xe8\x7b\xe9\xc9\xc0\xd1\x52\x73\x42\x05\xea\x34\x2d\x43\x54\xe3\xee\x73\x0a\x42\xdc\x2f\x71\xb6\x3c\xfd\x14\x3f\x97\x88\xd6\x88\xc2\x81\xef\xfe\xa5\xcf\x81\x63\x6b\x63\xc3\x0a\xb7\x9f\x60\xe1\xd0\x32\xbc\xd5\xf3\x4d\x93\xf1\x21\x9d\x91\x2c\x9c\x58\x65\x10\x36\x8e\x55\x22\x38\x3b\xf0\x0b\x65\xa8\x74\x25\x1e\xa0\x29\xfd\x6c\x1b\x2d\xe2\xdb\xdd\xab\x41\xc0\x43\x4b\x3c\xe5\x1c\xd7\xcf\xd4\x1a\x62\xc3\x0d\x7c\xbf\x54\x88\xe4\x52\x8b\x44\x91\x16\x97\x04\x89\xb8\xd0\x37\x05\x74\x5b\x78\x21\x56\x89\x0c\x0a\x0b\xbd\x28\x75\xaf\xcc\xb2\xd3\x5f\xd4\x20\x89\xb1\x22\x87\x5a\xf4\x5a\x91\x00\x6d\x73\x64\x20\x9b\x06\xab\x00\x0e\xac\xb8\x79\x26\x64\x86\x99\x0b\xcd\x6e\x19\xae\xbb\xf2\xb6\x60\x55\x5a\x05\xc2\x90\x1e\x06\xf2\x15\xa4\xfe\x94\xc6\x21\x33\x58\xcf\xa5\xe3\xb0\xd1\x2f\xfb\xb0\x6c\xcf\xd8\x07\xe3\xb4\x0c\x36\xcf\xe2\x7d\x1a\x6c\x82\xa5\x42\x76\x4c\x6d\xa6\x88\x40\x45\x7c\x59\x23\x6c\x49\xb7\xef\xaa\xbc\x69\x12\x2a\x6b\xb1\x04\x3c\x23\xd2\xe1\xa6\x18\x94\x18\xad\xd3\x38\x41\xd8\x94\x62\xf6\x67\xdb\xd6\x64\x76\xb7\x44\xc8\x4c\x20\x48\xbf\xde\xf4\x11\x1a\xb0\x1a\x5e\x96\x8b\xcb\x2a\xad\x97\xb9\x93\x70\xf2\x8c\x17\x12\x65\xc2\x40\xcb\x98\xc8\x7d\x37\x79\xd0\xc0\xca\x1f\xf9\x50\x08\xe5\x52\x27\x10\xd8\x62\x56\x07\xcd\x39\xb9\x77\xfc\x02\xae\xbc\x4a\x54\xb9\x17\xc8\x8b\xe6\x0a\x5b\x45\x07\x56\xa7\xc8\x38\x9a\x63\x36\x23\x63\x67\x64\xdd\x44\x5b\xd2\xed\x9c\x42\x33\x67\xb6\x95\xe6\xcb\xe9\xc6\x28\x4c\xb6\xfe\x8b\x79\xd5\x1b\x10\xf5\x21\x90\x0a\xcf\x08\x32\x23\xea\x64\x96\x2e\x45\x8c\x59\xb0\x61\xd0\x13\x6c\xab\xc3\x72\x14\x7d\x9b\xf0\x0e\xa1\x4f\x17\x78\x42\x92\xb7\x89\x9c\x80\xae\xad\x71\x1e\xbc\x75\x26\x1b\x80\xa0\x67\xb0\xe7\x57\x58\x86\xb5\xde\x8b\xbc\x29\x37\x60\xd9\x3c\x4b\xd5\xcf\xb7\x98\xa8\xab\x15\xb2\xc9\x54\xdb\x2a\x88\x84\xd7\x5e\x50\x69\xa3\xc9\xc0\x16\x5e\x7f\x55\x9b\xf2\x66\x03\x09\x0a\x7e\xb4\x8c\x63\xeb\x8a\x1f\x2b\xa7\xb2\x31\xc8\x40\xc7\x2a\x78\xe7\x53\xc4\x38\x23\x88\xca\xe2\x65\x55\x4e\x87\xf2\x10\x3d\x5a\xc4\x37\xc6\x17\x5f\xa5\xcb\x17\x5f\x7a\x69\x4b\x4b\x01\x9e\xe0\x6d\x03\x2e\xbf\x9b\x11\xad\xa8\x62\xb1\x00\x88\x4f\xc3\x87\xcb\x32\xdd\xca\x71\xee\x5c\xe0\xbe\x77\x08\xae\x41\xa4\xae\xe2\x08\xc4\xc8\xca\xe0\x90\xc1\x41\xb5\x2f\xd9\x8f\x2c\x4c\xcd\x88\x79\xcb\x06\x10\x22\x95\x28\xc5\x19\xf8\xf4\x18\x57\xc5\x57\x06\x76\x49\xf9\x2d\x3c\x70\x82\xb8\x34\x35\xb4\x5a\x56\x60\x95\x69\xc7\x5d\xbf\xc5\xba\x96\xe1\x2d\x1d\x34\xef\x8c\x3e\x11\xe6\x68\xfa\xc0\x9d\x09\x3d\x28\xd7\x69\xb2\x38\xc4\x10\x66\x4c\xe2\xd0\xf3\xb1\x9c\x23\x19\x83\xcc\x3e\xd8\x23\xbb\x2f\xd9\x7d\x63\x18\x8d\x01\x49\x2b\xa1\xdb\xbb\xc0\xf0\x90\x4a\x2d\x6e\xaf\xc9\x04\xc7\x12\xfd\x8e\x71\xf5\xbb\x00\xd9\xd8\x19\x2f\xe0\x53\x67\x82\x3a\xa8\x95\x6c\x81\x43\x6b\x09\x07\xe1\x00\x61\x6b\xe5\xca\x6f\x1b\x1b\x50\x04\xbe\xbf\xa8\x34\x3a\xac\x67\xc1\xb5\xd5\xbc\xea\x3d\xf6\xa8\x7a\x2d\x54\x0d\x9e\xa6\xac\x5e\x71\xd2\x4b\x86\x4e\xb9\xca\x45\xef\xf7\xa2\x93\x6b\xbe\x86\x08\xb0\x0d\xb5\xa5\x9d\x23\xa7\x56\x80\x20\x37\xdb\x25\x36\xc9\xf3\x6c\x93\xcb\x45\x39\x74\xcd\x96\xc1\x68\x41\xf9\x3d\x1a\xb1\x8f\x5c\xd8\x2b\x58\xda\x3a\x03\x13\x1c\x3d\x1e\x12\x16\x23\x9c\xab\xb9\x41\xdb\xb5\x7e\x85\x85\xa5\x06\x2d\x69\x00\xd9\x78\x28\x0d\x2a\x23\x2c\x62\x57\xf1\xe2\x89\xbb\x51\x8c\x58\xd0\x08\x54\x32\x80\x42\x4f\x50\xaa\xb6\x4d\xd5\x24\x52\xeb\x57\x6d\x6b\xd1\x54\x84\xb5\x56\x82\x75\xf9\x39\x2b\x15\x95\x85\x1a\x0c\x10\xe0\xc4\xa7\xf5\xd5\x39\x77\xd6\x46\xa7\xdf\x69\x7a\xae\x7b\x21\x0e\xac\x46\x61\x4c\x52\x76\x06\x5a\xd2\xf9\xd6\xf1\xda\x12\x6a\xf0\x34\x17\x10\xae\xdb\xd4\xe6\xd7\xd1\x9c\x26\x85\xef\xe2\x9b\x03\x3f\x4c\xdd\x64\x42\x9e\x48\x62\x30\xeb\x23\x01\x91\xf9\xc6\x6a\xf8\x2d\xfa\x3f\xa6\x30\x29\xfa\xc3\x88\x7d\x02\x36\x9c\x24\x0b\x40\xd4\xf4\x2d\x63\x55\x69\xe6\xb1\x71\x00\xca\xa6\x02\xa1\xf2\x40\xcc\x5e\xcf\xf1\x13\x19\x31\xd7\xcc\xff\x41\x8f\xe8\xf7\xe8\x0f\x6d\xea\x9d\x0b\xb0\x7f\x61\x3b\xc7\xc7\x20\x7c\x3d\xb8\xe5\x2c\xa3\xb4\xfc\xc6\x99\x41\x4a\x46\xc8\x06\x64\x0d\x0f\x8c\x4d\xd9\x13\x8f\x6a\x59\x1c\xe1\xa9\xc5\x82\x30\x35\x66\x3c\x26\x63\xd2\xe0\xd2\x5c\xc2\x24\xb4\x10\x70\xc5\x63\xb2\xd2\x21\xe9\x99\xe9\x4f\x60\xba\x91\xf9\xc4\x6f\x07\x24\xf8\xfb\x6c\x6e\x6f\x7d\x28\x53\x5a\xf3\xc8\x3d\xfa\xec\x26\xe3\xde\xd4\x99\xea\xc2\x44\x0f\xe0\x42\xb0\x03\x68\x76\xe8\x25\x58\x39\xf7\x7a\xf5\x38\x56\x1d\x01\xfa\x65\x3d\x73\x7b\x59\x05\xb8\xba\x50\xfb\x44\xd0\x19\xd5\xf2\x7b\x77\x87\x2d\x70\xc2\x4d\xbc\x19\x06\x64\xb4\x93\x3b\xa3\x58\x0a\x07\xb4\x72\xe8\xe9\xaf\x70\x42\x4e\x78\x5e\x15\xe0\xed\x02\x50\x19\xba\xfb\xad\xac\xbe\xd0\x7c\x78\x66\x32\x00\xc9\x9c\x9a\x9c\xfb\xc1\xe9\x05\xd2\xa7\x83\xa7\x06\x98\x0a\x16\x2d\x57\x73\x2e\xe8\xaf\xad\x19\x4a\xed\x32\x7a\xe1\x69\x2d\x12\xba\xcc\x38\xcb\xd2\x3a\x10\xab\x11\x29\x54\x49\x2b\x69\xd2\x99\xd0\x24\x07\x0c\x56\xcd\x66\xa7\x79\x62\x0a\x37\x44\x5c\xc4\xa6\x72\xba\x2c\xa5\x8f\x41\x18\xae\x13\xef\xb1\xf2\x0d\x52\x0b\x55\x69\x4b\x43\x18\x0b\xce\x52\x01\xf4\xaf\x39\xc9\x77\x94\x81\xf7\xa6\x31\xcb\xf7\x78\x26\x8b\x20\x64\xb3\x36\x9a\x37\x17\xeb\xfb\x0f\x3d\x53\x19\xe4\xac\x3a\xcb\xa2\x87\x80\x32\x2a\xb9\x29\x0c\xba\x96\x45\xe7\xd6\x40\xdf\xef\xc0\xa4\xf3\x1a\xf1\x1c\x75\x19\xa9\x81\xfd\x58\xf2\x7b\xf2\x19\x9c\x55\x16\xf1\x42\x76\x12\x57\x43\xa0\x22\x7d\xbc\xa0\xc9\x64\x03\x26\x57\x17\xaa\x97\x46\x45\x17\x06\x14\xcf\xd6\x1a\x92\xa9\x15\x87\xb4\x8b\x67\x41\x01\x21\x6e\x51\xbc\xec\x6b\xe0\xba\xeb\x22\xe4\x31\x5a\x4a\x31\x62\x2d\xc4\x75\xb8\x25\x5c\x34\xf3\xf8\x35\x0c\x10\xb6\xa1\x72\xd7\x75\xbf\x7d\xdb\x89\x30\x2c\x69\x5f\x8f\x44\x1d\x1e\x66\xe5\x61\xf0\x95\x40\xde\xc6\x80\xe8\x45\x9b\xd7\x3b\x19\x9e\x1c\xc7\x11\x8e\xe6\xad\x93\x9a\x70\x9e\x10\xcc\xda\xa4\xd7\xc6\xc7\xd5\x23\x62\xc0\x4d\x81\x75\x27\x09\x20\xfc\xba\x25\xb0\x55\x21\x0b\xf1\x9d\xc5\x80\xcc\x6e\x78\xb8\x89\x0d\x74\x03\x55\x84\x39\xcb\x0f\x65\xb3\x84\x54\xd7\xca\x42\xe8\x1f\xd8\x4e\x92\x28\x4f\x82\xb2\x90\x19\x11\x7a\xd4\x7a\x89\x9f\x08\xd3\x3a\x83\x1d\x87\x73\x66\x3c\xbb\x84\x68\x5f\x0c\xea\xc0\x77\xed\xfc\x69\x90\x75\x18\x8f\x18\x1c\x5c\x5e\x3e\xac\x9a\x56\xa5\x56\x33\x42\xbb\xd4\xc6\xa7\x33\x10\x22\xd6\x3e\x9e\x77\x65\x33\xf1\xda\x67\xd2\xf4\x3d\x86\x18\x83\xad\x5d\x6b\x81\xfb\xa5\x80\x6a\x30\x1b\xeb\xe0\xb8\x5e\xc9\x88\x0c\x51\x1b\xe5\xb0\xd3\x20\x68\xa3\x0d\x0e\xea\x45\xef\x92\xa2\xfc\x85\xbb\x0d\x3a\x0e\x65\xa9\xab\xba\xa3\xe3\x19\xac\x93\xcb\xce\xed\x85\x0d\xd9\x2e\xbb\x6c\x7d\x7e\x4f\x11\xe6\x68\x0b\xbc\x2a\x81\x01\x9d\x00\x72\xca\x7f\x32\x1a\x36\x95\xc6\x02\xe6\xca\x5c\xa4\x99\x5a\xd8\xaa\x68\x70\x2f\x86\x39\xbd\x06\xf1\xad\xc9\x3d\x5c\xbd\x23\xe3\x92\x83\xb8\xa9\x33\xe8\xc8\x9a\x15\x1a\x9b\x74\x0b\x1d\x22\x88\x54\x10\x1b\xda\xa2\x41\x4c\x81\xd9\x31\x4e\x5a\x6d\x59\x3b\x60\x9a\x90\x66\x5b\xa0\x34\x58\xf0\x57\x25\x72\xa2\x79\x17\x4e\x92\xca\xbc\x30\xa4\x43\x2b\x5f\x64\x6e\x52\x54\xc2\xed\xee\xac\x4e\xf0\x84\xac\xe5\x9e\xbe\x30\x1f\x2c\xa5\x22\x78\x05\x22\xbb\xb3\x2c\x59\x74\x8b\x28\x0f\x43\xef\x1a\x41\xd2\x56\x0d\x2c\x84\x56\x5b\x7a\x37\x95\xe1\xc9\x36\x1b\xa2\x24\x51\x2e\xa8\x5a\x8c\xad\xd1\xaf\x3b\xd3\xba\xb3\x5f\x9e\xda\x0f\xbb\x68\xd4\x27\xc8\xf5\xe7\x8c\x8c\x70\x4f\x09\x6a\x2a\xe8\xd8\x29\x74\xd9\x6e\xad\x25\x37\x82\x27\x2d\x5b\x58\x87\xde\xd4\x6d\xa8\xba\x8b\x4d\x87\x67\x2b\x73\x8c\xf9\xd4\xe1\x22\x75\x5f\xd8\x6a\xc9\x92\x35\xac\xa5\x0e\x7e\x39\x13\x94\x0b\x5b\x19\xa4\x4b\x50\x5b\x8a\x3f\x8f\x33\x2c\x70\x92\x90\x84\xca\x74\x73\xdb\xee\x9f\xfe\xb8\x74\xb4\xa7\xa6\x82\x8d\xb4\xf5\xa0\x3e\xd3\x34\x4f\x11\xcb\xd3\x89\x95\x72\xb1\x7c\x0c\xc1\x2f\x5d\xaa\xbe\xc1\x70\x72\x03\x2c\x01\x06\x88\x00\xce\x74\xc4\x02\x60\x6b\x6b\xaa\xc0\xd1\x9c\x92\x27\x80\xdd\x14\x8c\x48\x79\x84\xae\xb8\x22\x27\xe8\x12\x67\xf7\x20\xa8\x99\x92\x92\x33\x63\x1d\xc7\x12\x69\xa9\x35\x67\x54\x1d\x8c\x98\x45\xc3\x76\xab\x72\x1c\x71\x66\x10\x51\x23\x58\x58\xdf\x04\x98\x7b\x1d\x34\xa8\x72\x89\x8d\x54\xb6\x2c\xb6\xc0\xcf\xe3\x20\x7a\x75\x6c\xb2\x03\xd6\xa0\xe3\x5b\xfc\x6c\xe2\xb5\xcf\xb0\xc2\xa6\x5a\xec\x32\xc9\xdd\x06\x44\xd9\x0a\x42\x06\x08\xd8\x05\x8e\x70\x8b\x46\xe1\x6b\x9f\x99\xe8\xd4\xaf\xe9\x11\x39\x42\xdf\x25\x7c\x22\x0f\x90\xf4\xa0\xd9\xf0\x50\x12\x25\x0f\x8c\x83\x0a\xfe\x6d\x52\xc1\xbe\x71\xab\x5f\xf0\x7d\x28\xfb\x37\xa5\x9f\x0d\x08\x86\xfc\xd3\xc9\xf1\x71\xba\x38\x9c\xe4\xd1\x23\x51\xfa\x2f\x90\x29\x1a\x57\xc8\x21\x48\xe1\x26\x3c\xaa\x55\xab\x53\xc7\xb2\xea\x44\x91\x36\xad\x46\x12\xc0\x4d\xd7\x57\xba\x2f\xac\xea\xa0\x8f\x38\x6b\xae\x1a\x69\xa7\x2c\xf2\xb6\xe3\x55\x02\x5c\x7e\x1d\x6d\xc5\x14\x8e\x0d\x71\x9e\xa7\x09\x9e\x55\x54\x96\x35\x94\x94\xeb\x94\x5a\x2a\xd2\x73\x87\x78\x0b\x7d\xca\xca\x51\x66\x5f\x39\x77\x24\xb8\x15\xad\xbb\xe5\x68\xc4\x06\x12\x3d\x13\x53\x0f\x16\x72\x12\xc1\x3b\x91\x53\x39\xf7\x19\x89\x60\x2f\x85\x46\x0d\x1c\xae\x41\x4d\xb0\x8a\xa3\xd3\xac\x9c\xff\xc6\x6a\xa0\x38\x91\xe4\x40\x37\x0c\x90\x68\x2e\x90\x10\x3d\x0b\x9c\x65\x44\x8c\x98\x85\x36\x05\x00\x6f\xce\x6d\x90\x48\x5b\x34\x79\xaf\x51\xbe\xae\x46\x19\x26\x7d\x94\x13\x16\x57\x9d\x6f\xc8\x6f\x5c\x9a\xea\xb1\x24\x37\x50\xcb\xa2\x5d\x23\xbd\xdf\xde\x6c\xdc\x71\xcc\xab\xb4\xf3\x41\x25\x4c\x1f\xca\x4d\xa7\xa0\x40\xca\xa2\xaa\xa6\xb3\xf5\x79\xf5\xbd\x24\xe6\x00\x32\x36\x7c\x1c\x73\x22\x03\x23\x3e\xf2\xb6\xb8\x84\x4e\x89\x96\x3e\x46\x4c\x93\x71\xe8\x70\x30\x00\xdb\x0e\x6f\x5b\x77\x1a\x09\x2e\xa5\x8d\xbc\x37\xed\x2c\xcf\x9f\xda\xa2\x96\x9f\x41\x09\x3f\xbf\xbe\x1a\xd7\xab\xfa\x05\xcf\x5c\x7d\x3f\xfb\xb0\x31\xc9\xbe\xb5\xa9\x95\xd5\xfc\x8a\xb5\x58\xa3\x9e\xdf\xf1\xe9\xc5\xb9\x2f\x62\x55\xe9\xba\x5e\xd0\x2f\x44\x56\x6f\x2f\xe9\x57\x9f\x71\x50\xdc\xaf\xd2\xc4\x92\xf2\x7e\xab\x37\xab\x1c\xef\xbb\x0d\x6c\x5e\x65\xeb\x57\xf2\x87\x32\xcd\xac\x0a\x4b\xdf\xd1\x36\xb5\x5c\x2b\x11\x08\x8c\x2f\xed\x61\x07\xc1\x4b\xbf\x25\x15\x4e\xb3\x30\xe5\xd2\xe1\x86\xda\x69\x9a\xa3\xd6\x76\x09\xbe\x2a\x9e\x79\x84\x4d\x34\x4b\x75\x70\xb5\xad\x58\xcf\xe3\x75\x6f\x61\xd2\x77\x11\xc6\xfc\x7a\x39\xcc\xc9\xa2\x88\xda\x93\x56\x76\x73\x25\xb8\x5b\xec\xfe\x13\xe2\x21\xe1\x5b\x37\x74\xdb\x24\x45\x0f\x1d\x25\x08\x96\x36\x1c\x03\x72\xf9\x2a\x79\x3e\x6b\x98\x87\xfd\x98\x4d\x36\xf0\xa1\x2f\xc2\x10\x5c\x35\xb6\xae\x58\xe4\x0e\x22\x15\x82\x3c\x11\x01\xb4\x63\x63\x7e\x58\xf9\xa8\xe2\x44\x10\x1c\x2f\x82\x15\xf1\x01\x07\xa6\x67\x30\x8f\x49\x9a\x6a\x05\x1e\x54\x13\xc6\x0f\x79\xe6\x74\x96\xd2\x5b\x50\x41\x83\x4e\xf5\x8d\x15\x84\x2b\xe8\x2f\xd8\x21\xf9\x4c\xa5\xd2\x72\x45\x43\xac\xa6\x6b\x04\x24\x1e\xa8\xab\x35\x27\xf6\x86\x1b\x7d\x18\x7c\x77\x7d\x7b\x3f\x3c\x1b\x7d\x28\xa2\xf3\x5d\xfa\x99\x47\x84\x72\x00\xff\x9c\x8d\x98\x0f\xa8\xf5\x00\xc8\xb0\x97\x08\xc7\x71\x81\x6c\x60\x95\x48\x23\xb3\x2d\xe5\xc8\xc1\xa9\x58\x19\x4a\xbb\xa4\x99\x07\xc8\x41\xda\xd7\x93\xb5\xc4\x75\x56\x3a\x39\x26\x93\x6a\x49\xca\xcb\x8e\x2e\x9b\x10\xbb\x55\x19\x5d\x9b\x28\x07\x2e\xc8\xc8\xb3\xd3\x95\xe0\x76\x3e\xc6\xe6\x12\x5e\x8f\xdb\xb9\x0d\xd9\x60\x53\x3f\xd2\xcf\x24\xbe\x6d\x91\xaa\x76\x92\xd1\xd2\x29\x12\xb0\x71\x17\x72\x46\xd7\xd1\xf8\xfd\x54\x1e\xf4\x77\xdd\xd9\xd2\x75\x01\xc9\x56\xc0\xab\x02\xb6\xaa\x42\x18\x45\x44\x28\x4c\x19\x9a\xc2\xc1\x66\xd1\x02\x01\x60\x07\x01\x1f\xf6\x1f\x51\x4a\x19\x20\x07\x2c\x5b\xda\x87\xf2\x3c\xd6\x10\x5a\x2f\xcf\xaf\x1e\xee\x4b\xa2\xea\xf7\xd7\x0f\xe5\xa2\xee\x83\x9f\x97\xca\xaa\x95\x16\x96\x05\x0b\x05\x53\x2c\xb2\x10\x2d\xca\xac\x5f\x99\xc6\x89\x26\x0b\x45\x1e\x6e\x2f\xb6\x92\xef\x9a\x9d\x65\xad\x18\xe1\xa1\x74\xd5\x8c\x88\xd0\xe5\xd3\x98\x44\xab\x50\x4c\xbb\xd3\x91\x89\x82\xd2\xeb\x60\xad\x89\x16\xe1\x0c\x4b\x94\x61\x61\xfd\x50\xb1\x09\x80\x2a\x57\x06\x33\x9a\xd7\x32\x04\x89\x4f\x44\xfd\xa8\xaf\x3e\xce\x76\x91\x05\x61\x45\x59\xf0\x8f\x92\xf1\x93\x69\x78\x8d\x93\x66\x87\xb2\x24\xd5\xc5\x09\xcb\xd0\x03\xb2\x3d\x84\xb8\x0b\x47\xa6\x42\xfc\x40\x37\x07\x2b\xe2\xe2\x09\xb5\x4a\xca\x99\xa6\x48\x03\xa7\xea\x30\x58\x83\xe6\xf8\xd4\x7c\xdc\x11\x91\x2e\x88\x6a\xd7\x6d\x15\x4b\x89\x06\x37\xe7\x0d\x6b\x7d\x51\x75\x21\x7d\x59\xe5\x6c\x12\xef\xcd\xda\x35\x48\x52\x90\x9e\xb8\x17\xa8\x48\x76\xa6\xdb\xc1\x20\x19\xa7\xff\x4d\x39\x92\x60\x1f\xd0\x7a\x9b\x54\x86\x52\xda\xf1\x0a\x60\xde\xf5\x32\xf1\x8a\x65\x58\x13\xf4\x28\x1c\x90\x4d\x03\x09\x81\x7e\xea\x31\xc6\x07\x21\xf0\x0f\x37\x05\x73\x6d\x6c\xc1\xce\xc0\x90\x8a\xd9\x74\x41\x43\xfa\xd1\x50\xb4\x07\xcb\x00\xf8\x0f\x57\x90\xd1\xc5\x06\xdb\xdc\xf5\x70\xba\x21\xb5\xad\x07\xa0\x54\x8c\xcf\x99\xbf\x2d\x16\x35\xce\xb0\xb5\x3b\x80\x12\xe5\x2a\x25\x34\x15\xd6\x3b\x1a\xb1\x20\x60\x45\x1a\xb5\x47\x9f\x11\x57\x9c\x04\x2a\xde\x32\x00\xb6\x86\x24\x1d\x2f\xfc\x94\x76\xa0\x9a\x22\xaf\xe6\xe5\xf2\x22\xb5\x7e\xec\xe9\x94\x73\xec\x12\x11\x9d\x05\xc5\xc6\x01\x86\xf6\x25\x68\x2f\x28\x28\x60\x3b\x06\x73\x34\x18\x2d\x70\x50\xae\x2e\x48\x5e\x8f\x39\x91\xec\x2b\xe5\x53\x3d\x69\x62\x4b\xa2\xe0\xaa\x7b\x40\x4b\x75\x98\xda\x96\x97\x1f\xf0\x1d\xa0\x33\xad\xab\x38\x04\xc7\x6a\xa5\x99\xca\xf9\x78\x81\x12\xc2\x58\x24\xe8\xb4\xcd\xaa\xfe\x39\x23\xd1\x26\x10\x32\x37\x58\xe0\x94\x28\x22\x96\x85\x23\x95\x8b\x49\x83\x88\xe3\x76\xd0\xf6\x6b\x76\xd1\x54\xda\xa8\x96\x64\xf1\xda\xed\xc5\x2a\x48\x18\x3f\x8b\xb5\xd0\xaf\xf4\x34\x7e\xb4\x96\xff\x35\x67\x61\xfb\x29\xa6\x61\xa3\xad\x02\x04\xa0\x6d\xe7\xf4\x3a\x50\x28\xf7\x35\x50\x91\x52\xb8\xd0\x9e\x60\xa0\xac\x1e\x65\x1b\xf8\xc9\x2a\x5e\xba\x13\xde\xed\x32\x1c\x5c\x0a\x6d\xe5\x50\x95\x72\x27\x80\x4a\x40\xa5\x32\x38\x20\xcd\x00\x26\x20\xb4\x34\x45\x48\x06\x6e\x3f\x0b\x6f\x57\x18\x74\xad\x64\x55\x2d\x2e\x55\x59\xae\x15\x3c\x6e\x57\xe0\x0e\xbd\x44\xb3\x6b\x89\x66\x15\x29\x97\xa2\x6b\x35\x75\x12\x51\xc1\x99\xb1\x45\x9f\x2d\x40\x40\x79\x82\x90\x7b\x64\xaf\x48\x5b\x39\x16\xae\x7e\xca\xfc\xbf\xca\x1c\xdc\x11\x75\x48\xaa\x4d\x49\x95\x47\x81\x0b\x0a\x3c\x50\x49\x28\x0d\xd8\xb8\x1a\x18\xad\x09\x83\x34\x56\xfe\xf3\x2b\xe3\xc0\x82\xe4\xe6\x05\xcf\xd1\x33\x95\x73\xa4\xf8\x88\x41\x9c\xa0\xf7\x06\x28\x8e\xcc\x8b\x07\xf0\x16\xc0\x20\xc8\x7c\x92\x52\x85\x70\x30\xc3\x92\x49\xf2\xc0\x9e\x67\xfd\x01\xcc\xb8\x31\xcf\xbe\x09\xa2\x67\xc5\xa1\xd9\xc0\xbe\x56\x34\xb2\x6d\x2a\x7d\x10\xd3\xfc\xb2\xc9\xf4\x81\xc6\x13\x6a\x98\x8d\x67\xae\xcf\xa6\x47\xcd\xd6\x06\x0b\x1a\x0a\xc8\xae\x54\xaa\xca\xdd\x62\x0d\x3d\x2b\x32\xe9\x8b\x8d\xe8\x94\x4a\x5f\xbc\xbe\x8b\x5c\xfa\xb6\x32\x65\xcb\x72\x2b\xdd\x27\x2d\xf6\x6f\x97\xb3\xab\xb8\x0b\x9c\x0f\x25\xa5\x9b\x56\x49\x69\xdf\x50\xcd\x8a\x84\x80\xcd\xc3\xcb\xd7\x51\x07\x8b\xfc\xac\x90\x8a\x82\x74\xcb\x32\x26\x0c\xa9\x72\x7e\xc6\x15\xe4\xd4\x44\x50\xc2\xbd\x96\xe7\x39\x62\xcd\x12\xc8\x72\x9e\xb8\x6d\x8a\xc6\x4e\xd1\xcf\x82\xf3\xe7\x66\x61\x2d\x5a\x3f\xf9\x20\x37\xa3\x2c\xdb\x62\xec\x55\x11\xb3\x70\xf1\xb5\x05\x27\x69\xc1\x63\x93\x84\xe3\x86\x53\xd9\x3c\xf4\x5a\x02\xc5\xca\x73\x61\x2f\xdd\x1d\xaa\x76\x35\xee\xdc\x39\xdf\xc4\xcb\xc8\x96\x1b\xdb\x80\xe9\xd8\xeb\xf1\x15\x57\xed\x26\x55\x66\x01\x55\x74\x67\x58\xa8\x55\x78\x02\xdd\xf8\x01\xb8\x77\xed\xd8\xb1\x09\x75\xf1\x08\xdd\x95\x3d\x29\xcd\xd8\x56\xef\x7f\x89\x59\xaf\x5b\x1a\x38\xf0\x37\x0a\x1b\xef\x4b\x43\xcb\x01\xd4\x04\xb6\x61\x92\x15\x4e\xec\x85\xbb\x9c\xc5\x44\x30\x82\xd5\xfc\xf5\xb2\x2c\x4e\xb7\x35\x4f\x07\xe3\x7b\xd9\x8c\x8b\xd3\x9d\xd4\x85\x0f\x87\x5b\xae\x91\xbe\x72\x9c\xfa\xf5\x2e\x96\x22\x1b\x7c\xe0\xab\x04\xd7\x54\xc7\x06\xb3\x61\x00\x42\xb3\x0e\x95\x6e\x95\x88\xd1\xac\xce\xbd\x4c\x4a\x4a\x83\xdd\xa7\x96\x8c\xa2\x4f\x7b\x58\x5b\x79\xc5\x92\x7c\x11\xb9\x1f\x2f\x9f\x8e\xb0\xac\x8a\x73\x1e\x64\x28\x40\x29\x6d\x85\x29\xb3\xdc\x6b\x59\x52\x82\x96\x29\x53\xdc\x94\x87\xb0\xf7\x19\x2e\x5f\x7c\x82\x4b\x9f\xee\xd0\xa7\x3b\x34\xec\x51\x9f\xee\x80\xd0\xbe\xa5\x3b\xac\x52\x41\x97\x19\x40\xbd\x4f\x0e\xaa\x6d\x96\x4a\xdc\x98\xfd\x5d\xa1\x47\x6e\x1e\xd2\xef\x6c\x88\x61\x3c\x94\xfd\xc5\xfe\xd0\x18\x12\x55\xfb\xac\x3a\xdb\xd0\x9e\xc9\x16\x55\xb7\x00\x16\x71\x62\x71\xe8\x6c\xc0\x72\xd9\xfe\xb4\xcc\x54\x3a\x62\xdf\xf3\x67\xf2\x44\xc4\x01\xc2\x0a\xa5\x5c\x2a\xe0\xc3\x2e\x3e\x06\x0e\x42\x09\xd2\xdc\xc4\x41\x60\x74\x85\x53\x12\x9b\x8a\x87\x41\x58\xa3\x35\xd8\x5a\x57\x6b\x13\xdc\x2a\x20\x87\x9a\x6d\x70\x71\x13\x23\x66\x42\x0d\x4d\x78\x1b\xc8\x0a\xd4\x4d\x0c\x08\xe6\x77\xde\x11\xfc\xbb\x23\x74\xaf\xef\x27\x2a\xcb\xe3\x0d\xd0\xd7\xda\xc6\x36\x62\x33\xc1\xf3\xcc\xdb\xd0\xf8\xc4\x94\xbe\x35\xd1\x4f\x75\x47\x30\x0c\xc6\x79\x81\x23\x1c\x6b\x5d\x7c\x39\xe1\xbc\x49\x14\xea\x46\x10\x46\x21\x01\xe9\x63\xe8\x43\xeb\x6c\xa8\xbb\xf1\xdf\x06\xc0\x2d\xcb\x80\xd8\x5f\xc8\xb9\x7c\x46\x24\x58\x85\xbc\xd5\xbd\x94\x47\x5e\xc6\x2a\x68\x1c\xe7\x32\x9b\xa8\xf7\x5b\x38\xdb\x7e\x33\x0c\x42\xd1\xb9\x8d\xf9\x32\x49\xaa\xf6\x9e\x78\x31\x6b\x69\xe7\xe8\xd9\x36\x7e\x71\x93\x8b\x8c\x83\x24\x96\x2c\x1c\x6c\x83\x45\x7a\xcb\x78\x96\x9b\xb8\x36\x1a\x86\x39\x35\x52\x36\x95\xea\x12\xab\x68\xae\x39\x77\x81\x78\xb6\xa3\x78\xbf\x82\x2b\xbf\xac\x05\xb5\x61\x06\xa7\x61\xef\x2d\x2e\x85\x0e\x16\x75\x73\xef\xbb\xf0\x7a\x27\x49\xa4\xba\x3f\xe3\x76\xb3\x05\xad\x03\xbb\xa8\xfb\xc4\x3e\xd1\x13\x5d\x45\x45\xab\xc6\xdf\x8d\xb6\xca\x15\xb7\x76\x1e\x49\xb8\x05\x84\xcc\x99\x05\xec\x2a\x5e\xb4\x15\x5a\x5b\xdc\xff\x82\x6e\x96\x05\x64\x51\xf2\x9f\xb4\x38\xe2\x2d\xae\x29\xce\xb4\x12\xa1\xb8\xbe\x25\xc5\xcc\xc8\xb1\x26\x4e\x16\x61\x94\x0b\xea\xce\x7e\x25\x27\xbc\x9d\x3a\xc0\x42\x79\x1c\x56\x54\x8a\x70\x50\x6c\xce\x38\xfc\x71\xa4\x72\xec\x03\x13\x81\x26\x5c\x11\x74\x93\xff\xee\x1c\xeb\xc2\x89\x77\x0d\x7b\xba\x92\xb0\xb7\xd8\x65\xdc\x84\x6f\xd8\xe9\xa4\x51\x36\x0b\xc0\x11\x9b\xad\xc4\x5d\x6a\x1f\x34\x7e\xd9\xad\x7e\x43\xe3\xa7\x4e\xf6\xd9\xe4\xdb\x25\xe0\x4d\x1b\xc7\x66\x97\xe2\xdc\x6d\x20\xac\x95\x9e\x42\xd8\x4a\x6b\xbf\x03\xf4\x59\x0a\x8e\x7a\x6c\xa5\xa9\xff\xf2\x7f\x99\x5a\x59\x66\x69\xfe\x0b\x71\x31\x62\xe6\xf7\x03\x5f\xa7\x42\xbf\x50\x00\xc0\xe2\x94\x14\x10\x99\xa2\x0c\xa6\x07\x90\x22\x16\x0c\xcd\x80\xfd\x7a\x98\x7e\x3d\x86\xc7\x7c\x42\x04\x23\x7a\x68\x0e\x7c\xc0\x33\xb3\x14\x33\x3c\x03\x68\xe1\x03\x88\x8c\x03\x71\xb5\x50\x45\x0c\x49\x9b\x7a\x87\xc0\xad\x34\xb3\xb4\xf9\xb6\x45\xdd\x5f\xe8\xd3\x88\xb2\x16\xd9\xb4\x08\xaf\x68\xa6\xfe\x5b\xdb\xff\x66\x12\xfb\xfd\xe0\xee\x87\xf1\xed\xf0\xee\xfa\xe1\xf6\xb4\x24\xb6\x9f\x5e\x3c\xdc\xdd\x0f\x6f\x1b\x9f\x15\xb9\xaa\x7f\x7d\x18\x3e\xb4\x3c\x72\x0d\x5c\x0c\xbe\x1b\x96\x8a\x68\xff\xf5\x61\x70\x71\x7e\xff\xf3\xf8\xfa\xe3\xf8\x6e\x78\xfb\xe3\xf9\xe9\x70\x7c\x77\x33\x3c\x3d\xff\x78\x7e\x3a\xd0\x5f\x86\xef\xde\x5c\x3c\x7c\x3a\xbf\x1a\xbb\xb0\xe3\xf0\xd1\x4f\xd7\xb7\x3f\x7c\xbc\xb8\xfe\x69\x1c\x74\x79\x7d\xf5\xf1\xfc\x53\xd3\x2c\x06\x77\x77\xe7\x9f\xae\x2e\x87\x57\xcb\x8b\x75\x37\xaf\x46\x6b\x1d\xe0\xe0\x22\x0b\x8c\x46\x81\x98\x34\x59\x58\xd2\xa6\xbf\x82\xef\xe2\xc6\xd0\xe3\xe1\x81\xfb\xcb\x94\xd6\x3e\xd4\x2c\xd0\xf9\xc5\x0a\xee\x31\x62\xde\x71\xe9\x2f\x55\x85\x67\xd2\xa5\x1e\x97\x46\x7b\x82\x06\x70\x56\x40\x61\x28\x75\x0a\x99\x0d\x7e\xa4\xce\xd5\x0d\x74\x98\xd0\x94\x82\xd7\x1b\x1d\xa2\xea\x86\x97\x1b\xb4\x73\x82\x21\x58\xbf\x5d\xbc\xec\x34\xc8\x6a\x56\x33\x50\xca\x09\x72\x1c\x9a\x18\x73\x82\xc1\x9e\x5d\x30\x9c\xd2\xa8\x9a\x82\x01\xf0\xab\xa8\x80\x1a\xa9\xb6\x58\x22\xb0\x72\xcb\x73\x82\x7e\xf8\x4b\x31\x28\xf0\x60\x58\xcd\x3b\xaf\x55\xd4\xb3\x0f\x44\x6e\x56\x75\x15\x79\x96\x7a\x72\xc7\xdc\x9a\x96\xe1\xdc\xda\xca\xdd\xe0\x6e\xca\x59\x00\x37\x56\xf2\x3d\xe9\xe3\x6d\x66\x54\xa1\xf1\x13\x74\x07\x50\x27\xb2\x50\xdd\xf5\x2e\x66\x49\x3e\xa3\x0c\xd1\x34\x4b\x48\x51\xf3\x7d\x42\xe6\xf8\x89\x72\x57\xbe\xc2\x54\xf9\x80\x75\xb4\xa2\x15\x3a\x44\xad\x07\xe5\x04\x0d\xe2\x58\x96\x19\x5c\x89\x72\x1c\xcb\x3c\x2c\x0f\x3b\x44\x08\x63\xb1\x67\x9b\x15\x3a\x2a\x8e\x1c\xac\xd8\xee\xc1\x5c\xea\xec\xb0\x7c\xf7\x6e\x71\xfd\xeb\x15\x1c\x3b\x52\x1e\x6f\x24\x0c\xdc\x63\xf9\xe8\x58\xf3\x2a\x81\xc0\xc1\xea\x6c\xd7\xa3\xc5\xd7\xe9\xda\xa9\x5f\xd9\x31\x1c\xb4\xcd\xfa\x6c\x45\x85\x5e\xd1\xa5\x9b\x71\x52\x29\xdd\xd5\xb9\xbf\x52\xe9\xaf\xc6\xce\x76\xea\xed\x69\x96\xc6\xe0\x48\x8e\x3d\xfd\xaf\x31\x8f\x1b\xf8\xf4\xda\x7f\xb9\x54\x64\x1b\x07\xeb\xb6\xae\x0f\xa8\x96\xa4\x6b\xfd\x40\x4b\xe9\x70\x47\xf0\x4e\xdd\x85\x41\x28\xbc\x40\x23\x70\xf7\x61\xca\x6c\x39\x1e\xe2\xfd\x51\xae\x00\xb5\x3e\xc7\xbe\x44\x1c\x9e\xf0\xa7\x92\x72\x99\x12\x29\x71\x0b\x60\x49\x60\x12\xdb\x86\x31\xf8\x13\x6a\x3f\xec\x48\x4f\xee\x4c\xde\xeb\xaf\x96\x19\x7d\x6e\x43\xcd\xd8\x4d\x54\x0b\xac\xb1\x8b\xb4\x45\xd7\x26\xdf\x4e\xf3\x97\x83\x22\x98\x86\x8b\x20\xc6\xa8\xcd\xfd\xd3\xd1\xac\x56\x5d\xb0\xc6\x2a\x4b\xa1\x0b\x6f\xfd\x18\x9c\xa0\xf5\x8d\x11\xb1\xad\x5f\x05\x97\xd7\x67\x0d\xaa\x2b\xf9\x3b\xc3\x0a\xd4\x11\x4f\x53\x23\x17\x94\x6c\xa9\x07\x08\x9b\x34\xc7\x42\x9a\x92\x79\x34\x37\x5e\x26\x7d\x65\x1c\x8c\xd8\x73\xb0\x21\xa5\x40\xe0\x41\xd8\x12\xa0\x89\x7e\xd6\xc7\x8d\x3e\x95\xc2\xab\x41\x64\xa4\x10\xeb\x1b\x10\x82\x71\x08\x16\xe5\xa3\x56\x10\x78\xb0\x5f\x5b\x90\xfa\x06\xb5\x02\x2b\xeb\xdb\x56\x31\xd0\xcf\x2d\x28\xd4\xb7\x85\xa6\xdc\x75\x08\x41\xad\xc0\xa6\x11\xec\xa0\x54\xe0\xab\xc2\x7b\xfb\x74\x4d\x93\xdd\x9b\x4e\x2c\x46\x85\x9e\xae\x5b\xed\xdf\xbb\x19\xfd\xde\xf8\x1d\xf2\x16\x50\x93\xa0\x35\x8f\xf0\x8d\x0e\xb5\xcc\xea\x92\xed\x6d\x20\x86\x44\x87\x06\x35\xf0\x2b\x88\xb4\x1c\xdc\x9c\x7f\x75\x80\xbe\x0a\xb3\xcd\xbe\xda\xe8\x00\xda\x71\xdb\x72\x81\xa0\x4d\x95\x52\x0e\xca\xc7\x0e\xf6\xaa\x72\x12\xed\x9e\xd9\x83\x88\xda\xce\xa1\xfe\xb2\xf4\x0d\x38\xa7\xa1\xfc\x9d\xf1\xdf\xfa\x80\x67\xeb\x02\x32\x32\x2e\x95\x0d\x6b\x17\x8f\xd8\x64\x51\x75\xf2\x1c\x78\x2f\x4f\xe7\x53\xba\x75\x49\x37\xdd\x5e\x3d\x3d\x79\xc7\x81\xb8\xcb\xef\x83\x15\x09\xcf\x03\x13\x73\xcd\xa7\x01\x17\x6b\x8b\x52\xe8\x23\xd8\x9b\x66\x55\xb2\x97\xb9\xc5\x6c\xdc\x94\x55\xf2\xcf\x7b\x23\xb7\x0e\x61\xdf\x83\xa6\x15\xb1\x11\xff\x2d\xc2\x75\x4f\x65\x2f\x4b\x65\xbb\xc8\x78\x28\x0f\x6e\xfd\x0b\xf4\xd4\xc8\x71\x41\x33\xce\xe0\xaa\x95\x09\xcf\xe0\x4b\x75\xff\x56\x17\xcc\x5d\xd3\xe7\x1b\xac\xc9\x6a\xa7\xef\x9d\x09\x1c\x30\x6e\xd7\xfa\x58\xab\x43\x1d\x28\x5b\x84\x88\x53\x93\xdd\xa8\x68\x4a\x0e\x10\x67\xc9\x22\x08\x76\xb0\xe7\x15\xc8\xcd\xc4\x28\xcd\x09\x15\xae\x13\x8b\x31\xb8\x56\x3a\xfc\x9a\xd2\x78\x1b\x8d\x6c\x11\x69\x72\x35\xb8\x1c\x9e\x8d\x87\x57\xf7\xe7\xf7\x3f\x37\xe0\x47\x96\x1f\x3b\x08\xc9\xe0\x85\xbb\x9f\xef\xee\x87\x97\xe3\x4f\xc3\xab\xe1\xed\xe0\x7e\x05\xbc\xe4\xb2\xce\xda\xa0\x0b\x73\xd9\xa4\xbe\xad\x03\x5f\xe8\xcc\xbc\x0d\xbd\xd7\x41\x26\x83\x4e\x28\x69\x01\x9a\x34\xa9\xff\x2c\x26\x02\xc5\xe4\x89\x24\x3c\x2b\xcc\xaa\x8d\x0b\x16\x20\x50\x36\xb4\xbf\x0c\x85\x12\xda\xac\xae\xf1\x09\x32\xb5\xde\x82\x72\xb7\xbe\x41\x10\xf9\xb0\x20\xec\x2b\x85\xc8\xe7\x2c\xa1\x11\x55\x41\x6a\x20\x17\xd6\xbd\x62\xdc\x87\x10\x9d\xba\x82\xb8\x76\x16\x8d\xb2\x73\x9d\x3f\xf4\xa4\xd7\xb5\x7d\x7f\xa2\x3c\x22\xda\xca\x02\x42\x3b\x50\xec\x5b\x9c\xc6\x35\xc0\xb6\x0d\x46\xf7\x12\xe6\x81\x7a\x8e\x8e\x4d\xef\x6b\x01\x73\x6b\x1e\xe4\xea\xdb\x70\x59\x9c\x4c\xe9\x5c\x2f\x0f\x94\xe9\x46\xa9\x6f\x1c\xee\x52\x2a\xac\xb9\x03\xe4\x0d\x1b\xbb\xbe\x66\xc0\x42\xad\x5e\x0c\x33\x31\xa7\x18\x09\x92\x72\xa5\x15\x30\x13\x11\x70\xa0\x85\x2a\x8a\x13\xfa\x2b\x60\x54\x09\x72\x14\x44\x50\x38\x64\xaf\xc2\x7d\x60\xf1\x23\x8e\x46\xec\x6c\x78\x73\x3b\x3c\xd5\x0c\xe9\x08\x3d\x48\x80\x9f\x2a\x4d\xfd\xcc\x92\xb7\x11\xc7\xc2\x48\x06\xca\xa4\x22\xb8\x2d\x18\x8c\x08\xc1\x45\x77\xfe\xe0\xfb\x1b\xc2\x77\xcd\xe4\x0d\xcf\x4a\xb6\x29\x67\x00\xb8\x6a\xad\x8a\x1c\xe4\x0c\xec\x3c\x27\xeb\x16\x3f\x97\x56\x24\x84\xdf\x00\x49\xa4\xbc\xea\x2f\xb8\xda\x00\xe0\xd9\x7d\x7e\xa5\x3e\x6f\xe0\xdb\x65\xf3\xbc\x87\x10\x3b\xa9\x0a\x34\x50\x03\x18\xea\xab\xde\x54\xe6\xd9\x2a\x2a\x8a\xb7\x80\xea\xa8\x90\xfe\x84\xcc\x30\x43\x22\x67\xac\x02\x0f\x1b\x5a\xda\xea\x41\x33\xeb\x1e\x55\xbd\x66\x38\xe5\x39\x03\xa5\x01\xc2\x58\x1b\x06\x23\x33\xc2\xd4\x8a\xc1\xbc\x15\x10\x4b\x65\xa8\xfb\x8b\xc5\xd2\x30\xd0\x36\x38\x96\x26\x7f\x12\x94\x5e\x5e\xef\x5a\x76\x41\x79\x25\xa7\x92\x3e\x54\xfe\x7e\x6e\xd6\xb2\xb1\x7c\xdc\xba\xbb\x7b\x2c\x1f\x57\x77\x15\x93\xe8\x71\xdd\xcb\xa6\x9a\x99\x99\xd8\xca\xd5\x35\x63\xdf\x42\x3f\xb5\xa5\x59\xa0\x60\x79\xf4\x88\xbe\xbf\xbf\xbc\x40\x53\xaa\xe5\x5e\x7d\xad\x5c\x61\x2d\x63\x3f\x88\xc4\xd9\x85\xad\x6d\x35\x17\x89\xbf\x7b\x61\xe3\x9d\x28\x15\x48\x09\xfa\x46\xc3\x33\xe2\x8c\xbd\xc2\xa2\xed\x55\x4a\xb3\x08\xcc\x62\x9e\x9a\x79\x1c\xcb\x7c\x3a\xa5\x9f\x8f\x14\x16\xdf\xb4\xac\x87\x89\xaa\x18\xff\x9d\x4f\xc6\x7a\x44\x5b\x5e\xc4\x4d\xcd\x21\x5b\x50\xd9\x2f\x9b\x9d\xd9\x99\x79\xf7\xff\xf2\x09\x64\xbb\x67\x82\x03\x02\x20\x78\xe7\x6c\xa4\x82\x7d\xc5\x51\xd2\x11\x72\x09\x54\x25\x90\x93\x88\x0b\x41\x6c\x92\xbc\xa9\x2d\x9a\x61\xa1\x28\x58\x6b\x1d\x48\x4a\x09\x1d\xbf\xd8\xa2\xb0\xe4\xf7\x1c\x17\x48\xd4\x13\x42\xc0\xc1\x93\xd1\x64\x3d\xa5\xf7\xb4\xe4\x9b\xac\x9c\x40\x1b\x79\x6a\x71\x33\xc1\x20\xb3\x52\xc4\x1a\x3e\x11\xa6\x76\xa2\x9f\x40\x13\x0d\x69\xfb\xdd\xbc\x0c\xa6\xc4\xe7\xf9\x59\x71\xb9\xb9\x90\xde\x30\xaa\x49\x09\x0c\xf7\xbc\x4d\x94\xb2\x2e\xf5\x36\x47\xff\x53\x67\xdf\x31\xbc\x5a\x5f\x97\x15\xa1\xf1\x76\xb5\x8b\x52\xdf\x45\x58\xab\x83\xf6\xdf\x10\xc8\x47\x12\x63\xc5\x08\x00\x24\xac\x72\x5a\xdd\x73\xd3\xa7\xa6\xad\x4a\x97\x2b\xb7\x7c\x03\xd4\x9a\x52\x33\x9f\x08\xa4\x74\xee\x22\x10\x7d\x9d\xdc\x7d\x18\xc8\x83\x48\x20\x84\x7a\xa9\x15\xcb\x94\x19\xd7\x9c\xcf\x4b\x76\xb8\x83\x8c\x6e\x06\xa3\x85\x46\x92\x09\x12\xe9\xab\xec\x04\xdd\x24\x44\x4b\x5e\xb9\x96\xbe\xf2\x24\x71\x08\x5f\xcb\xa5\xc3\xb5\x50\xe9\x5e\x7c\x5e\x81\xee\xb1\x64\x62\x0e\xe1\x6e\xf9\xcc\x82\x35\xd8\x3d\xe2\x42\xb0\xbe\x60\x42\x06\x43\x62\x59\x8b\x04\x0e\xbf\x30\x71\xb3\x60\x4a\xc2\xa5\x8b\x8c\xfe\xaa\xd9\xaf\x20\x72\xce\x5b\x93\x1c\xc3\xd9\xbe\xcc\x1c\xdc\x52\xbe\xe0\x24\xdc\x7d\xd8\x16\x57\xdd\x41\xae\xa9\xdc\x81\x25\x11\x67\xd9\x14\x7d\xf5\x07\x1f\xfd\x61\xf1\x56\xed\xdd\x6a\x87\x06\xb7\x64\x61\x6a\x0b\xb1\xcf\x0a\xd7\x45\xa1\xcc\x2c\x8c\xef\xd5\x7f\x5e\x18\x90\x8b\x94\x00\xaa\x64\x51\x75\x0e\xe9\xbb\xb6\x6d\x8b\xf5\x3c\xc7\xb9\x58\x0b\x92\xa2\x40\x2d\x5f\x87\x73\xdb\x64\x94\x62\x58\x7a\x11\x9a\xd9\xa5\x2d\x26\x01\x62\xb4\x0d\x35\x92\x25\x24\x38\x4b\x36\x66\x19\x1b\x55\xbc\x76\xa6\xbc\xad\x5b\x0d\xa4\xe4\x42\x94\x79\x29\xef\x5a\x89\x02\x4b\x13\xe8\xb1\xc5\xd6\xc7\x16\xb3\x95\x45\x3c\xed\x01\x12\xa0\x12\x90\xf8\x5f\x38\xd0\xaa\x82\x83\x35\x7a\xaf\xca\x7c\x2a\xed\x4e\xa7\x34\xa7\xd2\x17\x9a\x97\x9c\x6d\xe9\x81\xd3\x93\x59\x8c\x21\x71\x74\x9b\x28\x9c\xd2\xfc\x8d\xf7\x00\xda\x24\x31\x32\xe8\x05\x06\xf9\xd8\xae\x9d\xf7\x9c\x64\x58\x10\xa6\x46\xec\x56\x8f\xc2\x7c\x51\x44\x62\xb8\x38\x1c\x87\x46\x0f\x35\x6b\xa7\x08\xdb\xaf\x60\xd1\xdb\x02\xe1\xe4\xd8\xbc\x04\xaa\xe9\x0b\x26\xd9\x7f\x67\xde\x31\x98\x07\x16\xf3\x47\x4f\x95\x4e\x0b\x35\x5e\x0b\x90\xd1\x9c\x02\xe4\x40\x4c\xa4\xbd\x90\xa8\xb2\x98\x12\x5e\xfc\xce\x89\xc3\x5f\x86\xcf\x3c\xff\x6a\x62\xd8\xce\x50\xc0\x9c\x81\x4e\x8e\x58\xd0\xc7\x12\xb8\x4e\xa3\xac\x6f\xa8\x4a\xc0\x3e\xd3\xd8\x3b\xbe\xe0\x9f\x66\x87\xb8\xa0\x33\xca\x82\xa2\x49\x76\x7a\x29\xce\xc0\xbc\x6b\xce\x20\x9f\xfa\x3b\xed\xde\x66\x19\x1c\xc1\x88\xff\xe7\xbf\xff\x76\x44\xdb\xbc\x1f\x72\x6c\x57\x60\x1f\x76\x72\xbd\x6d\x09\x77\x3e\x40\x11\x69\x41\xa7\x08\x74\x5a\x59\xca\x9c\x28\x7e\xb5\x97\x9b\x26\x1a\xae\xe6\xc6\xdd\x5b\x26\x77\xf0\x8d\x88\x7c\xc9\xd9\x30\x57\xcc\xdb\xae\x25\x95\x90\x1d\xa0\x47\x62\x4e\xb2\x37\x10\x84\x05\xc9\x6b\x66\x9a\x11\x2b\x3e\x91\x06\x0f\xc5\xc0\xbb\x9a\x1f\x8a\xd5\xe9\xb8\x30\xcb\x78\x7f\x11\x29\x51\xb8\xc3\x83\x68\x64\x57\x3e\xc3\x44\x91\xea\xf6\x2b\x37\x6d\x85\x73\x07\x38\x87\xdb\x44\x6d\xce\xb1\x7c\xb9\xd0\x9c\xc6\xb2\x4f\xc6\x9a\x1e\x0a\x0f\xab\x82\x74\xcc\x20\x4d\xb6\xa7\xde\x90\x5c\x12\x61\x38\x9d\x87\xc3\xb2\x94\x10\xa2\x38\x42\x8c\xe6\x0a\x5f\x23\x49\x31\x5d\x2b\x9f\x40\xbf\xdf\x8c\x31\x59\x72\x36\xe0\x19\x11\xe3\x38\x57\xb5\x63\xb1\x2c\xc6\x5f\x7f\x74\x96\xab\xc5\xea\xf6\x65\x82\xeb\x65\x6f\x96\xe1\x7a\xea\xf7\x5b\x9a\x5d\x2d\x31\x07\x21\x3e\x65\xa9\xb9\x05\x35\x93\x54\x50\x33\x6d\xcc\x69\xc9\x44\x02\x37\x30\x53\x00\xa9\x57\x68\x52\xf6\x8a\x36\xd8\xde\x30\x72\x34\xc9\x0b\x93\x92\xaf\x96\x10\x1f\x8d\xd8\x47\x53\x6e\x04\xb4\x3c\x33\x80\x08\x12\x7e\xc8\xe7\x8c\x4b\x52\xca\x40\x6b\xa8\x80\x60\x33\x48\xed\x30\x9a\x85\xf5\xe2\xa3\xed\x65\xf5\x37\xc7\x3f\xad\x6f\x78\x7d\xca\xcd\x14\xb8\x95\x38\x18\xd1\x8c\x6a\xda\x19\x37\x9e\xb4\x97\xab\xc2\x5b\xc4\x74\x01\x0e\x96\x4a\x16\x07\xc8\x4f\xaf\x42\x10\x09\x79\x22\x60\x4e\x87\x31\x86\x75\x2e\xca\x76\xbd\x16\x76\xb2\xea\x00\x15\xe9\x9f\xc0\x16\x50\x5c\x1d\x41\x39\x49\xae\x89\x16\xcb\xe9\x3f\x5b\x67\xaa\x35\x05\xa6\xac\x21\x9e\x0f\xc2\x7a\x1f\x0b\xa2\x10\xf9\xac\x88\xad\x08\x7a\xef\x72\x09\xeb\xe9\x07\xa8\x39\x1d\xaa\x5d\x76\x7c\xf1\xda\xcc\x03\x97\x41\xee\x92\x25\x63\x77\xe5\xdb\xe4\xc1\x39\x66\xb1\xcd\x88\xb5\x4a\x86\x16\xb6\x60\x76\xc6\xe8\xe6\x73\x05\x6c\x5e\x67\x00\x94\x6e\xda\x34\x88\xee\x70\x91\x39\x85\x51\xab\x2c\x10\x5e\xc1\x85\x96\xdc\x73\xa6\x68\xa2\x89\xc3\x8e\x41\xa2\x29\x44\xc6\x59\xa0\x42\x88\x6c\x6f\xc3\xc2\xa3\x52\x52\x36\x1b\xdb\x95\x74\xc9\x9d\xdd\x2e\x86\x32\x4d\x5d\x9a\xa6\xcc\x8f\xdf\xb9\x86\x96\x1b\xd5\x0d\x59\x03\x4e\x99\x4b\x2b\x05\x8d\x83\x71\x37\x19\x0b\x30\xe7\xb2\x51\xc7\x34\x36\x4b\x41\x4d\xe1\x69\x98\xe8\x3a\x76\x77\x90\xe9\xea\x38\x0e\xc5\x15\x22\x6d\xaa\xa8\x49\x00\x83\x48\x7d\xd5\x92\x0b\x2b\x5b\x73\x60\xcf\x99\x17\xd1\x6c\xd9\x2b\x9f\xe9\x5f\x49\xa7\xc5\xae\x3b\x9b\x8e\x80\x93\x64\x82\xa3\x47\xaf\x85\x79\x5b\x04\x17\xae\x6c\x80\x96\x2b\xa1\x2e\x9a\x21\x2e\x3d\xd0\x08\xa4\x9b\xd0\x5b\x68\x90\x7c\xec\xb0\x8b\xce\xcd\xaa\x59\x88\x34\x03\xdd\x64\x46\x6f\x72\x1b\x62\x92\x25\x7c\x91\xb6\xdc\x67\xd5\x14\xc2\x6d\x22\x75\xda\x32\x18\x77\x7a\x95\x55\x98\xde\xda\x97\x59\x2d\x1f\x69\x07\xb8\x52\x6b\x70\xc9\x4f\x09\x9f\x80\x49\xd5\x9a\x1f\x5c\x8e\x4d\x90\xea\x51\x3d\xcf\xeb\x66\xfe\x54\x4f\x24\x95\x59\xa2\x95\x99\xf6\x1e\x4c\xce\xc9\xcb\xee\x9b\xc1\x28\x58\x6d\x1d\xec\x1e\xad\xdd\xf8\xf9\x4b\x20\x18\x5f\x38\x49\xc0\xbc\x6b\xf8\x57\xc5\xca\x66\x92\xfd\x8e\x8c\x93\x5a\xf1\x11\x53\x78\xe6\x36\xd7\x0a\x97\xfc\x99\x11\x21\xe7\x34\x2b\xd5\x4b\xdc\x3a\x3c\xdc\x52\xb4\xfd\x8f\x09\x86\x5e\x83\x77\xf2\xec\xd0\x20\x94\x68\xfa\x90\x19\x8e\x0a\xab\x68\x94\x60\x29\xe9\x74\x11\x00\x8b\xf8\x48\x5b\x48\xdf\x2a\x9b\x11\x82\x12\x65\x4d\x8c\xc6\x8c\x6f\x37\x99\xf5\xdb\x67\x15\x3e\x94\x8f\x1f\x8d\x43\x04\x37\x7d\x9f\xd4\x51\x64\xdc\x4d\x6d\xd1\x64\x5a\x91\x68\x0d\x84\xc0\x66\x99\xf0\x4b\xc1\x7f\xda\xcd\x08\x85\x30\x69\x87\x6d\x15\x19\x0f\xf8\x11\x82\xe1\xa8\x52\x2a\x25\x6c\xbe\x56\x9c\x9c\x5d\x58\x13\xa7\x07\x0b\x01\x4c\x85\xe2\xe3\x03\x24\xb7\x02\xd9\xea\x42\x17\x67\x24\x21\x3b\x89\xb8\xde\x80\x48\xaa\xe1\x0c\x01\x79\x2c\x25\x8d\xa2\xcc\xc0\x6a\xe3\xc2\x06\x81\xe0\x2d\x50\x3d\xcd\x43\xff\xc9\x0c\xd4\xc6\x82\x37\xed\x22\x18\x06\x61\x95\xbb\xea\x2e\x4d\x98\x7f\xa6\x05\x4b\x72\x45\x37\x25\xba\x2a\x3a\xf5\xf2\xca\x3e\x92\xda\x1b\x87\x4c\xd7\xc6\xf5\x89\x74\x09\xef\x58\x79\x00\x36\xe2\x40\x75\x3e\xdd\x8d\x2e\xac\x03\x55\x71\x34\x23\xca\x54\xff\x8f\xe9\x13\x8d\x73\x9c\xbc\x2b\x9a\xd8\x59\xc2\xc7\x8e\x56\xbf\xf9\x90\xaf\x77\x6a\xef\x88\x92\xee\x4a\xa8\xc1\x28\xda\xcd\xd9\xc3\x2d\xd8\x8f\x63\x69\x04\xd7\x2f\x5e\x6e\xd9\x1a\x24\xc1\x8e\xcc\x42\x05\xfc\x86\x04\x2a\x33\xc7\xd8\x7e\xe1\x61\x01\x4a\x80\x58\xb8\x04\x22\x68\xd6\xe8\xed\xb9\x5e\x95\xb4\xbf\x74\xd1\x6b\x7d\x1a\xaf\x8e\xaa\xa0\xee\x5e\x1e\x5c\x4f\x1e\x74\x20\x9b\x7b\x78\xf9\xb7\x1d\x83\xfd\xbc\x7f\xf6\x40\x38\xac\x5d\x89\xbb\x13\x11\xdf\x11\x99\xec\x85\xa4\x58\xdb\x8a\xd7\x92\x17\x0f\x1d\xca\x51\x81\x19\xb4\xbf\x5b\xb4\x1f\x27\xf9\xd6\xba\x81\x5e\xee\x82\x5d\x4d\x2f\x3b\xa1\x0f\x00\xfc\xc4\x90\x17\x9d\xdb\x0a\x22\x70\x78\x83\x58\xba\x9a\xef\x61\x45\x94\xa2\x1d\x5e\xa7\xf8\xc4\xda\x72\xbe\xc4\xf6\xda\x24\xb8\xce\x9b\xfb\x92\xa4\xb6\xee\x58\x76\xa1\xa3\xbc\xb0\x17\xc7\x52\x63\xf0\x41\x1f\x2c\xdc\xed\x16\x6d\x80\xd6\x71\x5b\xb6\xcb\x43\xd6\x54\xf6\x6d\xfb\x34\x7e\x97\xe3\x37\xce\x04\x99\xd2\xcf\x1b\x89\xe2\x37\xf0\xa9\x55\x2f\xf5\x32\x57\x0a\xc9\x81\x7b\x06\x0a\xcf\x05\x01\x8d\x76\xa5\x6d\xb1\xa9\x11\x2b\x32\x23\x6d\x5a\xe4\x23\x59\x20\x2e\x4a\x3f\x6d\x0a\x02\xb9\xfb\xa2\x77\x66\x5f\xe7\x4a\x65\xf2\xe4\xf8\x78\x46\xd5\x3c\x9f\x1c\x45\x3c\x35\x71\xf8\x5c\xcc\xcc\x1f\xc7\x54\xca\x9c\xc8\xe3\x3f\xfe\xe1\x0f\xc5\x16\x4f\x70\xf4\x38\x33\xb0\x3a\x75\xbf\x53\x79\xcb\x09\x96\xdb\x45\xf6\xb8\x14\xb6\x17\x4e\x65\x0e\xba\x71\xc9\xa3\xfa\x1b\xa9\x70\x9a\x85\xa1\xa0\xa6\x6c\x9c\x54\xb8\x28\x56\x01\x79\x89\x7a\x9a\x68\x8e\xb3\x8c\xb0\x76\xb3\x83\x49\x34\xdd\x82\xf5\xb8\x54\x55\x3b\x42\xf2\x39\x4b\x30\x2b\xc3\x2f\x40\xe5\x25\x41\x22\xc2\x94\x85\x06\x28\x4a\x49\x03\x35\x1a\x08\x20\xc3\xff\xd7\x4b\x45\x84\x39\x52\x59\x94\x54\x73\xc3\xb1\xe5\x4d\x5d\xd1\x4b\x1c\x2c\x5d\xb5\xa4\x6c\xb1\x76\xc4\xad\xda\xb2\x24\xc5\xbb\x7a\x69\xf1\xf5\x4b\xda\x08\xce\xc6\xe4\xb3\x66\x72\x72\x53\xc0\xae\x07\x49\x24\x1a\xfc\x74\x87\xe4\x82\x29\xfc\xf9\x04\x5d\x52\x06\x02\xec\xf7\x3c\x17\x12\x9d\xe1\xc5\x21\x9f\x1e\xa6\x9c\xa9\x39\xba\x84\xff\xb5\x3f\x3d\x13\xf2\x88\x7e\x26\x58\x58\xfe\x60\x4b\xd2\xf9\xfa\xe6\x9a\x84\x44\xce\x24\x22\x4f\xfa\x84\xfe\xe1\x3f\x51\x6a\x5a\x3e\x41\xdf\x1e\xff\xe1\x3f\xd1\xef\xe0\xff\xff\xff\xe8\x77\x2d\x9a\xfe\x7a\x90\x5f\x50\xb9\xf8\xb6\xec\xce\x3d\xa8\xac\xd4\x06\xd5\xdc\x4f\x05\x2f\x76\xaa\xb1\xe5\x47\x1a\x3d\xf2\xe9\x74\xac\x09\xc3\x24\xf2\x8d\xb1\xa8\xc1\x45\x6f\x88\x9f\x4a\x6d\xed\x69\x53\xc9\xae\xa8\x21\x63\x3b\x35\x88\x0f\x8e\x5d\xcb\xbc\xa8\xbc\x0b\x41\x44\xa5\x6a\xc6\x54\xc2\x57\x24\xd6\x5c\x75\x9d\xd3\xe1\xac\x7b\x2e\xf9\xdb\x59\x70\x42\x84\x94\xb0\x9e\xba\x0f\xfc\x0b\xa3\x58\x4d\xa0\x8f\x5d\xc8\xc6\xe3\x50\x0b\xaf\xfd\x62\x62\x26\x61\x6a\x6f\x15\x2f\x29\x6b\x9d\xaf\x0e\x95\xbc\xe3\x62\x2b\x7d\xeb\x91\xb4\xa6\x32\xac\xa8\x97\xe4\x6a\xf8\x86\x95\xfd\x21\x43\x9c\x0b\x8f\x63\x6c\xec\x22\xb6\xaa\xe2\x6a\x2b\x26\x15\x26\xb8\xac\xdb\xa1\xd7\x53\x3f\xf3\x9f\xac\x1a\x26\x44\x9a\xb9\xb7\x8b\x7a\x71\x30\x5a\x2d\x22\x69\x96\xd8\x30\xe2\x06\xb0\xc3\x55\x1b\x7a\xe7\xf1\x2d\xa0\x71\x08\x7b\x84\xfc\x0d\xe6\x24\x5b\x0b\x20\xd0\xbc\x9f\xb9\x88\xc8\x29\xdf\x2e\xec\x35\xa1\xac\x16\x2f\xdf\xbd\x14\x91\x5f\xbd\x0b\x5b\x74\xca\xe1\x01\xf3\xb8\x50\x16\x8c\x5b\xc0\x56\xa1\x08\x80\x48\xcb\xb3\x01\x40\xbb\x5d\x60\x5d\xd6\x6a\x23\x6c\xc1\xb5\x8d\xe1\xb8\x60\x78\xae\xb4\x46\xa5\xa2\x86\xc0\x9a\x17\x2e\x89\x5d\x83\xa0\xa2\xad\xc7\x11\x54\x89\x29\x22\x95\x2a\xd5\xd8\xb1\x29\x95\x22\x36\xc4\x2a\x35\x05\x9b\x0e\x90\xc0\x10\x94\xa9\xe6\xba\x3d\x49\xc4\xe1\x14\x47\x94\xcd\x0e\x02\x98\x4a\x80\x8c\x08\xaf\x83\x26\x22\xbd\xc7\xf2\x71\xb7\x81\x86\x5b\x17\xb0\xa4\x71\x51\x44\xcd\x02\xcb\x18\xc7\x06\xad\x61\xf4\x29\x2c\x1f\xdb\x90\x95\x6a\xb0\x6e\x4b\x46\xe7\x97\xc2\x81\xc1\x2d\x1b\x9f\x4b\x41\x27\xa1\x3e\x05\x35\x1b\x5c\x49\x65\x0b\xf2\xe8\x32\xfe\xb0\x47\x61\xa9\xa2\x9b\x2e\x19\xbf\x9c\x73\xa1\xc6\x1b\xe2\xc2\x56\xd3\xe8\x19\x39\x4c\x00\xd0\x85\x3f\x11\xf1\x44\xc9\x73\x19\x5e\x75\x1d\x5a\x34\x46\xb3\x20\xaa\x0e\xf0\x37\xd3\x8c\x43\x0a\xcd\x14\xa5\x98\x2d\x0c\xa3\xd4\xcc\x05\xcb\x47\xe9\x0b\xb9\x22\x99\xe2\x24\x39\x40\x82\xe4\xd2\x14\x38\x96\x24\x99\x1e\xba\x52\x18\x31\x4a\xf8\x8c\x46\x38\x41\x93\x84\x47\x8f\xd2\x64\xb8\xb1\x99\x61\x52\x99\xe0\x11\x91\x32\x90\xac\x8a\x6c\x76\x9b\x63\x08\x55\x5c\x15\x11\x29\x65\x54\x2a\x1a\x39\x91\xa9\x00\xa5\x30\xb5\xc4\x23\x0c\x26\x61\xc8\xd8\x84\xe1\x6a\x49\x8f\x18\x70\xce\x9c\xd9\xa2\x49\x70\x5d\x5b\xcc\x3d\x17\x24\xde\x76\x80\x76\x00\x21\xe8\x28\x64\xac\xca\x07\x72\xc5\x91\x3a\xb5\x9f\xc1\x31\x5e\x46\x02\xb7\xe5\x13\xe5\x09\xd2\x9f\xb4\x12\xac\x11\xc4\x94\xfb\x10\xf8\x92\xe4\xe2\x23\xc3\xf7\x0c\xd1\x0c\x86\xdc\x82\x63\xb6\x8a\xa6\xf5\x2a\x82\xc8\x03\x75\xba\xaa\x5e\x73\xca\xa2\x24\x8f\x7d\xa5\x46\x2d\x02\x3c\x69\x22\x71\xcb\xa3\xd7\x5e\x0b\x0a\x07\x08\x4b\xf4\x4c\x92\x44\xff\xd7\x44\xc0\x1f\xfa\xc2\x09\x9a\x25\x9b\xe2\x16\xd0\x89\xe3\xd2\x6d\x14\xb5\x77\xe8\x94\x37\x58\xcd\x4d\xce\x7f\xca\x95\x29\x92\x69\xd0\x29\x9d\x7d\xcb\xc0\x19\x4e\x12\x3e\x81\x93\x0e\xc0\x95\x2e\xcf\x35\x48\xab\xcb\xa3\x88\x90\x98\xc4\xe8\xeb\xe0\xe0\x7a\x3c\x8a\x6f\x9a\x61\x14\x4b\x2b\xb2\x07\xa0\x95\x55\xc3\x5a\x2b\x74\x65\xb9\xce\xdb\x11\xba\xa9\x00\xb3\x84\xf5\xdb\x71\x15\xa6\xeb\xa0\xb6\x85\x6f\x03\x74\x59\x99\xc4\xcb\xed\xd0\x9a\x40\x97\xa5\x3e\x77\x00\x74\x59\x99\x67\x4b\xec\x3e\x9f\xbd\x68\xce\xb1\x9e\xd4\x05\xef\x9e\x08\x66\x00\xc2\xcc\xdd\x59\x22\x41\x77\x20\x17\x4d\x84\xb8\x5f\x20\x9e\x95\x6a\x88\x6f\x0b\xe2\x59\x19\xcc\x3e\x83\x78\x56\x86\xba\xbf\x20\x9e\x0d\x03\xed\x00\xe2\x69\x9c\xfb\x63\x4d\xd4\xdd\x98\x02\x24\xb6\x4c\xf2\xe9\x1d\xa4\x7a\x2f\x1d\xe3\xa9\x09\x1c\x30\xd7\x98\xbb\xa3\x2d\xa6\x35\x8c\xd6\xe6\x40\xb6\x85\x43\x55\x9c\x10\xeb\xd2\x9e\xf7\xbe\x19\xf0\x87\x75\xcd\xee\x07\xa1\xb5\x1b\xec\x90\x11\xce\x6c\x4e\x79\x5b\xa9\x99\xfd\xc9\x9e\xdd\x0c\x1f\x15\x30\x08\x4b\x2c\xbf\x13\x82\xd8\x65\xa5\x6a\xc3\x9c\x3f\xdb\xca\x49\x40\x86\x86\x28\x5b\x49\x10\x3a\x1d\x5b\xa5\xad\x6d\xe5\x28\x53\x64\x56\xd5\x69\x8b\x43\x43\x99\xfa\xd3\x1f\x57\x72\x22\x03\xb1\xe8\xd4\xc3\xa0\x76\x82\x77\x76\xd8\x67\x24\x46\xd1\x5c\x6b\x45\x52\xab\x2f\x7a\x3a\xe6\x66\x95\x28\xc5\xd4\x29\x52\xb9\x34\xae\x25\x2a\x47\xac\x84\x49\x7a\x84\x3e\x42\x41\x58\x9c\x66\x5a\xff\xf2\xf3\xa3\x9a\x92\x46\xf9\xb7\xdf\xfe\x89\xa0\x6f\x51\x4a\x30\x2b\xe9\xb0\xa0\x36\xe9\xab\x0f\x30\xfc\xd4\x9c\x8c\x58\xe3\x56\xa0\xe1\x67\x53\x63\xca\xc5\xfb\x9d\xb3\x29\x77\x3a\x31\x14\x3a\xc4\xd1\x1c\xc9\x7c\x62\x2a\xf5\x06\x36\x0c\x27\x48\x5f\xf0\x19\x38\xaa\xe1\x46\x76\x83\x5e\x76\x0a\x5f\x36\x06\xc0\xba\x1b\xbb\xde\xc6\x03\xb8\x47\x0e\x25\x29\x61\x3b\x35\x38\xcd\x0c\xe7\x0b\x0f\xbe\x34\xb8\x2f\x07\xc6\x87\xa0\xf5\x33\x6c\x2d\xfb\x5a\x96\x86\x70\x5e\xf0\x92\xe5\x09\x16\xf6\xe8\x8f\x98\x56\x34\x04\x79\xa2\x3c\x97\xc9\x02\xc5\x9c\x91\x03\xa0\x84\x3c\x9a\x1b\xc7\xaa\xd6\x59\xb0\x2d\x58\xf1\x44\x65\xae\x15\x5a\x68\xcb\xd5\xc7\x90\x0a\x1b\x4c\xaa\x39\x85\x7e\xb4\xfa\x4d\xe0\xab\x30\x4b\x0e\x75\xd3\xa2\x42\xd8\xd8\x0a\xcf\xef\x08\x1b\x5b\xa2\xaa\x1e\x36\xd6\xc3\xc6\xd6\xd7\x65\x1f\x61\x63\x2b\x7b\xde\x0d\x36\xb6\x69\xcb\x37\x80\x8d\x2d\x35\xf3\xc5\xc0\xc6\x56\x56\xf4\x8b\x81\x8d\xad\xcc\xab\x87\x8d\xfd\xf2\x60\x63\xb7\x04\x46\x6d\xe6\xc5\x06\x5f\x49\x51\xb6\x58\x9b\xc8\xbe\x92\xe8\xfc\x5a\x13\x58\xf4\x58\x0e\x6a\xf3\xd7\xd5\xf6\x60\xac\xcd\x4c\x68\x3d\x30\xd6\x46\x55\xbd\x9d\xd5\x6d\x0b\xf0\x04\x8a\xc1\x2b\x83\xb1\x96\x26\xd0\xc7\x57\xae\x1f\x5f\xd9\x48\x7c\xb6\x6f\x3d\x3c\x17\x74\x59\xbd\x90\x3b\xc2\xb1\x96\xf6\xa7\x53\x24\x26\x88\xee\x3b\xa0\xc4\x97\x95\xe6\xef\x4b\x87\x7c\xa5\x2c\x1f\xae\xa2\xb4\xc0\xd0\x5a\xc2\x73\x68\x71\x46\x09\x0f\xfd\xff\x3d\xe5\x6e\x10\x19\x5c\x59\x5e\xef\x57\x31\xb4\xd8\x81\x54\x3b\x53\xa8\xd3\x4a\x77\x93\x28\xeb\x92\x27\xd7\x74\x31\xbb\x41\xdc\x65\x24\x6a\xb1\x31\xd3\x94\xee\xaa\xd9\x55\x17\x99\xc7\xc2\x02\x85\xbc\x96\x17\xaa\xaf\x27\x33\x1c\x23\xe3\x57\xd2\x61\x01\xa8\xc3\x7c\x39\xa3\x52\x89\xd6\xd8\xa6\xda\x08\xb7\x71\x95\x66\x79\xe7\x80\x98\x60\x55\x67\x9b\x7d\x96\x92\x94\x8b\x55\x81\x55\x8d\x5f\xda\x52\x37\x9b\x7c\x4a\xb2\x39\x49\xb5\x24\x33\x5e\xb7\x91\xae\xfb\xed\x93\x86\x6d\xee\x9a\x09\x74\x2c\x11\x41\xe0\x08\xd5\xef\xc6\x06\x91\xb2\xf3\x76\x6f\xbb\xcd\x16\x33\x73\x4d\x87\x90\x03\x53\x5e\x6e\x70\xb3\x2f\x95\xdc\xdd\x40\xdf\x8d\x31\x1d\x3e\xa4\x66\x75\xd4\xc6\x92\x78\x8d\x65\xb8\x53\xc5\x57\xb6\x10\xf4\x1a\xae\xfc\xb2\x77\x5e\x73\xc2\xb0\x0a\xf0\xfa\x01\x1e\x2d\xa8\xa9\xf5\xe5\x81\xc8\x1c\x49\xc4\x61\xa8\x19\x94\x06\x53\x5f\xaf\x12\x95\x38\x8d\x72\x0b\x22\xc9\x45\x6b\x94\x69\x17\x83\x76\xa4\x72\x9c\x80\x26\x11\x56\xaf\xac\x6e\xea\x64\xd1\x90\xf6\xd8\xcd\x63\x42\x99\xfa\xf3\x7f\xac\xb5\x9b\x5a\xb5\xb2\xeb\x06\x15\xb7\x70\x14\x11\x69\x6c\xec\x36\x0a\x19\x4f\xf8\x13\x14\xdb\xda\x66\x57\xf5\x51\xd6\xf3\xd6\x0c\xde\x43\x11\xc7\x05\xa9\x1b\x71\x61\x2e\x78\x3e\x9b\x3b\x1b\x92\x3e\x33\x7a\x6a\x4d\x7b\xf9\x63\xcd\x46\xbe\xf6\x5e\x7e\x97\xd3\x64\x33\x0b\xdd\x5d\xa9\x0c\xd9\xa7\xf3\x7b\x24\xe7\xfe\xb4\x4e\xa0\xd9\xc6\x8d\xad\x0f\xba\x7b\x9f\xf6\x5b\xef\xaf\x81\x6e\x0e\x1c\xfc\xe6\x94\x27\x09\x78\x1a\x24\x49\x9f\x88\x68\xee\x1e\x26\x7c\x4f\xd7\x43\xce\xf3\x03\x80\xaf\x8b\xc4\x88\x4e\xf2\xd7\x8d\x11\x0d\x25\x72\xa3\xaf\x06\x2d\x98\x50\x35\xce\x08\x6b\xb2\xb1\xfd\x54\xaf\x00\xf3\xce\x02\x06\x5d\xf4\xd8\xce\x82\x06\xdd\x92\xbc\x72\xe0\xe0\x8a\x79\xec\x6b\xf0\x60\x85\xd9\xf9\x58\xbe\xe2\x9a\x71\x81\x43\x46\xf1\x19\xe8\x25\x1e\xb1\x41\x29\x9f\xc2\x55\xca\x9e\x2c\x8a\x80\x6c\xa3\x43\x84\xcc\x0c\xea\x6c\x58\xc3\x0a\xb8\xd1\xf4\x5f\xa0\xe9\x18\xf0\x5a\x13\x52\xe8\xc2\x06\x21\x9a\x9c\xc4\x87\x38\x5a\x44\x09\x8d\x02\x9d\x79\x26\x70\x36\x6f\xe2\x78\x6e\xe7\x7b\xd4\x9d\xb7\x42\xdd\x69\x2b\x48\xb5\x4e\xdc\xb6\xa3\x2b\x86\x53\xd2\xa3\x01\x35\xa1\x01\x1d\x78\xbc\x0b\x56\x94\xd6\x7a\x43\x18\x85\xfa\xb9\xeb\x21\x81\xde\x00\x12\x68\x93\xc3\x57\xe0\xfd\x94\x8e\x5d\x0f\x53\xf4\xa1\x13\x4c\x91\xbf\x04\xf7\x0a\x79\xa6\xfd\x3c\xbe\x31\xa2\x49\x7d\x60\x6f\x09\x4b\xd4\x20\x2e\xac\x23\x37\x2d\xc3\x25\x5a\x46\x17\x9d\xd6\xe5\x6d\x51\x82\xd6\x5b\x99\xb5\x00\x80\x1a\xef\xae\x3d\x81\x03\x6a\xdf\x86\x3d\x39\x37\xbb\xcc\x6a\x59\xaf\x76\x68\x98\xd9\xb2\x8e\x82\xb5\x5e\x92\x8b\xa7\x87\xf7\x95\xe8\x52\x14\x59\xdb\x2c\xd9\x65\xe0\x7c\xd0\x44\xa0\x39\x4f\x62\x07\x42\xe1\x57\xcb\x77\xe0\x33\x01\xfc\x02\xb9\xcd\x80\x62\xe7\xa0\x6d\x15\x05\xc1\x96\xa5\xb4\xf8\x4d\x84\xe1\xee\x80\xd1\xec\xc2\x8a\xe0\x39\xc9\x26\xf6\x83\x95\xb2\x88\x2c\x9b\xbf\x97\x8c\xb1\xb4\x42\x60\x35\x6f\x1e\xe6\x4a\xbb\xef\x8a\xc1\x2d\x13\x3d\x02\xe3\xa0\x68\x2a\xf5\x69\xe8\x0c\x9e\x3e\x51\x67\x88\xc0\x61\x8f\x4b\xbd\x74\x6e\x76\x9d\x3c\x75\x55\x62\xd9\x20\x18\xac\x56\xb9\x6d\x7b\x70\xa0\x14\x7f\x1e\x67\x58\xe0\x24\x21\x09\x95\xe9\x8b\x05\x03\x9f\x96\xdd\xb5\xfa\xac\x0a\x6e\x4c\x44\x2c\x4f\x27\x86\x14\xdd\x40\x6c\xb1\x3f\xc5\x91\xc8\x59\x08\x6d\xe6\x37\xc6\x17\x13\xcc\xe1\x5e\x00\xab\x52\x34\x87\xaa\xad\x53\x4c\x05\x23\xb2\xb5\x46\x26\x89\x72\x41\xd5\x62\x6c\x4b\x8e\x76\x3f\x70\x77\xf6\xcb\x53\xfb\xe1\x72\x0f\xb7\xcb\xea\x77\xfd\xf9\x12\xa7\x19\x11\x50\x26\xc8\x15\xbc\x09\xca\xaa\x5a\xd4\x06\xe2\x6b\x0d\x41\xf8\x73\xed\xda\x6e\x0b\x1c\xc6\xcf\xe3\x20\xa3\x6a\x1c\x55\x89\x63\xd5\x61\x6d\xc2\x9d\x5a\x36\xc9\x17\x46\x5e\x6a\xf1\x22\xbf\x40\x95\x11\x9b\x36\x61\x9a\xd6\x03\x0e\x5c\xc1\x60\xaf\x2c\x36\x26\x48\x79\xb7\x4a\x55\xcb\x38\x2d\xd6\x4f\x53\xf0\xd1\x92\xc1\x0e\x82\xaf\x3a\x8c\x38\xe8\x64\x47\xc3\xd6\x07\x5d\x88\x3c\x53\x74\x52\x87\xb6\x51\xbb\x2b\x21\x3a\x48\x20\xcd\xda\xb9\x19\x4a\xdd\x9a\xba\xa2\x25\x4e\x6c\x67\xa7\xe5\x7f\x8b\x23\xe6\x10\x82\x0c\xc2\x52\x98\xc7\x77\x9d\x52\xa5\x5c\xa2\x80\x31\x40\x6b\xea\x2c\xdb\x66\xbf\x72\xe1\x1e\x18\x2a\xbd\x1a\x13\xd1\xd1\x88\x0d\x24\x7a\x26\x88\x11\x0b\x21\xd1\x50\xc3\xd5\x5b\xb5\xa1\xf6\xd3\x84\xe8\x9e\x7c\x6c\x8a\x16\x1e\xa8\x92\xbe\xfc\x98\xe9\x63\x8a\x13\x49\x0e\x74\xc3\x50\xb5\x54\x71\x08\xfe\xc4\xe8\x59\xe0\x2c\x23\x62\xc4\x6c\x16\x07\x38\x5c\x38\x4f\x4c\xfb\x6d\x21\xae\x76\x0d\xc8\x38\xc2\xd1\xfc\x95\xf6\x08\x43\x32\x4e\x34\x27\xb1\xcb\x17\x2e\x6f\x8f\x9b\xb7\x31\x58\xaf\xb1\x59\xe7\x53\x57\x3e\xeb\xc0\x76\x92\x44\x9a\xa3\xf8\x32\xd3\x19\x11\x7a\xd4\x9a\x86\x9f\x08\x43\x74\xea\xc6\x61\x63\x77\xd0\x33\x78\xa6\x34\xe9\x3f\x61\x9a\x98\x04\x7c\xd7\xb5\x13\x02\x8d\xf9\x7d\xc4\x8c\xbb\x9b\x45\xa5\x0c\x55\xca\xa8\x9c\x6b\x4e\x9d\x83\x4f\x12\xd4\x8c\xb6\xc4\x19\xf6\xb4\xce\x69\x1e\xea\xd7\x97\x73\xd0\x27\x2a\x38\x4b\x21\x49\xc6\xe2\x32\xb9\xe5\x93\x44\xf9\xe3\xd1\x98\xe2\xb8\x52\x22\x8e\x63\x59\x36\x7e\x1a\xb5\x92\xfe\x5a\x32\xbb\x1c\x96\xb2\x02\xa3\x00\x56\x08\x82\x38\x5d\x65\xb1\x65\xf2\x6f\x9f\xda\x50\x4f\x6d\x68\x5e\x9b\x7d\x4c\x6f\xf0\x87\x78\xdd\x14\x87\xb6\xed\xdf\x85\x64\xbb\xc3\x54\x87\x37\xce\x09\x78\x99\x74\x80\xb7\xcd\xdf\x78\x89\xd4\x8d\x3e\xc1\xe1\x0d\x13\x1c\x3a\x5b\x6a\xcb\xb1\xd9\xed\xc7\x76\xad\xe4\x80\x15\x60\x4e\x4d\xbd\x5c\x12\x25\x68\x24\x77\xc1\x1f\x64\x86\x3b\x46\xb5\x81\x16\x98\xad\x90\x9a\xf4\x0b\xde\x09\x09\x71\x62\xbe\xce\xdf\x44\x10\xfc\x18\xf3\xe7\x9a\xad\x4e\x86\x68\x1a\x97\x5c\x8b\x3d\x82\x44\x54\x92\x52\x24\x0b\x95\x88\x11\x69\x8d\x9d\x78\xc4\xe6\x94\x08\x2c\xa2\x39\x64\x37\x16\x1b\x63\xb2\x64\x0d\xa0\x91\x89\x65\x08\xbd\x4d\x6b\x6c\x7a\x87\x75\xaf\x5a\x98\x3c\x3e\x9d\xdd\x73\x3d\x92\xd4\x7c\xe2\x85\x19\x2b\x65\x84\x26\xb9\x4e\xdb\xbf\x6d\x20\xbe\x5f\xec\x17\x0d\xc6\xf7\xc1\x44\xc1\x17\x1d\x03\xf2\x0b\x6a\xe8\x83\xf2\x5f\x28\x28\xbf\x61\x89\xd7\x0b\xcc\xdf\xc8\xe4\xf7\xfa\x31\xc3\xae\xe7\xd7\x88\x1b\x5e\x15\xb4\x95\x4f\xc6\x2f\x7e\xf4\x1a\xe7\xdc\xf5\x04\xfe\xe4\x89\xc2\x48\xc4\x42\xd3\xd9\x84\xc4\x31\x70\x5a\xc5\x6d\xa5\xe8\x82\x76\x9c\x79\x40\xdf\xbd\x58\x6a\x62\xc7\x09\x67\x33\x49\x63\x03\xb6\x92\x61\xa8\xd8\x1a\x1a\x2f\x00\x5c\x00\xf6\x37\x49\x88\x70\x5e\x09\x81\xbe\x96\x94\x59\x34\x45\xff\x5b\xcc\x89\x64\x5f\x29\x63\x2c\xc0\x6c\x81\x1e\x19\x7f\x4e\x48\x3c\x83\x1d\xaa\x0e\xe6\x10\x51\x72\x80\xa8\xf2\x9f\x09\x40\x23\xe0\xb9\x1a\xe9\xb1\x43\xac\x99\xd1\x00\x88\xfd\x36\xa8\x89\xee\x9b\xf9\xe6\x08\xa1\x73\x86\xa6\x38\x52\x07\x48\xe6\x93\xa2\xfd\x98\x9b\x22\xd7\x5a\xfb\x0e\x26\x5e\x34\xd2\xc7\x8c\x37\x74\xde\x7c\x36\x1c\x77\xd0\xe4\x3a\x48\x28\xde\x2a\xb6\xee\x09\x6f\x03\x31\x7a\x99\x4b\x1b\x84\x81\x38\xf3\x47\xdf\xc2\x2b\x79\x8c\x68\xc0\xfb\x34\x78\xcb\x8c\xc7\xad\xb6\xce\xca\x54\xd6\x1d\x4b\x11\x08\x69\x05\x25\xeb\xa8\x82\x76\xcd\x72\x6b\xa9\x49\x2a\x41\x70\x6a\x9d\x03\xfa\xaa\x01\xb1\xc6\x84\x41\xea\xd1\x53\x61\x24\xcc\x75\xb6\xf8\x82\xb2\x47\xbd\xbb\x05\x2a\x36\xd4\x97\x87\x9e\x9b\x36\x2d\xd3\x37\x1e\x39\xe5\xcc\x38\x08\xb7\x92\x3b\xe9\x8c\xe1\x64\x4d\x1b\x47\x6d\xe5\xea\x3e\x3d\x27\x67\x59\x71\x41\x4b\x11\xc6\xd8\x87\x4c\x8f\x6b\xd9\x90\x2a\xf3\x0d\xe5\x3d\x8c\x62\x92\x11\x16\x13\x16\x2d\x80\x44\x18\x20\xe7\x08\x86\x13\x84\xe1\x3b\x9c\x1c\xa1\x33\x93\x5f\xe3\x25\x3c\x7b\xad\xc3\x85\x9e\x62\x46\xa7\x5a\x4f\x00\x23\xac\x1d\xe5\x88\x99\x61\x3a\x1f\x48\x50\xb4\xdf\xaf\x58\xd3\xce\xe8\x1b\xe4\x6a\x4b\x54\x62\x56\xfe\x1e\x2d\xbf\x70\xa0\xb7\x65\xbb\xa3\x9b\x73\x35\x08\x64\x3e\x39\x84\x7f\x97\x12\xce\x1c\x50\x4f\x81\x22\x43\x12\x02\xe6\x40\xeb\xf1\x82\x8b\xb1\x0d\x58\x6e\x17\x7e\xbb\x15\x79\x1c\x41\x1f\x25\xa5\x26\xa5\x8c\xa6\x79\x1a\x38\xef\x4c\xc5\x82\xc8\xda\x2f\x4d\x26\x46\xa6\xf5\x80\xc8\x81\x97\x23\x7d\xb9\xb2\x05\x9a\xd1\x27\xc2\x46\x2c\xe3\x94\xa9\x23\x74\xc5\x15\x09\x4a\x44\x18\xe8\x28\x9e\x29\x9a\x1a\xb4\x53\x41\xf4\x39\x30\xa0\xd8\x00\x34\x39\xc7\xea\x00\xc5\x39\x1c\x55\x46\x94\x66\x1d\xfa\xc6\x55\xb0\x33\x10\x1f\x2d\x46\xcc\xdc\x74\x53\x4c\x93\x5c\x10\x2b\xb3\x62\x93\x17\x53\x0c\xb9\x18\x99\x45\x42\x0b\x26\x91\xd2\xd9\x5c\xe9\x2d\xd2\x32\x9e\xf5\x37\xce\x35\x37\xe2\x23\x36\x21\x08\xa3\x8c\x4b\xaa\xe8\x93\xf7\x5f\xd2\x29\xc2\x52\x82\x05\xe5\x08\x9d\x95\xec\xff\x54\x82\xea\xdd\x16\x57\x4b\xd9\xd8\xda\x9e\xdb\xf3\x71\xb6\xde\xc8\x52\x2f\x76\x95\xf1\x44\xf2\x24\x57\xa1\x0b\xb6\x79\x6f\x0b\xd3\xb8\x03\xee\x07\x03\x31\x9f\x8e\x98\xa3\x6b\x79\x84\x06\x12\x49\xae\x77\x49\x9a\xad\x8c\x04\x55\x44\x50\x83\xe2\x44\x94\xd9\x04\x7f\x4e\xfd\x19\x48\xb1\x78\xd4\x22\x54\x68\x81\x37\x98\xa2\x25\x6b\xc7\xc4\x48\x48\x00\x6b\x15\x6e\x07\x98\xfe\x11\xe3\xec\x90\x91\x19\x5e\xb5\x23\x23\x56\xda\x12\xf4\x35\x9d\x16\x0a\x69\x9b\xcf\x31\x58\xbb\x31\x44\x3e\xb5\xed\x92\xe9\xb8\x6d\x93\xa6\x09\xc7\x2b\xdc\xc6\xd3\xe2\xd0\xa3\xbf\xf3\x89\x19\xa3\xd6\xfb\xb9\x02\x29\x50\xab\x57\x53\x2e\xc8\x1c\xb3\xf8\xc0\x6d\x56\x79\x6c\x70\x33\x5a\x53\x9b\x53\xc6\x40\x12\x74\x20\xc2\xc4\x60\x31\x61\x16\xec\x85\x55\xdc\xec\x56\x14\xfb\xb0\xd6\x5d\xe1\x5b\x83\xda\x27\xc6\x00\x61\x58\xde\x22\xb3\x47\x5c\xd2\x34\x4b\x8a\x9c\xa6\xc0\x36\x3a\xd5\x22\x96\xe3\x91\xfc\x09\x4c\x57\x4e\x6b\x83\x5b\xdd\xee\x9c\xa6\xb3\x86\x91\x7b\x46\x0a\xb7\x86\xb3\x79\x99\x32\x98\x01\x0b\xfb\x5a\x12\xfd\x4f\x45\x0a\xb5\xcf\x08\xeb\x23\xe6\x44\x90\x6f\x80\xcb\xd8\x66\x03\xe3\x99\x16\xa1\x0d\xcc\xab\x5d\x3f\x14\x19\x27\x77\xe9\x9c\xd8\xc3\xe0\x5e\x6d\xb8\xa8\xbe\xa3\x0c\x97\x32\x6f\x37\x10\xfc\x92\x7c\xad\xe4\xaa\xc0\xed\xb7\x68\xab\x69\xa2\xf0\xaa\x32\x23\x6b\x50\x82\xd9\x67\x82\x74\x77\x96\x9a\x5d\xc5\x1b\x0c\x11\x01\x73\x92\x64\x28\xa6\x53\x30\x4b\x29\x60\xdf\x1e\x58\xcc\x60\xc1\xeb\xc3\x9e\xe6\xcc\x80\xc4\x19\x8f\xc8\xb3\xc5\xdb\xb6\x57\x63\xd1\xf8\xd1\x88\x9d\xab\xaf\xa4\x16\xd1\x39\x9b\xe9\x8b\x26\x7e\xa2\xb2\x28\x72\x12\x71\x26\xf3\x94\x08\xdb\x85\xbe\x91\x35\x45\xda\x02\x01\xd8\xc9\x50\x7a\x6c\x7a\xef\x9f\x70\x42\x63\x57\x88\x47\xff\x68\xce\x9c\x1e\xa5\x74\x1e\xc5\x86\x90\x30\xbb\xb9\xb1\x5e\xab\x37\x13\xeb\x7f\x0c\x25\x77\x94\x16\x42\x3e\xb6\xb6\xfa\xe3\xaa\x88\x6f\x57\x7d\x89\x78\x3f\xa9\x4d\x0a\x2d\x17\x8c\xec\x2a\x9c\xad\x42\x31\x74\x88\xba\xb9\x09\x01\xd6\xfd\x38\xa3\x8f\x19\xdc\x5a\xec\xa7\x32\x41\x3b\x6a\xc3\x59\x42\xf1\x9e\x50\x90\x0d\xa4\xc2\x5a\xbc\x30\xd7\x01\x17\x56\xc3\xb1\x77\x4e\xfb\xd6\x9e\x6d\x59\x26\x42\x46\x38\xa9\xef\xf0\x12\x7b\xb3\x79\x7f\xb9\x12\x60\x8f\x9b\x69\x7b\x69\xd2\x6f\xc4\x93\x64\x9d\x12\x26\x95\x99\x9f\x16\x9f\x2f\x1f\x51\xd1\x8f\xde\x00\xb7\x17\x70\x6a\xcc\xe5\x8d\x13\x6b\x4a\x91\xca\xee\x52\xf8\x92\x51\xc3\x16\x96\xb5\x8e\x18\x9f\x42\x91\x9b\xa4\x2d\xaa\x2b\x13\x3c\xa5\xeb\xa0\x2c\x9b\x40\xa7\x5b\x67\x17\x5f\x61\x65\x70\xd6\x73\x10\x4d\x0d\x79\xd9\x1e\x21\x5f\x0f\x5b\x71\x73\xc9\x19\x4a\x71\xb6\xd1\x82\xaf\xf2\x0a\x0d\x50\x6a\x5c\x72\x76\xf5\x00\x6f\x91\x40\xbd\x18\x58\xe4\x67\xbc\x28\x52\xa3\xdb\xf0\x73\xd9\x5a\xe4\xf0\xa0\x5f\x3f\x67\x53\xbe\xc6\xe1\x2c\x52\x99\xed\xe9\xc3\x8e\x66\x83\xf3\xe7\xbd\x14\x66\xf7\xcd\x9a\x76\x39\x8f\xa7\x4d\x44\xbd\xf6\xc9\x74\x2b\xf8\x92\x36\xca\x90\x89\x84\xe6\xc9\x75\xee\xd6\xf2\xd1\x0a\x5a\x44\x30\x9c\xe5\x4b\x75\x59\xa2\xc3\x9d\xaf\x51\xa5\x1d\x64\x4c\xe1\x2e\x98\xfa\xa6\xb9\xd5\x57\x58\x33\x7b\x48\x3a\x2d\xd6\x96\xd8\x0d\xeb\xe1\x00\xbb\x1e\x3d\xea\x6f\xf3\x09\x5d\x59\xe4\xa0\xfb\x62\x00\x37\x93\xd6\xce\x55\x44\x66\xda\x14\xb5\x29\x4d\xb4\x88\x7d\xde\x60\xe0\x74\x09\x62\x3e\xa0\xca\x84\xca\x3b\xe9\x29\x17\x34\x28\x0c\xea\x64\x24\x44\xa1\x40\x49\xe8\xe4\x09\x14\x7a\x30\x2d\xce\xf9\xb3\x89\x4e\x17\x54\xf3\x2c\x23\xac\x2a\x30\xf7\x68\x5e\x40\xad\xb5\xc4\x18\x9b\xfc\x07\xdc\xc4\x0c\x62\x5b\xfb\xb8\x18\x55\xcb\x96\xee\xa2\xc4\x53\xf7\xfc\x3b\xd7\xeb\xbd\xfe\xa2\xbe\x37\x8d\x23\xbc\x2f\xb7\xbe\xf6\xe8\xbc\x94\xbf\x7e\xc0\xd4\x47\xf8\xd4\x29\x3d\x18\x4d\x05\x01\x07\x7f\xea\x31\x35\x0c\xa8\x2e\xe7\x70\xdf\xdd\x9d\xfd\x70\xfc\x70\x8e\x88\x8a\x50\x42\x1f\xc9\x88\x45\xf2\xe9\x40\x8b\xc7\xff\xc8\x89\xd2\x3f\xb7\x78\x04\x68\x4a\x98\x04\x4e\x40\x55\x0d\x7b\xa8\x79\x21\xdd\xc2\xe8\xff\x9e\x95\xbf\x5f\x42\xf2\xb5\xf4\x61\xa0\x5d\x57\xef\x06\xc8\x14\x4a\x7a\x98\xa5\x95\x0d\x14\x63\x6c\x91\xc3\xa6\x6a\x98\x1b\xa4\x0b\xb1\xbf\xe7\x6c\x4d\xa1\xeb\xb4\xf8\x28\x18\x45\x8b\x4c\x97\x66\x18\xb0\xae\xd7\xcb\x43\x32\xdf\x34\xb6\xbe\x8a\x89\x14\x69\xd9\xce\xb6\x5c\x14\x0e\x45\x4a\x10\x02\x2c\xc4\xd3\x93\xbd\xeb\x2d\x12\x87\x9f\x58\xf0\xd1\xd1\x88\x5d\x3a\x8f\x73\xf1\xab\x2c\xf4\xf0\x74\x12\x40\x80\x97\x5b\x81\x66\x63\x2a\xfd\x0f\x50\xd0\x45\xe6\x89\x32\x15\xed\xa6\x94\xe1\xc4\x0f\xd4\x3c\x69\xe2\x12\x02\xb3\x68\xbe\xad\x09\x99\x4e\xc7\x24\x59\x47\x12\x3d\x9f\x0e\x13\xa9\xe9\x3b\x7a\x6c\x39\x9d\x9b\xd4\x6c\x2c\x26\x63\x2b\xd1\x9a\xba\x4f\xa8\x30\x41\xe3\xc4\x54\x94\x23\x08\x7c\x94\xd5\xec\x31\x03\x10\xa1\x77\xd1\x4a\xea\xc6\x45\x69\xd2\x36\x7c\x48\x36\xf4\x82\xb0\x1a\x31\x91\x33\x28\x36\xe1\x23\x16\x30\x2a\xf0\xc2\x23\xe7\x3f\xb0\xde\x9c\x99\x66\x13\x06\x8e\xdb\xbc\xac\xf5\x33\x9e\x4b\xb0\xd5\xa4\x44\xe9\x0b\xea\x6b\xa8\x03\x6b\x42\x86\x0e\x50\x26\x68\x0a\xe6\x56\xf9\x4d\xc3\xd6\x9d\x62\x85\x13\x3e\x1b\x08\x45\xa7\x38\x52\xf7\x78\x2b\x0d\x1c\xdb\x66\x36\x0d\x3f\x75\xc3\x40\xe7\x67\x7a\xf1\x67\x84\x11\x01\x13\xd5\x3a\x79\xf3\x11\x86\x27\x1b\x71\x6e\xb0\xb2\x59\xc3\xa8\xf4\x16\x0b\x9c\x2b\x9e\x6a\xfd\x16\x27\xc9\xe2\xc0\x58\x64\x09\x9a\x63\x39\x77\x1b\x6d\x8c\x69\x5d\xee\x26\xbb\xb8\xa7\x38\x9a\x93\x3b\xa8\x8a\xdc\xb4\xb8\x95\x51\x7e\x20\x2c\x4f\x3f\x9c\xa0\xff\x29\xe6\x78\x3a\x38\xfd\x7e\x38\x3e\x3b\xbf\x1b\x7c\x77\x31\x3c\x0b\xe6\x63\x9f\x5c\x9e\xdf\xdd\xd5\x7f\xfd\xfe\xfc\xbe\xfe\xe3\xcd\xf5\xcd\xc3\xc5\xe0\xbe\xa9\x95\x8b\xeb\xeb\x1f\x1e\x6e\xc6\x1f\x07\xe7\x17\x0f\xb7\xc3\x86\x4f\x1f\xee\xdb\x1f\xde\xfd\x70\x7e\x73\x33\x3c\x73\xab\xf2\xb7\xe0\x74\x81\xf5\x18\x52\x2f\x9a\xa7\x51\x3d\x80\x87\xa8\xfc\xe2\x09\x7a\xa8\x96\x3e\xb0\xb1\xc8\x06\xc7\xe2\x19\x4b\xcd\xc3\x20\x14\x7e\xc4\x90\xfb\x5c\x2f\x4a\xdb\xa7\x26\x5c\x27\x9a\x13\x94\x70\xfe\x98\x67\x96\xb5\x99\xf8\x30\xc6\x8d\xe1\x87\xc8\xa0\xb5\xef\xcf\xef\x4f\xea\x25\x18\x7c\x63\x01\x62\x96\x3b\x03\x30\x2e\xec\xd8\x29\xd8\x52\x1c\x34\x7f\x61\xbd\x0d\x7a\xf0\x3b\xb3\xac\x1f\xd3\x1a\x66\xaa\xd2\x4d\x1c\xdb\xa2\xbf\x6e\x62\x41\xc3\xe5\x7d\x5d\xb6\x9a\x7e\x39\x4c\xed\x29\x34\x21\x11\xce\x4d\x50\x93\xbe\xa7\x84\xe0\x22\x1c\x70\x41\x0f\xbb\x6b\xd4\xd2\x51\x63\x83\x95\x3d\xd3\x13\x97\x8f\x34\xcb\x48\xfc\xa1\x2e\xbf\x94\xab\xc3\xda\x9a\xe4\x7c\x8a\x82\x33\xa9\xf5\x7a\xd0\xf9\x5d\xe1\x94\xf9\xc2\x7b\xd2\x20\x70\xa3\x08\x65\x01\x20\x67\x7d\x27\xf8\xc2\x16\x14\x5c\x63\x58\xa1\x67\x02\x29\xd5\xb9\xad\x1c\x65\x74\x6f\x7d\xb6\xa1\x3b\x63\xd3\x76\x75\xe0\x4a\xa9\xd6\xad\xcc\x78\x17\x02\xb7\xfe\x5e\x92\x26\x46\xbc\x45\x5e\xec\x99\x69\x14\xb8\xb3\x8b\x79\x83\x11\xb7\x04\x37\xb8\xdb\xa0\xc1\x42\xbe\x44\xbe\xaa\xdf\x48\x2b\x2e\x0b\xcd\xb6\xbb\x8c\xc7\x61\x81\x94\x00\xae\xbb\x0f\xac\x04\x82\xbc\x72\xad\xee\x79\x8c\x17\x9a\x38\x20\xd6\x58\xe6\x59\xc6\x85\x42\x2d\x6d\x18\x37\xbe\x19\x1f\xdc\x39\x76\x1e\x9e\xc7\x41\x23\x5a\xc2\x90\x0d\xb5\x34\xba\xc1\x23\xd8\x75\x2d\x18\x47\x18\x20\x0b\x8a\xa0\xaf\x7b\x94\x96\x54\xea\x12\x85\x36\x09\xbf\xdb\x64\x18\x64\xfa\x82\xef\x5a\x86\xaf\xa9\xf7\x6b\xd7\x42\xe3\x96\x27\x64\xaa\xc6\x8d\x5e\x9f\x25\x06\x4e\xdd\x22\x6b\x43\x94\xa1\xb3\xf9\x0e\x5a\xec\xae\x25\xfc\xd1\x06\xf6\x68\xd5\x20\xb0\x10\x08\xce\x95\x91\x4f\x0b\x1d\x06\xb9\xd5\x04\xf3\x82\xed\xd4\xe6\x82\x79\x21\x50\xcb\xfc\xc6\x1f\xea\xd3\xa6\x8e\x46\x6c\x08\x01\x14\x85\x22\xe2\x52\xc4\x40\x0b\x58\x29\xff\x97\x8a\x8e\xbe\x6a\xb4\x66\x3b\xc2\x6b\x41\xf7\xb6\x5e\x7e\xb2\x40\x45\x61\xd9\xd2\x77\x5d\x4e\x8f\xb1\x7a\x3b\x11\xd0\x4c\xd8\x96\x71\x57\x24\xb3\x96\x79\x33\xcf\x22\xd2\x07\xe2\xc3\x74\x57\x47\xe8\x27\x67\xf9\x81\xc0\xd7\xa2\x26\xb3\x8d\xdd\x48\xf0\xc2\x81\x42\x36\x2d\xec\x2e\x70\x16\x77\x1d\x0a\xbb\x7c\x81\x3d\xa0\x53\xc3\x2a\x97\x14\x70\xc6\x8c\x45\x76\x8d\xb4\x8f\x53\xff\xd1\x1d\x59\x1e\x15\xf0\x11\xca\x70\xda\xc8\x2a\x10\x3a\x58\xb2\xf8\x5f\x66\xb3\x4c\x26\xaa\x2b\xac\x65\xcb\x22\x5a\x0f\xaa\x3e\x3f\xe0\x01\x34\x89\xaa\x68\x4a\x93\x04\xe4\x80\x23\x34\x80\xf2\xc0\x90\xc8\xa9\xaf\x42\x17\x60\x41\x67\x8c\xaf\xca\x31\x6b\x21\xa6\x28\x20\xa6\xbb\x76\x62\x92\x40\x4d\x45\x1e\xff\x6e\x28\x6a\x07\x98\x2e\x9a\xb7\xe0\x3a\x22\x76\x77\x24\x97\x35\x94\xf7\xb7\x88\x8e\xae\x0d\x37\xf8\xf0\x5f\xcd\x43\xff\x94\x63\x81\x99\x82\x98\x5f\x2b\xba\x0b\x12\xa4\x1e\x91\xcf\x10\x9f\xc1\x8c\x21\x18\x7e\x0a\x37\xd7\xb9\xfc\x21\xdc\x0b\xd1\xf8\x00\xd1\x23\x72\x04\xd5\xd9\x84\x96\x25\x26\xc5\x9b\x73\x2d\x39\x8c\x58\x2d\x96\xf1\x08\x0d\x12\xc9\xed\x17\x84\x45\x09\x94\xe3\x0e\xc2\x93\x3d\xe5\x5b\xb7\xd2\x64\x01\x0a\x0a\x6c\x65\xd1\x3c\xb7\x0f\x82\x0f\xa1\xc8\x18\xf8\xc4\x13\x38\xe9\xc5\xef\xbf\xe7\x99\xf1\x56\xb4\xc5\x49\xbc\x60\x39\x87\xda\x35\xf4\x62\x9b\x64\x4a\x05\x2e\xdb\x20\x78\x03\x36\xa6\x88\x31\x0d\x10\x58\xd0\xd7\x58\xa1\x84\x60\xa9\xd0\x1f\xbe\x59\x2b\x36\xc4\x4d\xb0\xe0\xae\xf6\xf8\x16\x89\x62\x2e\xd5\x20\x14\xee\x7c\xc7\x50\x3b\x0e\x0b\x85\x30\x62\xe4\x39\x8c\x2c\xe5\x10\x0c\xec\x0a\xc2\x91\x20\xb7\xd5\xc4\x93\x99\xcc\x7c\xc8\xd6\x30\x2a\x53\x0b\x1f\x71\x70\xc7\xd6\x7d\x6a\x87\xd5\x40\x59\x56\x79\xb2\x21\x9e\x00\xc9\x55\x04\xfd\xcf\xb1\x1a\x31\xcb\x59\x5d\xd8\x48\x90\xe6\x35\x48\x92\x72\xa0\x3d\x86\x5c\x12\xa6\x27\x0c\xf5\xd9\x8f\xfc\x02\x5d\x81\xfa\xe5\xa3\x9d\x4b\x76\xba\xe2\xb0\x98\x78\x3c\x8f\x77\x14\xb6\xdd\x28\xed\x34\xd9\x97\x5f\x51\x08\x6e\xe8\xfe\xc2\x14\xca\xef\x20\x0c\x93\xa6\x21\xaf\x38\x58\x75\x9b\xfe\x12\xd9\x78\xd7\x1d\x74\x17\x95\x9b\xed\xe3\x70\xcd\x3e\xf3\x06\x73\x7b\xcb\xe6\x06\xb2\xc5\x36\x0a\xb8\x0f\xbb\x7f\x2d\x8f\x6f\x69\xe8\xe7\x31\x24\xfd\xad\xe6\x82\x45\x12\x9d\x63\x1d\x26\xf6\x3a\x0e\x72\x7a\x82\x14\x02\x08\xfe\x73\x8c\xcf\xbe\xd9\xe2\x79\xcd\xde\xf7\xf4\x0f\x8a\xf9\xbb\xa9\xf8\x20\xb8\xfa\xc4\xdb\x85\xbd\x41\xfc\x77\x1c\x41\xa4\x3f\xf4\xe4\x72\x0c\xea\x80\x4c\x0e\xc6\x1a\x83\x31\xbf\x51\x3c\xcc\x04\x8f\x88\x94\x47\x68\x08\x17\x8d\xfd\x27\xc2\x53\xe7\x90\x08\x5e\x1e\x31\xad\x99\x38\xfc\x96\xa0\xfd\x32\x89\x37\x9d\x00\x03\x06\xb7\x95\x2f\x27\x5d\x5d\xa3\xa4\x4d\x9b\x70\x58\x74\xd0\x06\x94\x35\x40\xc3\xd9\x09\x8a\x79\xf4\x48\xc4\xb1\x20\x31\x95\x27\xe0\x5b\x57\xad\x4e\xbd\x54\x6b\xdb\x5b\x4b\x1a\x6d\x81\x02\x2b\x92\xe2\x4e\x4d\xff\x36\xc0\xda\x85\xd7\x1e\x20\x3a\x05\x75\xc2\xe5\x64\x98\x20\x64\x07\x77\x43\x98\x12\x0b\x88\xeb\xf7\xa6\xac\xca\x42\x38\x4d\x43\x0b\x6d\x6d\xd9\x44\x62\x17\x31\x38\x1b\x4e\xfb\x7e\x4e\x24\x71\x01\x07\x66\x52\x8a\xdb\x58\x66\xc3\x2e\x32\xac\xe6\x12\x52\x57\xcb\x6b\x60\x95\x2e\xf8\x54\xaf\x10\xce\x20\x5e\xc1\x58\x29\x8a\x8f\x7c\x82\xa5\x54\x34\x49\x46\x8c\x11\x12\x4b\x04\x59\xa6\x5f\x35\x66\xc8\xeb\x4f\x0f\x10\x8e\x63\xf4\xbf\xbf\xfe\x78\xf1\xf3\xfd\x70\x7c\x7e\x05\x46\xeb\xf3\x8b\xe1\x37\x07\xfe\xc7\xeb\x87\x7b\xff\xab\xb1\xb0\x3c\x11\x81\x52\xfc\x08\x2a\x1e\x93\xc4\x26\x4f\x90\x11\x0b\x47\xea\xb0\x03\xf4\x13\x49\x5c\xa4\xab\x15\x53\x3c\x84\xa2\xdd\xc3\xd6\x8a\xc5\xc6\xe6\xb7\x86\xf2\x7b\xeb\x3f\x59\x4e\x83\x8e\x78\x7c\x17\x4e\x0c\x84\x1c\x19\x2c\x83\x64\x72\xab\xfb\x16\x04\x47\xd8\x8c\xb2\xb6\x78\x3c\xc2\x9e\x5e\x52\x88\xff\x81\x2c\x7e\xd4\xea\xf5\x0d\xa6\xa2\x33\xed\x35\xa3\x01\xb9\x13\xa3\xf5\x74\x2c\xab\x87\x4a\x1a\x59\xd8\x64\xdb\xb4\xc6\x7c\x36\x01\xc1\xbd\xf9\x74\x2d\xbc\x14\xf9\xac\x84\x43\xa9\xf0\xf9\x1c\x0e\xca\xc9\x5f\x34\x05\x0d\x8e\xd8\xfd\xf5\xd9\xf5\x09\x22\x09\x9e\x70\x08\xe5\xb7\x21\x41\xae\x09\xbb\x60\x11\x4f\x83\x86\x4a\x08\x25\x07\x28\x2b\x10\x4a\x42\x23\xda\x91\x69\x63\x05\x52\x49\xc6\x45\x1d\xdf\x63\xb7\x2a\xa0\x9d\xec\x0d\x17\x5d\xae\x7f\xfd\x1a\x2c\x1d\xcf\xb4\x22\x57\xe1\xbc\xf6\x6e\x9e\x12\x6c\x6a\xe9\x1b\xb7\x90\xb5\xe5\xdb\x00\xd6\x24\x29\xd5\x53\xd4\x07\x47\x1e\x59\x17\x7c\xf1\x26\x67\xe8\x87\xbf\x48\x34\xc9\xd5\x88\x95\xdb\xe0\x0c\x0d\x7e\xba\x43\xdf\x61\x15\xcd\xbf\x19\xb1\x6b\xad\x66\xfe\xf0\x97\x16\x28\xa5\xb5\xd1\x09\xf5\x9a\x9c\x61\x85\x2f\x38\x8e\x29\x9b\x35\x41\x13\x16\xf5\x63\x86\xf7\x83\x13\x74\x6d\x75\xf8\x22\x13\xc4\xa7\x04\x07\x0d\x01\x43\x86\x89\x38\x2e\x02\xac\x9c\x95\xe1\xdb\x8c\x66\x06\x17\xd6\x88\xdd\x1b\x4c\x46\xcd\x55\xa9\x42\x19\xb7\x35\x8c\xb4\x56\x66\xd0\x2a\xb1\xcb\x90\x22\xc9\x02\xe9\xd5\x01\x32\xf6\x9b\x61\xe5\x31\x90\x67\xea\xcc\x7e\xc4\x40\x41\xf7\xb9\x29\x09\x8f\x70\x02\x31\x79\x87\x81\x4d\x4f\xab\xed\x3c\x87\xfc\x70\x53\xf4\x7c\x51\x0e\x9d\xf5\x90\x05\x5e\x28\x0b\x37\x0a\x0c\x00\xb0\x8f\xd6\x1b\x9b\x72\xcd\x71\x0c\x16\x1b\x18\xdf\x12\xb3\x3a\xfa\x43\x8f\xcd\x66\x96\x45\x3f\xf5\x69\x5b\x3c\x67\x0e\x8b\x24\x02\xf3\x3d\x5b\x40\xf8\x36\x14\x1d\xe1\x10\xfa\x51\x70\x67\x4b\x94\xb5\x5d\xf4\x77\x62\xf0\xd9\x88\x99\x48\xc1\xd2\xbe\x84\xe8\x3d\x41\xef\x9c\x41\x20\x63\x3d\x57\x2c\xcf\x6c\x60\xa3\x95\xf5\x33\x41\x0e\x7d\x06\x54\x5c\x5a\x53\x7d\xc3\x1e\xa1\xdb\x50\xbd\x8e\x79\x94\xa7\x0e\x59\x19\xb2\xa7\x8a\xb2\xf2\x25\x89\xc7\x5c\xec\xab\x28\x1e\x50\x5a\x14\x81\xf4\xf1\xce\xfa\xb1\x21\x98\x41\xf8\x69\x5d\x52\x6f\x17\x7c\x81\x77\x6c\x17\xb5\x66\x1a\x1a\x67\xe5\x96\x4a\xad\x6d\x9d\x97\x78\x55\xa0\xbf\x72\x01\xc2\x16\xf9\x9c\x71\x30\x72\x9b\xf4\x2c\x1e\x7f\x25\xd1\xf9\x8d\x96\x80\xb4\xc6\xeb\xcf\x60\x2e\x95\x09\x2e\x83\x74\x1d\xf3\xb5\x49\x17\x38\x40\xdf\xa2\x51\xfe\xed\xb7\x7f\x8a\xd0\x67\xf7\xc7\x9f\xff\xf3\x3f\xff\xf4\xe7\x75\xd2\x49\x9c\x42\x0e\xed\x16\x6b\xe4\xcb\x49\x95\x45\xa2\x70\x07\xea\x9c\x6a\x8b\x5d\xb0\x07\xb0\x6d\xf9\x37\x41\x79\x0c\x62\x87\xf0\xcc\x9e\x70\x19\x9e\x4c\x54\x3a\x9a\x45\x24\x81\x24\xea\xa0\xcc\x21\xbc\xb0\x6b\x25\xfa\xff\xb5\x04\xac\x6c\xac\x8f\xca\x66\x31\x4e\x34\xf1\xe2\xb5\x6e\x04\x7d\x6d\xed\x7f\x0a\x1c\x88\xdf\xb8\x0b\x8e\x27\x31\x11\x66\x4c\xde\x64\xe7\x0d\x89\xc0\x1c\xc8\xe7\x2c\xe1\xb1\x83\x47\x2d\x72\x01\x29\x08\x08\xc3\xcf\x58\x73\xee\x03\x0b\xa3\x65\x3e\x32\x9e\x97\x29\x8e\x0c\x2a\xa8\x44\x5f\x7f\x3e\xd1\xbf\x1d\xa0\xc5\x09\x04\x91\x1e\xa0\x5f\x4f\x2c\x5a\x0e\x16\x6a\xac\x7f\xfa\xc6\xc9\xda\xb6\x09\x18\x34\x95\xe8\xab\xe3\x27\x2c\x4c\xcd\xe8\x63\x33\xa2\xaf\x2c\x67\xf5\x75\xf1\x42\xd9\x3c\xe1\xfc\xd1\x06\xd8\xd6\x3e\x3c\x76\xc0\x6b\x40\xde\xde\x6f\x62\xb6\xde\x27\xe6\x2b\x74\x08\x2f\x10\x74\x94\x4d\xd0\xd1\xdf\x25\x67\xe8\x68\x81\xd3\xc4\xfe\xea\x9e\xda\xf8\x5f\x2c\x6d\x4e\x5c\xec\x83\x7c\x92\x85\xb1\x94\x7e\x97\xf0\x09\xcc\xea\xd2\xcd\xd4\x44\xd0\xc2\x40\x8b\xdb\xa7\xb8\xb0\xec\x44\x5c\x22\x2a\xe0\x07\xa5\x5c\x99\x57\x80\xc7\x35\xcd\xea\xb3\x1f\xd2\x7f\x1b\xbf\x30\x2c\x8a\x4b\xe2\x33\xc6\x61\x1f\xbd\xa6\x1b\xfd\x8c\xbe\xb6\x2c\xe8\x1b\x7d\xc7\xd8\x70\x65\xb3\x0c\x4d\x1d\x2c\x7c\x07\x3f\x07\x1d\x50\x86\x4c\x5a\xe6\x92\x2f\x7f\x3d\x3e\x3a\x3a\xf2\x5f\x43\xd6\xfa\xff\x8b\xa8\x92\x24\x99\x9a\x96\xdc\x0d\xb6\x18\xb1\x4b\x57\x78\xc1\x19\xaf\x0b\x48\xc7\x4c\x70\xc5\x23\x9e\xa0\xc3\xc2\xa0\x1b\xf3\x48\xa2\x7f\xd7\x62\x6d\xb0\x94\xf0\xa3\xd6\xe3\x5a\x60\x60\x0d\xd2\xf3\x2b\x1d\x2a\x6b\x10\xaf\x1e\xab\x10\xc5\xcd\x2b\xb6\x58\x86\x55\x3c\x80\x16\x34\xe5\x1c\x5b\xa4\x37\x21\xf4\xcb\xe4\xb3\x82\x47\x2d\x40\x7a\x8d\xa1\xec\xcd\x37\x65\x8d\xdd\x16\x78\x7a\x86\xac\x5b\x16\xc0\xe2\x5d\x59\xce\x60\xe6\x79\x10\xba\x4f\xf4\xe5\xc2\xc2\x52\x00\x32\x4f\x53\x2c\x16\xc7\xc5\x69\xab\x13\x67\x81\xb4\x06\x3c\x26\x71\x0b\x00\x2e\xdc\xc4\x1e\x2d\x1b\xc5\x60\xc5\x4b\x77\xa3\xf9\xb3\x1b\x41\x2d\xc3\x00\xb1\x80\xb0\x88\xc7\x96\xae\x8b\xec\xd3\xb2\xc4\xe2\xdf\xa9\xcb\x2a\x2e\x22\x46\x16\xc6\x38\xa6\x0c\x84\x87\x7d\xc3\x7d\xdc\xc2\xbe\xf9\x18\xaa\xe2\x92\xd9\x1a\xee\xd1\xf3\xeb\x3b\xf7\x4d\xf7\x4b\x17\xd6\xa1\x2c\xb2\xe3\x24\xc4\xc7\x63\x33\x24\xf0\x73\x71\xfd\x42\x6c\x87\xb1\xce\xe4\x3e\x37\xd7\xfc\xfb\x94\xdf\xd0\x44\xdf\x5a\x40\xe3\x47\x23\x56\xfa\xf9\x00\x91\x84\xa6\x94\xf9\xd8\x3a\xc3\xdc\xf9\xd4\x48\xcf\x8f\x54\xe9\x2d\x93\xf1\xa3\xe6\x60\x0e\xd7\x29\x50\xa9\x06\x6c\xe1\x48\xc7\x3b\xa6\xac\x05\x22\x97\x7a\x5c\x85\x8e\xae\x85\x59\xdd\xc4\xa1\x15\x48\x69\x40\x78\x70\x7e\x47\x4c\xb7\xe6\xce\x52\x11\x2e\x1c\xb4\x17\x34\x77\xe8\x00\xf1\x03\x0e\x00\x7d\x94\x62\x7e\xbd\xfc\xdb\x20\xa0\x0c\x59\x9e\x6e\x9b\x6c\x62\xc3\x87\xdf\xca\x4c\x77\x23\x88\xbb\xa9\x6c\xe2\x12\x61\x79\xea\x0e\xd4\x1a\x14\x37\xb4\xe2\x4f\x4c\xa2\x04\x1b\xa4\x1a\xdd\x10\x44\x3e\x1e\x18\x07\x69\x16\xf4\x65\xae\x17\xd3\x8d\xa9\xb1\x93\x10\xf6\xb5\xf9\xf7\x37\xc8\xde\x0d\xdf\x1e\xd8\xfb\x5c\x48\x8f\x00\x62\xf6\x1c\x6a\x34\x92\xd8\xd8\xd0\x01\x95\x78\x86\x45\x6c\xac\xe5\xa1\x56\x61\x32\x78\xb5\xfc\xb5\xe0\x39\x7a\xa6\x72\x3e\x62\xf7\xdc\x19\x1c\x11\xe3\x1e\xd7\xf9\x00\x94\xd1\x5a\x7f\x58\x02\x13\x80\x51\x37\x51\x80\x66\xc2\x5b\xe5\x1a\x41\x14\xec\x98\xf1\x98\x6c\x07\x60\x74\x5f\xf8\x2a\x9c\xff\x5a\x10\x93\x0f\x06\x37\x45\x5b\x3a\x2d\x91\x72\x4d\xdb\x7c\x75\xe3\xe1\x1e\xb2\xed\x40\x49\xe0\xe7\xb5\xd0\xb5\x43\x6c\x30\x7f\xab\x41\x2b\x4e\xe3\x0c\xb2\x81\x4b\x6b\xef\xd1\x92\xb7\xdd\x84\xa8\x01\xad\xa8\xd3\xdd\x6f\xe6\x1e\xc1\xb2\xfb\x00\x63\x8c\x66\x82\xe7\x99\x4f\x99\x77\xe9\x7e\x66\x1b\xac\x4c\x73\xce\xa6\xfc\xc4\xea\x54\x17\x94\x3d\x1a\x8a\x7f\xa9\x3d\x32\x80\xd8\x24\x2e\xc1\xb8\xb9\x2a\xad\x30\x87\x43\x44\x59\x94\xe4\x70\xf1\x49\x85\xa3\x47\x03\xea\xdd\x66\xf4\xd5\xdf\x8c\x57\x27\x53\xb6\x48\x4c\x79\x92\xd8\x6e\x8b\x0b\xb4\x28\x63\xfd\x44\x31\xc2\xe8\xe1\xf6\xbc\xb9\xef\x47\x5a\x77\xe6\x34\xdf\x9e\x65\x02\x81\xff\xf9\x81\xae\x15\x77\x59\x81\xc5\x23\x25\x52\xf7\xc6\xa5\x36\xd0\x55\x4d\xa4\x9f\xb0\x22\xdb\x66\x42\x19\x0c\xb0\x35\x22\xf5\x6a\xe0\x6a\x4b\xad\xc7\x5b\x22\x93\x15\xa8\x62\x10\x1a\xd4\x0e\x91\x16\x06\x6b\xc1\xc3\x35\xb0\x1b\xe0\xfd\x6e\xf3\xa9\xbc\xbb\x62\x3a\xcb\x87\x99\x10\xb2\x06\xda\xc0\x9d\x7e\xbd\xe3\x20\x4b\xaf\x2e\x1b\xe3\x33\x36\xd5\x17\xea\xa0\xbf\xb1\x95\xec\xd6\x61\xb6\x8e\x1c\x8d\x78\x2d\x7d\x8e\x88\x1f\x89\x0b\xc3\xf1\xb2\x98\xeb\x77\x06\xbe\x2d\x5e\x2a\xee\xe0\x2d\xb4\x0d\x84\x1f\x88\xad\x5b\x86\x4d\x68\xf1\x6b\x9c\x76\x06\x94\x2b\x3a\x3e\xb3\x1f\x5f\xd6\xe0\xe5\x3c\x2b\xba\x84\x2c\x3e\x0f\x9c\x92\x62\xa6\x4f\xb6\xeb\xf5\xff\x63\xef\xdb\x9a\xdb\x48\x92\x73\xdf\xf7\x57\x54\x84\x1f\x24\x9d\x03\x82\x3b\xb3\x61\xc7\x58\x11\x7e\x80\x28\x6a\x87\xbb\x12\xa9\xe5\x65\x66\x7d\x0c\x07\x54\xe8\x2e\x00\x6d\x36\xaa\x5a\x5d\xdd\xa4\x60\xaf\xff\xfb\x89\xca\xcc\xba\xf4\x15\xdd\x00\x29\xc9\xeb\x79\xb0\x77\x44\x00\x75\xbf\x64\x65\x7e\xf9\x7d\x1d\x4e\x48\xb4\x08\x0f\x6a\xd2\x5d\x76\x50\x83\xb0\xc6\x81\xba\x87\x54\x95\x2d\xe5\x11\xfd\xf0\x3c\x45\x3f\x53\xb1\x01\x17\x84\xd7\x0b\x32\x27\x5a\xd5\x15\x81\xda\x42\x29\xcf\xd7\xf8\x40\xd2\xa2\xd0\xaf\x5a\x66\xd8\xe7\x3c\x1c\x31\xc3\x07\x68\xb2\x86\x71\x4f\x30\xbf\xfb\x76\x9a\x6b\x65\x95\x5c\xd4\xdd\xca\x4e\xdd\x38\x60\xce\xf3\x89\x18\x91\xca\x91\x89\x3b\x36\x7b\xa5\x9b\x33\xe5\x48\x6d\xee\x4b\xbe\x75\x8c\x00\x56\x21\x98\xf2\xbb\xb0\x71\x4b\x01\xbc\xb8\xdd\x6d\x38\x5a\x84\x3b\x6c\x02\x89\x62\x76\xb5\x60\x2e\x67\xf6\x2b\x9e\xdd\xcb\x3c\xcc\x72\x34\xc0\x01\x1f\x8a\x68\x68\x78\x5f\x71\x3f\xea\xd4\xb9\x8e\x4e\x8c\x4d\xde\xac\xeb\x88\x9b\xf7\x9d\xbb\x8d\x48\x31\xc8\x71\x78\xf6\xaa\x35\x91\x7c\xfe\x11\x95\xdf\xd8\x2e\xd6\x94\xf8\xdb\x2a\xde\xff\x96\xb2\x8c\x11\x54\x50\x28\xfe\x8f\x18\xd2\x74\xe7\x97\xa9\x27\x89\xab\x55\xd6\xdc\xad\xc5\x51\xa7\x71\xc2\xb7\x8b\x5c\x75\xcb\x59\x0d\x18\x26\x5b\x44\xc5\xbf\xb3\x41\x79\x8b\x1d\xfb\x5c\xf2\x14\x2f\x37\x49\xcb\xd1\x36\x1b\x4c\xe5\x1f\xff\x89\xcd\xe0\xf6\x61\x1f\xe0\x5c\x84\x00\x3f\x94\x56\x28\x96\x6c\x33\x91\x6b\x25\x79\xa7\xae\xdb\xfd\x4f\x7a\x41\xda\x34\x0b\x1e\x45\xaa\x6c\xea\xd0\x8c\xe8\x49\x4b\x69\x61\xa7\x38\xbb\x2f\x97\x22\x97\x02\xb5\xeb\xe0\x7b\xcc\x7e\x6f\x50\x73\x15\x2f\x8b\xcd\x8f\x8b\x28\x4d\x06\x0b\xe6\x40\x76\xd1\xcc\xfc\xec\x0c\x7f\xd5\xd7\x81\x4a\xf9\x95\xa6\x4b\x86\x9f\x31\xfc\x6c\xca\xde\xf0\xe8\x5e\xc8\x98\x65\x69\xb9\x4e\x88\x4c\x00\x6e\x28\x38\x2e\x03\xf7\x6c\xb5\x63\x68\x5b\x60\xf9\xe6\x1a\x9a\xcb\x2d\xbf\x47\x12\x5b\x32\x22\x23\x9e\x76\x52\x51\x39\xb3\x7a\x91\x34\xd7\xee\xde\xd9\x72\xf7\x61\xb3\x98\xfa\xda\xd3\x25\xe6\x56\x3c\x6e\x14\x45\xa4\x2b\x56\xfd\x88\x8d\xeb\x56\x6b\x83\xf3\xc5\xe6\xe5\x3b\x15\x43\x6a\x0c\xee\x5e\x70\xf7\x02\x11\x73\x29\x19\x07\xda\x98\x17\x9a\x95\x99\xb5\xcf\xc0\x0f\x99\x42\x54\x18\xa7\xc0\x7c\x90\x25\xe6\x95\xb6\x11\x73\x09\x48\x5b\xe6\xba\xd7\x10\xbb\x62\xc2\x03\x62\xda\x8e\x86\x15\x92\x26\x1c\x17\xe3\x6c\xf0\x38\xef\x59\xa7\x03\x51\xc4\xc5\x46\xc8\xc5\x01\x74\xc2\xc3\x27\xad\x82\x18\x26\x33\xd8\xf9\x73\xdd\x10\x96\x32\x21\x01\x29\x72\xe1\x87\x5c\x99\xc9\xaa\x66\x46\x27\x9a\x69\x5e\x24\xda\x9c\x65\xad\x23\xee\xa9\x2a\x8e\x19\x75\x3e\x8e\x1f\xa3\x85\x1b\xa3\x36\x16\x2e\x2b\x61\xca\xde\x81\x17\x2c\x78\x19\x28\xc7\x34\xd1\x75\x60\x15\x1b\xd1\x49\xb9\xf8\x14\x70\x1e\xdb\x83\xe0\xfb\xbd\xce\x4d\x97\x81\x32\x65\x33\x1f\x7d\x40\xae\x0d\x8c\x2b\xec\xe9\x91\x48\xb5\x38\x64\xf1\x0d\x72\xd4\x41\x84\x1e\x16\x10\x03\x4b\x4a\x9b\xbf\x7b\x06\x79\xd7\xcc\x47\x48\xf2\xe4\xf7\x42\xf6\x79\x63\x86\xb7\x10\xdd\x65\xbd\x2e\x01\xe7\x87\x53\xe8\x8a\x3b\xa4\x81\xc3\xb7\x9d\xa7\x37\x49\x56\xa7\x66\xc8\xcd\x33\x24\xba\xa7\xd4\x12\xf4\xc6\x12\x41\xca\xe3\x46\xe9\x70\x9f\xd9\xf9\xc3\x97\x6c\x5e\x3a\x96\x70\x48\xcd\x71\x03\x8c\x98\x1c\xa9\x42\xfe\x14\x68\xb5\xdb\xa4\xe8\x69\x76\xf3\xcd\xec\x11\x0a\xc3\x00\x51\x2c\x5b\x54\x73\x37\xff\xf9\x27\x7d\x05\x3b\xf6\x29\x32\xf5\xdb\x65\x59\x8f\x47\xc9\x1f\x18\x1f\x70\xf8\x2f\xaf\xe9\xca\x63\xc7\x2d\x91\xa9\x98\xf9\xe5\x35\x5e\xc0\xf5\xdb\x77\xab\x26\xfc\x3a\xa8\x6f\xfb\x56\xf6\x87\x00\x44\xc0\x96\x65\x82\x1a\xea\x15\x83\x50\x59\x8b\x03\xf8\x8f\xe1\xfa\x4f\xb4\xbb\x4f\xda\xd7\xd8\x47\x15\x1f\xb3\xb0\xc6\xd3\xe9\x35\xd7\xf5\x00\x8c\xb1\x6e\xd3\x7c\xef\x19\x89\x4c\x75\xa3\x43\xe3\xc5\x70\x95\x74\x80\x03\x2c\xcb\xd5\x0d\xa8\x8b\x74\x31\x56\x04\xc4\xfb\x36\x05\xcd\xcc\xb3\xa9\xc6\x25\x44\x74\x4d\x0a\x45\x97\xfd\xf5\xcf\xd9\x9f\x6e\xae\x2e\x4f\xb6\x3c\xd7\x1b\x0e\x19\xc1\xb6\xac\x89\x15\x6c\xc3\xe7\xb1\x8d\x7a\x25\x72\x2e\x4f\xd8\x5a\x4d\x30\xc6\xfa\x9a\x6d\x8a\x22\xd3\xaf\x4f\x4f\xd7\x49\xb1\x29\x97\xd3\x48\x6d\x4f\xfd\xd0\x9c\xf2\x2c\x39\x5d\xa6\x6a\x79\x9a\x0b\x40\xd9\x9e\xfc\x30\xfd\xf1\x07\x98\x99\xd3\x87\x1f\x4e\x21\xb2\x36\x5d\xab\x7f\x78\xff\xe3\x3f\xff\xe1\x9f\x4c\xc1\xd9\xae\xd8\x28\xf9\x9a\x02\xb8\xbd\x65\x9f\xa0\x55\x7e\x8a\x3f\xa9\xd5\xf2\xcf\xd3\xdf\x87\xcd\xa0\xaf\x6e\x55\x2c\x52\x7d\xfa\xf0\xc3\xc2\x4e\xcc\x34\xdb\xfd\x86\x4b\xfd\x66\xb8\xd4\xfb\xa4\xf8\x0d\x97\xfa\x4d\x71\xa9\xc3\x2d\x1c\x77\xc6\x00\xd1\xa7\x3f\x1f\xcd\xdf\xdd\x19\x69\x5d\xef\xfb\xce\xa1\x96\xcb\x21\xcc\x1a\x38\xe2\x8a\xb8\x17\xa3\x9e\xd8\xb5\xee\xba\xa7\x43\x87\x8b\x6d\x2c\xd9\x7e\xa7\x31\x3f\x2a\x49\x1a\x50\x20\x49\x04\x44\xce\xe8\x12\xcc\x78\xd2\x86\x36\x25\xb4\xd3\x31\xe3\xf7\x9c\x94\xe4\x4f\xcd\x45\x4e\xdd\x3d\x90\x87\x3c\xc5\x5f\x5b\x6c\x96\x7a\xb4\xfc\xe3\x4f\xc1\xda\x3d\x50\xc7\xd5\x91\x11\xe3\xe2\x81\xb6\xd8\x76\x75\x34\x63\xc3\xf5\x61\x20\xbf\x19\x52\xfe\xb9\x38\x9d\x53\xc5\xa7\x0a\xed\xc5\x61\x59\x14\x40\x24\x9c\xc8\x9a\xb2\x32\xcf\x94\x16\x7a\xca\xde\xd5\x94\x0e\x3d\x70\xf1\xfa\xdd\x19\xfb\xe1\xa7\x7f\xfe\xc3\x5c\xbe\x6c\xb9\xb7\xe1\xbc\x57\xf9\x9a\x70\x94\x70\x5b\x6f\xb9\x2e\x44\x7e\x9a\xaf\xa2\x53\x3c\xe5\x4e\xcd\xef\x4f\xa8\xd2\x13\xb5\x3a\x71\x94\xc4\x27\xc4\xce\x3a\xdd\xc6\xe3\x08\x06\x2a\x4b\x0f\xef\x1a\xba\x68\x34\x5c\x4a\x48\x45\xa4\x56\x8e\x7c\x1e\xf3\x5c\x50\xa7\x42\xad\x5a\xfe\xe3\x4d\xaa\x96\xfa\x95\x23\x40\xe3\xda\xd6\xe1\x19\x89\xba\xb7\xe6\xd3\xb0\x93\xdb\x25\xf2\x9c\x8e\x0a\x7b\x96\x84\xcf\x91\x31\x03\xdf\xbe\xd9\xfc\x75\x8f\x7c\x0c\x3c\x57\xa5\xb4\xec\xce\x4a\x0a\xb5\x02\x14\x1f\x58\xc2\x16\xa5\x00\xbe\x5a\x73\xd3\x7a\xee\x85\x5c\x64\x78\xc1\x40\x54\xa1\x7b\xb8\x8f\x64\x38\xdf\x37\xce\xcf\xc1\x70\x7e\xec\xb8\xd3\x81\xf2\x8d\x06\xfc\x58\x28\x21\x6e\xa5\x31\xa8\x0a\xf3\xfd\xbd\x11\x54\x77\x0e\x78\xf5\x21\x4f\x26\x9c\xf1\x1c\x8c\x34\x71\x52\xa8\x13\x20\xad\x01\x2a\x14\xd4\x1c\xe8\x82\x55\x40\xe4\x79\xcc\x35\x69\xbe\x3f\xa0\x9d\x68\x98\x7f\x09\x1a\x4a\x36\x89\x46\x0a\x4f\x82\x64\x25\x52\x8a\x9c\x62\x6a\x7b\x6f\xd4\x91\x71\xe9\x70\x2a\xfb\x11\x59\x81\xae\x75\xc0\x07\xef\xf0\xf8\x3c\x38\x04\xa6\x0c\xac\xcf\x8d\xda\x2a\x63\xce\xa8\x52\x07\x1f\xe2\xeb\x05\x2e\xe1\x4e\xdb\x6b\xcb\x33\x24\xa9\xfb\x76\xbd\x31\x5b\xcb\x7c\x84\x4e\xbd\xf0\x4b\xa3\x24\x36\x96\x55\x51\x81\x3d\xed\x77\x6c\xf0\xfd\xeb\x06\x50\x0f\xa8\x53\x07\x52\xb1\xc4\xf1\x9c\xfc\xa7\x79\xd7\x98\x25\xe5\x5e\x0a\xee\xe6\x46\x90\x0e\x72\x31\x86\x74\xaf\xd6\x9a\xef\xcc\x96\x2e\xb7\x23\xe7\xc0\x81\x8c\x87\x4c\x00\x97\x08\xbb\xb5\x78\xdb\x93\x56\xc0\x6d\xd7\xbe\xb4\xc2\xac\xf1\xc2\xf2\x85\x8e\x6b\xea\x8d\x2b\x80\xa8\x41\x9b\xed\xf6\x74\x4b\x80\xce\xc6\x31\xc6\x03\xc1\xda\x16\x1d\xa0\x1b\x39\x7e\x33\x82\xc0\xca\x98\xb1\x83\x4a\x70\x71\x36\x46\x30\xd8\x0b\x5d\x03\x38\xce\xc5\xd6\xe7\xb1\x6a\x83\xf3\x23\x43\x9d\xcf\xde\x31\xad\x6c\x3c\x1e\xdd\x0f\x1f\xbc\x0a\xf5\x2e\x13\x13\xb6\x2c\xe1\xf3\xcb\xab\xdb\x10\xad\x91\x60\x6f\x4f\xa2\x8d\x88\xee\xc1\x61\x82\x57\x9e\x13\x6d\x24\x36\xbc\xb9\xf4\xd2\x5f\x85\xb2\xd0\x83\x9d\x63\x43\x77\x8a\x00\x2a\x67\x71\xa2\xb3\x94\xef\x20\xc8\x2b\x11\xa7\xef\x03\xc4\x2e\xc1\xc5\x1c\x05\xfb\xfc\xc5\xc3\x67\xda\xcc\xca\xcc\xff\x6e\xec\x58\xf2\x7c\x99\x14\x39\xcf\x77\xcc\x0f\x66\xf3\x3c\x60\x5a\x6c\xb9\x2c\x92\x68\x2e\xb7\x82\xcb\x10\x95\x47\x41\x6e\x33\xc8\xb1\x12\xc4\x17\xbc\x5a\x89\xa8\xf0\x84\x83\x60\xbc\xbb\x91\xda\xb7\x07\xc7\xf5\xdd\xed\xbc\xde\xae\xff\x9c\x48\x4c\x6f\x4f\xb6\x80\xf9\xa4\x35\x44\x57\xe3\x81\xc1\x1b\x90\x8a\xa3\x2b\xd7\x3e\x06\xe1\x5f\x76\x4d\xb1\xa5\x28\x1e\x05\xe4\xd3\x53\x02\x60\x9b\x8d\x7f\xb4\x5c\xc0\x71\xea\xbf\xed\xba\xc9\x01\x12\x0c\x37\x58\x08\x26\x73\xc4\x3f\xb2\xc6\xe0\xf3\x82\x52\x12\xc1\xdb\xf3\x82\xfc\x56\x2f\xe0\x9a\x36\xaf\xc7\xfc\x41\xc4\x73\x59\xa5\x55\x22\x9b\xd1\x6f\x38\xe6\x85\xb0\x9e\xe6\xb4\xb1\x63\x3c\xc8\x97\x7f\x0e\x54\x12\x9e\x44\xd2\x25\xdd\xf5\x08\x73\x61\xa7\x9f\xf3\x55\x65\x35\x01\x43\xeb\x7e\x00\x24\x4b\x68\x2b\x74\x43\xba\x78\x15\x3c\x85\x5b\x94\x8e\x34\x06\x19\xe5\x1c\x00\x96\xfc\x92\x0d\x4f\x67\x5b\x19\x73\x69\xb3\xa9\x57\x65\x8a\x2c\xa1\x5d\x52\x61\xc4\x21\x65\x33\x3f\xbe\x5d\x06\x90\xf3\xab\xb1\x40\x5b\xcc\xc1\x1e\x02\x30\x32\x9e\x75\x76\xd5\x0b\xa9\x51\x90\xda\xca\x0a\x81\xe3\x79\x2d\x0a\xb8\xcd\xe3\x32\xc5\xe4\x60\xf0\x98\x03\x1f\x15\x4f\x53\x96\x14\x7a\x2e\x1d\x7d\x16\x92\xa1\xc3\x09\x6b\x5d\xea\x56\x91\x56\x3a\x5d\x5b\xd2\xee\x07\x3b\x2c\x89\x92\xa2\x01\xe1\xde\x85\x52\x1c\x59\x26\x38\xe6\xb2\xe1\xb4\xcd\x65\xf8\xe6\xaa\x4f\x02\x25\x7e\x81\x46\xf9\x53\xe4\x60\xf5\x20\xf2\x41\xd8\x7d\xf4\x94\x4c\xd9\x0c\x7b\x67\x1e\x5c\x56\x65\x13\x5b\x4b\xf9\xf3\x84\xb4\x34\xaf\x9a\x42\x5b\x1f\xb9\x7f\xb7\x82\x9c\x73\x54\xa6\x3c\x4f\x81\x93\x7e\x55\xa6\x2c\x59\x05\x82\xa1\x30\x07\x48\x9e\x64\xa6\x2b\x52\x70\x57\x5b\x2f\xb9\xe6\x5b\x11\xe4\x6d\x93\x7b\x27\x0d\x30\x14\xc8\x08\x8d\xc1\x79\x53\xd6\xab\x29\x7b\x5b\x57\x98\x87\x3d\x11\x90\x2e\x26\x1a\x8f\x3f\xd7\xde\x20\xe5\x10\x95\xea\x93\x95\x79\x52\xbe\x08\x76\x5d\xc7\x0c\x02\x79\xfb\x38\x80\x86\xa5\xee\xef\x47\x0d\xb7\xa6\x1c\x9b\x9f\xd6\x60\x1b\x6e\x43\x74\x34\xd0\xde\x0a\x23\x1b\x19\x12\x56\x1e\xd0\x50\x47\x08\xda\xd2\xd8\x6d\x8f\x3e\x29\xcc\xe3\xc8\xa6\x06\x6a\x3f\xe3\x1b\x1a\xac\x9c\x10\x8e\x33\x64\x64\xd7\xbc\x18\x8b\xcd\x71\xc9\x38\xe3\x1b\xda\x8a\x83\x1a\xd2\x4c\x38\x3d\x46\xb6\xd3\x4b\xf0\x8f\x6f\xa8\x93\x7e\xf6\x7a\x00\x5e\x13\xbf\x92\x43\x69\x99\x0e\x5d\x0f\xf4\x5c\xd2\x65\x37\x3e\xfd\x73\xe6\xd7\x1c\x08\x1e\x31\xd3\xfc\x29\xbb\x92\x02\x91\x73\x6a\x15\x5c\x2a\xd4\x00\x52\x46\x02\xb2\x79\x19\x48\x50\xa7\x89\xbc\xb7\xd4\x12\x66\xcb\x4d\x18\xf7\xa5\xc3\xa9\x87\xcb\x06\x4f\x91\x0e\x5b\xb2\x4d\x9a\xe1\x08\xf3\x72\x58\x82\x66\xfb\x9b\x3f\x00\xa0\x8e\x3f\x01\xda\xfa\x31\x7c\x5a\x7a\x91\xe4\xee\x15\x57\xd1\x18\x0f\x51\xa3\x45\x52\xec\xf6\x8d\xef\xc7\x4d\x15\x85\x38\x42\xc8\xe8\xee\xf2\xed\xf9\xbb\x8b\xcb\xaa\xfa\xd0\x5f\xee\xce\xef\xaa\x7f\xb9\xbe\xbb\xbc\xbc\xb8\xfc\x63\xf8\xa7\x9b\xbb\xb3\xb3\xf3\xf3\xb7\xd5\xef\xbd\x9b\x5d\xbc\xaf\x7d\xcf\xfc\xa9\xfa\xa5\xd9\x9b\xab\xeb\x9a\xde\x91\x15\x2b\x0a\xfe\x74\x7b\xf1\xe1\xfc\xed\xe2\xea\xae\x22\x99\xf4\xf6\x5f\x2f\x67\x1f\x2e\xce\x16\x2d\xed\xb9\x3e\x3f\xbb\xfa\xe5\xfc\x7a\x8f\xe2\x91\xef\x6f\xeb\x90\x3e\x05\x7c\xec\x60\xfd\xab\x19\x5b\xe5\x89\x90\x71\xba\x43\xec\xbd\x7d\xd9\xd6\xc0\xb4\xe1\xdd\x9b\x6c\x85\x2a\x8f\x81\xd0\xdf\x6e\x50\xa1\x1e\x58\x30\xb0\x34\x4a\x99\xe5\xfa\xbe\x93\x23\xb1\xc8\x9b\x51\x81\xde\x4c\xa1\x22\xdf\xb9\x5c\xb4\xbe\xe6\x78\x06\x25\xaa\x84\x65\x22\xef\x6b\x0b\x58\x46\x79\x99\x15\xc9\xb2\x3b\x29\x62\x20\xb3\xd0\xf8\xb7\x37\xf2\xfd\xb5\x93\xa3\x5c\xb6\x1f\x8c\x95\xdc\x80\x63\x80\xc7\x50\xc2\xa1\xb2\x6e\xee\xd7\x16\xac\x99\x95\xcb\x34\x89\x58\x12\xd7\xfd\x29\x98\xc2\x86\x2e\xe3\x3a\x2d\x68\x26\x72\x30\x55\xcd\x0b\x20\xcb\xc5\x09\x2f\x8b\x8d\x55\x9c\x77\x99\x8c\x48\xd3\x29\xa2\x5c\x60\x2c\x40\x68\x70\xd2\xa2\x9e\x57\x50\x13\x34\x86\x32\xb8\x63\x20\x8b\x99\x06\x14\xed\x1d\x31\x02\xfc\x25\x96\x3e\xc2\x49\x8a\xdf\xef\x1d\x1a\x6a\x71\xa2\xeb\x62\xce\x70\xc3\xe3\x87\x56\x15\xcc\xf4\xdb\x9c\xd4\x4e\x15\x0b\x27\xd9\xe6\x6e\xb4\x77\x63\xdf\x1a\x0b\x17\x4a\x35\x99\x81\x4a\xa7\x8f\xce\x72\x01\x97\x08\x41\x01\xac\xff\x02\xa0\x2b\x94\xeb\x01\x29\x1e\xe6\xa9\xb6\x14\x1b\x9e\xae\xd0\xe2\x30\x53\xe3\xf7\x55\x73\x89\xde\xaa\x7b\x21\xaf\x71\xc2\xbe\xc9\x71\x28\xf1\xe5\xe3\x73\xfa\x9d\x47\xc8\xbb\x30\x4d\x1b\xed\xaa\xb2\xb9\x6e\x60\x4c\x15\xf8\x4e\x08\x3e\xc6\x94\x0e\xcf\xd8\x6b\xd3\xe4\x56\xab\xe4\x8b\x29\x70\x2e\x45\x2b\x67\x29\xe0\x85\x2c\xbb\x92\x3b\x97\x01\x1b\x85\x14\x35\xf7\x42\x82\x9e\x18\xca\x0d\xef\x5d\xb3\xe3\xfc\xe7\xcd\xb9\xe8\x71\xe8\x83\xcf\x2f\xa9\xc8\xac\x85\x51\x1e\x3b\x4e\x05\xe6\xd8\x4c\xd9\x5b\x22\xde\x30\x7f\x39\x7b\x7f\x71\x7e\x79\xbb\x38\xbb\x3e\x7f\x7b\x7e\x79\x7b\x31\x7b\x7f\x33\x74\xfb\x3d\x45\x5e\x54\x6d\xf7\xd5\xd3\x83\xdc\x09\x71\x4a\x3b\xcf\xa7\xe7\xba\x4e\xf9\x6d\x07\x53\xb2\xbf\xf5\x49\x9c\x2d\xe2\x44\x47\xe6\xfa\xdb\x2d\x84\x8c\x81\xec\xf9\xa0\xa5\xda\x5e\x54\xbd\x17\xee\x1b\xcc\x7d\xc3\x9e\x20\x78\xdb\x3d\xd8\x15\xed\x3e\x07\xd4\x1d\xb8\x21\x73\x61\x36\x7f\x6c\xde\x07\xee\xb6\x99\xee\x57\xf8\x30\xc5\x1d\xd7\xb7\x6a\x11\xf5\x3e\x61\x7b\x13\xad\x4b\x6e\xce\x47\xfb\x35\x80\x1c\x76\x8c\x0a\x31\xf0\x85\x8c\xd3\x49\xa0\x96\xca\x12\x3d\x97\x5b\x2e\x63\x5e\xa8\x7c\xd7\xd1\xc5\x61\x87\x67\xb8\x6d\xaa\x47\x68\x78\x65\x4b\x21\x62\x3b\x0b\xf8\x55\x2e\xeb\x4b\x09\x79\xa9\x6f\xaf\xfe\x7c\x7e\x79\xb3\x38\xbf\xfc\x65\xf1\xf1\xfa\xfc\xdd\xc5\x5f\x1d\x12\x32\xe3\xba\x4d\x1d\x31\xcb\x85\x39\x5d\x2c\xcd\x47\xeb\xf9\x82\x92\x85\xb6\x1c\x92\xa9\x4a\x56\x73\x69\x4f\x96\xdc\x17\xbf\xc9\x55\xb9\xde\xb4\x17\x54\x6f\xe5\xc7\xd9\xed\xcf\x07\x35\x13\x48\x98\x50\xd7\x0c\x77\x5b\x13\x11\x9a\xac\xe8\xdc\x43\x18\x69\xad\x79\x40\x25\x06\x5f\x6d\x8b\x32\x74\x9c\x68\x07\xbd\x5e\x9a\x87\x56\xaf\xf1\xdf\xf2\xf5\xae\x05\x74\x1b\x9c\x9b\x95\x6b\x04\x10\xca\x28\x8f\xd9\x28\xed\x75\xcb\xdf\x2a\x37\xd8\x8f\x27\xa9\x58\xaf\x45\x8c\xcb\xab\x5e\x30\xf9\xe0\xe8\x08\x8c\xfc\xbd\xde\x36\x8a\x24\x60\x77\xc4\xc5\xec\xf0\x5e\xc3\x0f\xf0\x8f\xee\x27\xed\x67\xc5\x99\x15\xc9\x8e\x94\xd4\x05\x97\x1d\x81\xe4\x87\x26\x42\x73\xd0\x51\x74\x95\x33\x97\xfc\x44\x0e\x13\x1b\x32\xf0\xfb\xa0\x0b\xf0\x72\x3c\x2e\xd4\xb5\xe3\x5a\x64\x29\x8f\x84\xcb\x61\x40\x06\x3c\x78\xd7\x1f\x12\xc0\x23\x99\x40\x49\xfe\x96\x40\x3e\x30\xd0\x14\x6f\x59\x02\xe0\xb9\xbd\xb6\xe7\xf1\xf3\xbb\x56\x7a\x1f\x6e\xc4\x7b\x05\x8e\x66\xd4\x69\x22\xe8\x3b\xfa\xa2\x40\xfc\xac\x13\x96\x3c\x6a\x39\xd4\x6a\xfe\x85\x26\x1e\xdf\xcc\x55\x47\x37\xb7\xcc\x72\x6e\x79\x38\xd3\xb1\xcf\x5f\x58\x14\x79\x2f\x19\xe5\x53\x84\x23\x3e\xe6\x6a\x9b\x68\x31\x2b\x8a\x3c\x59\x96\xa1\x1a\xdf\x48\xc0\x5c\xe5\x71\xe2\x3b\x9c\xe5\x2a\x2e\x23\x4b\x09\x04\xbd\xf5\xb0\x1f\xf2\xf2\x59\xab\x23\x66\x27\x66\xf5\xd1\xcb\x4d\xc4\x27\x00\xe8\x9f\xcb\xae\x18\x9b\x3d\x18\x3b\x7c\x7f\x1f\xed\x55\x7e\xcc\x92\x6c\x59\x14\xdd\x83\x69\xd7\xc0\xb0\xc4\x5a\x66\xbf\x0e\x16\x70\x07\x6a\x8a\x96\xcb\x92\x63\x00\xbd\x6a\xa3\x74\x31\x80\xb8\xab\x66\x1c\xb8\x6b\x18\x36\xa6\x9a\x31\x83\x76\xc3\x86\x6b\x34\xe7\x8b\x68\x53\x6d\x38\xf4\xa6\xca\x9a\x57\x6f\xae\x33\x8f\x8f\x73\x9b\x0c\x0a\xa3\x4d\xd0\xd1\x90\x90\x63\xbb\xa2\x80\xe6\xe4\x1c\xc7\x79\xbb\x43\x8b\xd1\xbd\xe8\xf0\x32\x80\x73\x34\xe5\xa5\x8c\x36\x2c\x4b\x39\x26\x93\x6f\xb8\xc6\x25\x6d\xb1\x24\x7c\x99\xa4\x49\x01\x2c\x3d\x18\xe2\xac\x8d\xb0\x79\xe6\xf1\xfc\xde\x12\xe3\x72\x4f\xc9\xd4\xb7\xe8\x8f\xc4\xec\xba\x5e\x7d\x55\xd4\xae\xdf\xb2\xe1\x31\x34\x6c\x59\x12\x62\xd7\x4f\x87\x39\x88\x61\x59\xfa\xbe\x8c\x9b\x59\x2a\xf1\x63\xfd\xe7\x95\xf1\x6e\xb1\x5e\xc6\x23\x56\x88\xf1\x7d\xc4\xed\x53\xe7\x83\x6f\xdd\x59\xab\x54\xf1\x0e\x4d\x62\x5b\x36\xd2\xbb\x77\x95\x1d\xab\x72\xd9\x45\x28\x8c\xad\xea\x2f\xbd\x2f\x18\x62\xf7\xed\x53\x39\x4b\xc3\x03\x90\x17\xa2\x48\xc6\xf9\x7b\x82\x4e\xf3\x42\x9c\xc0\xcf\xdb\x0b\xa7\x0c\xc3\xc1\x7d\x6e\x2c\x34\x2f\x32\xe2\x8c\x36\xc0\x12\xb6\xad\xae\xda\xed\x7c\x0c\x26\xfc\xc8\xf9\x4a\xe4\x9e\xa5\xb4\x5f\xb7\xe0\x0f\x3f\xb6\x0c\x4b\xa3\xd3\x7f\x29\xb9\x39\x0f\xaf\x56\x37\xc8\x95\x73\x4c\xa7\x8b\xa4\xb9\xad\xda\x8f\x9f\x7a\xad\xb7\xd5\xf0\x5a\xb8\xf0\x07\x67\x22\xb7\xf5\xe6\xc6\xfc\x7a\xf8\x29\x74\x51\x71\xa3\x65\x79\xa2\x80\x33\x46\xad\x90\x81\xb1\x9b\x6e\xb2\xb5\xde\x23\x46\xf2\x73\x29\x4a\x61\x16\xd0\xb2\x8c\xd7\x4d\x2f\xf7\x08\x4b\xd9\x77\x69\xa3\x1e\xd9\xb6\x8c\x36\xcc\x16\xce\x62\x91\xf2\x5d\xa5\x6b\x60\x24\x16\x2a\x05\x02\xe7\x03\xd9\x64\xa3\x52\x17\x6a\x0b\x00\x63\x5f\x6e\x5e\x4a\xd8\xe5\x8c\xdb\xdd\xd5\x76\xbe\x57\xd8\xe5\x0e\x0c\x6d\xde\x7c\x3c\x3f\xbb\x78\x77\x51\x8b\x2b\xce\x6e\xfe\x1c\xfe\xfb\xd7\xab\xeb\x3f\xbf\x7b\x7f\xf5\x6b\xf8\xb7\xf7\xb3\xbb\xcb\xb3\x9f\x17\x1f\xdf\xcf\x2e\x2b\xd1\xc7\xd9\xed\xec\xe6\xfc\x76\x4f\x80\xb1\x59\x6b\xf7\x44\xf0\x80\xfc\xce\x42\x9e\x2d\x0b\xb8\xf5\x33\x50\xad\xaf\xd9\xcc\x52\x01\x56\xc8\x2a\x6d\x90\x18\x50\x25\xa8\x89\x4d\xb1\xe4\xb7\xbc\xe0\x67\xbc\xe0\xa9\x5a\x4f\xd9\x8c\x11\x20\x1c\x81\xfe\xda\x58\x48\xc4\x93\x66\x66\x07\x8b\x30\x66\x52\xe4\xdf\xf0\x5e\xe6\x50\xad\x88\xa1\x30\x15\x21\x21\xbe\xcd\x6a\x9b\xcb\xf3\x07\x21\x8b\x12\xd8\xba\x79\x9a\x32\xaa\xd6\x7e\x21\xc8\xd8\xb7\xad\xd4\xc9\x36\x49\x79\xee\x15\xe9\xae\xa8\x2c\x78\xa5\xd8\xb6\x3a\x82\xa6\x66\x3a\xb8\x7d\xc8\xdd\x5d\x30\x68\xf7\xd9\xfb\x0b\xb0\xfb\xa2\xc2\xca\xad\xd8\xca\xe7\x12\x19\xf0\xa8\xc6\x2d\x87\xe4\x93\x42\x91\x67\x15\xab\xa7\x2f\x77\x2f\x44\x7d\xcc\x26\xb6\x31\x88\xe7\x7a\x51\xba\x46\xda\xff\x38\x97\x45\xbe\x1b\x6c\xcc\xdd\x42\xb6\xb5\x06\x83\x9c\xb0\x6c\x55\x95\x3a\x74\x7c\x31\x5b\xfa\x25\x58\x78\x16\x68\x49\x71\x19\x17\x7e\x41\x5c\x4b\xc7\xa3\x23\x35\x37\xef\xf7\x3a\x0e\x21\x21\x0e\x8c\xc2\x52\x95\x32\xd6\x84\xba\xdb\x26\xf2\x74\xcb\xbf\xbc\xb2\x3d\x45\x82\x09\xa7\x15\x01\xe4\x61\x22\x35\xcf\xaf\x9d\x39\xe4\xfa\x87\x6b\x2e\x7b\xc6\x6b\xbf\x89\x6c\x4f\x56\x78\xeb\xf9\x87\x39\xe2\x07\x1f\xc4\xae\x6d\xfe\x1a\x7a\x3f\x88\x51\xa4\x0d\x0f\x85\x64\xb9\x30\x5f\x74\xe0\xc4\x14\x31\xa7\xee\xdf\x90\x84\x50\xd1\x24\x6c\x3f\xbb\xc3\x78\xff\x51\xdb\xa6\x15\x69\xf0\x0c\x82\x4d\x54\x93\x99\x33\xc4\x1d\x58\x97\x37\x25\x5d\x50\x40\xd5\x4c\xd6\x7f\xa8\x25\x5b\x41\x06\x12\x69\x8e\xe7\x02\x42\x1c\x30\x15\x96\x61\x1c\x28\xa6\x1a\x60\x06\xbb\x04\x52\xa1\xc1\xf1\x2f\xcd\x1b\x53\x7c\x2e\x29\x76\xfb\xc3\xef\xc7\xdd\xb3\x45\xbe\x63\x56\xcd\x22\xcc\x80\xa2\x04\x40\xba\x73\xa1\x5d\xa5\x4c\xda\x78\xe7\xae\x4b\x69\xae\xe2\xa7\x80\xbd\x0c\x8f\x6b\xd6\x2a\xa5\x7f\xee\x4d\x12\xb2\x2e\xf9\x1c\xbf\xff\x6c\x34\xa2\xbf\xd4\xd8\x43\xa9\x3a\x80\xa4\x53\xe9\xe1\x85\xb6\xe4\xd1\xfd\x23\xcf\x63\xf4\xdb\x02\x0e\x65\xca\x7e\x56\x8f\xe2\x41\xe4\x13\x16\x89\xbc\xe0\x44\xdd\xa5\x21\x10\x0f\x1b\x8a\xca\x99\x4b\xc8\xd0\x40\x1e\x34\x09\x72\xed\x45\xb2\xde\x98\x47\x74\x00\xa3\x50\xb9\x39\x8e\x0a\x64\x6d\xcc\x44\x44\x64\x49\x1d\x03\xb0\x4a\xf9\x43\x93\x8b\xec\x10\x96\x07\x76\xe1\xd2\x4c\x6d\x9c\xd2\xaa\x36\xf4\x01\x5f\x68\xc0\xe8\xd0\x44\x7a\x9b\x09\x5b\xab\x94\xcb\xf5\x74\x3a\x65\xa2\x88\xa6\xaf\x46\x2d\x74\x2a\x30\x8c\x7c\x3a\x78\x75\xaa\x94\x16\xe9\xce\x11\xfc\xb8\x04\x18\x40\x5c\x7e\x29\x84\xd4\x09\xfa\x79\x5a\x96\xff\x4d\x3d\x2a\xf0\x75\x83\x28\xed\xcf\xf3\xd1\xe9\x95\x1d\xe5\x80\x08\xd4\x88\x92\xf0\xfb\xed\x2f\xaf\x83\xd2\x85\xdb\xcb\x92\x4a\x8e\xcd\x81\xfd\x45\x75\x49\x9a\x1f\xc4\xbb\xd7\x5a\x12\x91\x94\x1c\x94\x37\xd8\x3e\x66\x8d\x54\xce\x23\xb2\x38\x7b\x12\x32\x47\xe6\x62\x0e\x71\x04\xdc\xd4\xa7\x7b\xf4\xb6\xd8\xaf\x4b\xd1\xda\xa1\x91\xb9\xae\x3e\x29\x7d\x8c\xe9\x84\xe9\x72\xe9\x0e\x5e\x5c\x2e\xf3\x15\xdc\xe9\x71\x10\x0e\xa8\x44\x3b\x20\x07\xcb\x87\x4b\x1c\x79\x54\x10\x1d\xd1\x85\xca\xf9\x5a\xb0\xad\x88\x93\x72\xdb\x7a\xd8\xb8\xe6\x1e\x83\xfb\x53\x69\xb9\xed\xa6\xf1\x3b\xd6\x80\xf6\x8d\xc4\xff\x3a\x83\xea\x06\x1b\xd0\x5e\xbc\xdf\xca\x03\x51\x7b\xd1\xf7\x4f\x63\x6d\x6e\xca\x3c\xd1\x40\x38\x79\x48\xca\xa3\x2b\x06\x8b\x86\xc8\xe9\x2e\x43\x9f\x73\x65\x76\x4f\x6c\x48\x8b\x7e\xa2\x71\x56\x21\xdc\xda\x7d\x29\xd4\xd1\x84\xa3\xe7\x08\xc4\x67\x0e\x8a\x70\x83\xd9\x18\xd0\xc0\x13\xdc\x09\x0a\x24\x4c\x46\xa1\xd8\xca\x26\xd1\xdd\x8b\x80\x96\x2c\x06\x82\xf8\x47\xe4\xb8\xf9\xf3\x4f\xda\xa2\x37\x08\x60\xe3\x2d\x96\xc2\x57\x82\x01\x91\x87\x1f\x2c\xae\x0a\x7b\x88\x45\x00\x79\x58\xcc\x65\xd1\x5a\x80\x87\x1d\x42\x59\xf8\x93\x5f\x78\x99\xb6\x7f\x9d\xca\x87\xaf\xa2\xd8\xd4\xec\xd7\x1b\x86\x43\x4d\x54\xe2\x79\x5f\x43\x83\x42\xf6\x23\xbb\x60\xb8\x16\x07\x58\x82\x95\x79\xc0\x41\xb7\x5c\xf2\x66\xd8\x45\x11\x6d\xbc\xe5\x51\x55\x8d\x26\x25\x41\xea\xe7\xd6\x93\xa3\x23\x68\x36\x44\x1f\x26\x6b\xa9\x42\x5d\x0f\x25\x05\x44\xa6\xcc\x01\xa4\xc2\x62\x59\x52\xec\x87\x78\x8d\x64\x0c\xdb\xb7\xd4\x0a\x85\xd0\x1d\xea\x67\x25\xc0\x08\x4f\x8a\x04\x79\x86\x2c\x3e\x16\xdf\x44\x24\x4c\x57\x27\xcd\xae\x32\x37\xcc\x65\xb5\xaa\xc6\x20\x59\x0c\x56\x92\x0b\xe4\xba\xd5\xc6\x7a\x2b\x92\x07\xb3\x51\x9b\xcb\xda\x2d\x50\x38\x01\x9a\x6b\x6f\x2e\xb1\xd9\x01\x61\xee\xbd\xd8\xe9\x50\x05\x8f\x56\x14\xeb\x5a\x90\x89\xe9\x0f\xcd\xd7\xfe\xa9\x80\x81\x5b\x04\xaa\xfe\xc3\xee\x32\xac\xf4\x83\xf9\x71\x0f\xb8\xb3\x51\xb8\x59\x83\x3e\x4b\xd1\xfb\x14\xe9\x98\xf0\xe3\x4c\x73\xe8\xf1\x5b\x80\xce\x0b\xf1\x77\x61\xca\x09\x3c\x7c\xcd\xfb\x76\x2e\x89\x53\x3b\xb8\xe4\xcc\x81\xd3\x9c\x36\x4a\x9d\x46\x26\xdf\x5d\x85\xf6\x05\x68\x0f\x2d\x05\x64\xb5\x4a\x1b\x6c\xb5\x22\xaa\x73\x09\x55\x63\x72\xa9\xf5\xe1\xb5\x56\x78\x20\x28\x90\x26\xb7\x13\x08\x18\x64\x70\xe1\x37\x89\xf9\x0f\xe5\x14\xf1\xf5\x13\x09\x33\x7c\x33\xd9\x8a\xc1\xb3\x08\xbc\x9b\xf3\xb3\xeb\xf3\xdb\xaf\x06\x14\xb4\x28\xbd\xd1\x48\x41\xdb\xce\xb7\xe7\xef\x66\x77\xef\x6f\x17\x6f\x2f\xae\x9f\x03\x2a\x48\x1f\x1d\x80\x15\xbc\x21\xaa\xfe\x33\x25\x0b\xf1\xe5\xa8\x3b\x39\x2f\xe5\x82\x8f\xc8\x59\x71\x62\x1d\x7d\xe6\x0e\x16\xda\x94\x1a\x70\x3a\x00\xc4\x3b\x89\x37\x9a\x53\x16\x58\x79\xa7\xe1\x2a\x49\x53\x48\xe1\x75\xee\x75\x4a\x0f\x33\x83\x0a\xe7\x8f\xa5\xda\xa4\x33\x75\x2e\x97\x15\x25\x08\x70\xf9\x6d\xcc\x23\x18\x93\x77\x33\x33\x00\x79\x02\xa9\x91\x7d\x6a\x04\xeb\x44\x0a\xdf\x0c\x94\x3e\x2e\x25\xeb\xa4\x90\xa6\x49\x7c\x4e\x48\x14\x19\x5e\x43\x6d\x4d\xbb\xe2\x2a\xeb\xd3\x9a\x9f\xf6\x43\xd7\x43\xdc\xc4\x89\x44\xc3\xb4\xb2\x9b\x6f\xda\x97\xee\xa9\xdf\x02\x30\xee\x66\x26\x39\xc4\x20\x40\x5d\xd8\x4f\x24\x4d\x04\xaa\x14\xf9\xe0\xc4\x7d\x82\xd0\x21\xb5\xaa\x8d\xb3\x39\x0a\xcd\x58\x27\x10\xa9\xe0\xc4\x4a\x12\xa5\xa5\x2e\x44\x4e\x6e\x93\xd9\xaf\x37\x73\xf9\xc6\x5c\x5f\xaf\xe8\x16\x22\x25\x1b\xac\x02\x81\x2b\xaa\x52\xbf\xb5\x50\xc2\x13\xec\x25\xfa\xa8\xb7\x82\x4b\x8d\xca\xef\x69\x2a\x72\xbf\x32\xb0\x3d\x42\xc4\xa4\xfe\x07\x34\xac\xfe\xf7\x24\xfe\xad\x60\xd7\x9a\xf6\xd2\xa7\x24\x7f\x5d\x5f\x4f\x5d\x19\xe2\x00\x15\x7e\xce\x95\xd3\x92\xb1\x32\x74\x15\x11\xca\xba\x75\x11\x55\xf3\x47\x06\xad\xa5\x5b\x2c\xee\xb7\xa5\xf4\x84\x4b\x69\xc0\xbd\x1e\xde\x12\x6c\xa3\xcc\x01\xea\x64\x5e\x7c\x98\xd9\x31\x54\xa4\x00\xfa\x32\xc3\xd8\x7a\xeb\xd4\xa4\x0e\x8f\xc1\x7e\x40\x51\xc7\x41\x6b\x67\x2d\x54\x38\x5e\x53\xcb\xc6\x76\x7a\x55\x14\x9f\x87\x72\x6e\x66\x41\x86\x52\x15\x96\x3c\xc2\xe1\xfa\x08\xa4\x68\xbe\xe0\x58\x4b\x7a\xdb\x48\x4c\x20\xd6\x4a\x59\x1c\xa9\x44\x76\x1b\x82\x21\x2b\xe9\xb4\xd8\x8a\x30\x11\xdf\x26\xdf\x3b\xf2\x8e\x31\x8b\xef\x70\xad\xcb\xea\x9a\x73\x44\x90\x07\x81\x1d\x2e\xaf\x2e\xcf\x43\xa8\xc2\xc5\xe5\xed\xf9\x1f\xcf\xaf\x2b\x89\xd8\xef\xaf\x66\x95\x64\xea\x9b\xdb\xeb\x5a\x0e\xf5\x9b\xab\xab\xf7\xe7\x0d\xcc\xc3\xf9\xed\xc5\x87\x4a\xe1\x6f\xef\xae\x67\xb7\x17\x57\x95\xef\xbd\xb9\xb8\x9c\x5d\xff\x6b\xf8\x97\xf3\xeb\xeb\xab\xeb\x5a\x7d\x77\x67\xfd\xe8\x89\x4a\x37\xda\xdd\x3f\x3e\x38\x1b\x70\x62\xb6\x6e\xe3\xaa\x16\xe8\x11\xbb\x78\x20\xf2\x6c\xdf\x72\xb4\x79\xd6\x71\x48\x95\x8f\x1b\xc3\x34\x75\xd4\xaa\x7b\x7a\xf1\xd2\xca\xd0\x65\xfc\xb8\x63\xcf\xdc\x6a\x8b\xa7\x40\x02\xf6\x1a\x80\xae\x96\x9a\xe3\x56\x17\x90\x8c\x86\x43\x9b\x41\x04\x6b\xcd\x3b\xa5\x7b\x64\xfc\xec\x2d\xb5\x75\xec\x6b\xa7\xe7\x60\xda\x43\x65\xf3\x54\x34\x16\x7d\x8d\x0e\x2a\xb3\x59\xe2\x49\x6c\x0d\x05\xfb\x61\x70\x71\x43\x37\xcc\xca\x09\x96\x63\x97\xca\x64\x7b\xbe\x49\x3f\x6d\xda\xd8\xf6\x53\x25\xcd\xb6\xd7\x38\x36\x46\xb4\x1b\xb8\x8e\xc6\xb4\xfb\x96\xeb\xfb\xb1\xed\xa6\x4a\x9a\xed\x06\xb3\xef\xa0\x76\x83\xc3\xbb\x68\xe7\x3f\x19\x71\x88\x85\xc5\x54\x9b\xe7\x92\xb3\xdd\x57\x02\x31\xd7\x61\x6d\x34\x1b\xe0\x79\x9f\x97\x19\x1f\x1e\xc8\x80\xd6\xb8\xed\xca\x6b\x74\xe0\x37\xf0\x29\xf4\x70\x99\x0b\x7e\x1f\xab\x47\x9a\x8f\x3a\x32\x94\x0d\x3a\xcd\xab\x03\x64\xce\x70\x7b\x45\x80\x04\xbc\xa9\x12\x51\x6a\xbe\x78\x80\xc9\x25\x44\x68\x8d\x36\x58\xa0\x42\x5a\x67\x90\x01\xce\x1e\xe9\x67\x67\x2e\xd1\x9a\x6f\x53\x32\x35\xb3\x6a\x5a\x44\x9c\x0f\xd0\x55\x67\x43\x63\x70\x5d\x07\x13\x4b\x09\x1c\x65\x0e\x60\xba\x65\x0e\x6f\x26\x18\x90\x44\x82\x33\x39\x37\x0f\x9e\x5c\x44\x89\x16\x81\x9a\x53\xeb\x8d\xfd\xf9\x38\xed\x87\x82\x17\xad\x6e\xd7\xc1\xfe\x70\x1e\x15\x25\x4f\xd9\xe7\x52\xe4\x3b\xa2\xce\x43\x5f\x25\xfe\x25\xe2\x12\x33\x45\x0a\xb1\xcd\x20\x1d\x3b\x4c\x71\x98\xcb\x5f\x01\x28\x81\x53\xf0\x42\xb3\x3f\x02\xe4\xc1\x7e\x99\x2e\xe1\x2d\x2f\xe0\x2e\xfe\x0b\xd6\xe1\x3e\x9b\xce\x65\x45\x1d\x25\xf8\x55\x45\x28\x65\x3a\x97\x56\x9e\x20\x56\x91\x9e\xc2\x8b\x6f\xaa\xf2\xf5\x29\x09\xfb\x9a\xc5\xae\xee\x97\x4a\xdd\x9f\x0a\x79\x0a\x3e\xa9\xe2\x94\x97\x85\x3a\x05\xb8\x14\xce\xbf\x3e\xb5\xfa\x9f\x56\x40\x55\x9f\x6e\x92\x07\x01\xff\x6f\xba\x29\xb6\xe9\x3f\xe8\x6c\xf3\xe5\x64\x9d\xe6\x27\xe6\xb7\x27\xe1\x6f\x4f\xec\x6f\x4f\xec\x6f\x4f\xcc\xcf\xf0\xff\x65\x3b\x0c\xef\x88\x2f\xdc\xdc\x65\x93\xb9\x4c\xa4\x16\x79\x01\xd6\xcf\x63\x9e\x14\x5e\x86\x66\xc7\x5e\xfc\xd7\x7f\xb1\x69\xce\x1f\x31\x95\xf1\x2d\x2f\xf8\x47\xf4\x2f\xfe\xf7\x7f\xbf\x80\x80\x2a\x26\xf5\x64\x3c\xff\x5c\x8a\x62\x2e\xb5\x30\x9b\x90\xfd\x9f\xb9\x84\x08\xec\x76\xb7\x28\xd0\xef\x8a\x3e\xc8\x58\xb3\x7f\xc1\x32\x2f\x90\x46\x32\xd6\xa6\xa4\x8e\x74\x82\x84\xa7\x2d\x92\xd1\x1d\x2e\xfa\xcf\xe9\x5b\xfa\xfe\x88\x6d\xfd\x39\xad\xee\x6a\x2b\x84\xa2\x3f\xa7\x70\x81\xa6\x8a\x5b\xb0\x16\x73\x8b\x17\xde\xc9\xd4\xb8\xb6\x3d\xd2\x80\x06\x3c\x6b\x98\xbe\x7d\xaf\xdc\x20\x95\xb5\xf5\xdc\x37\x8e\x11\x88\x15\xf8\x38\x04\x44\xcf\x13\xb3\x43\x6e\xd0\x13\x0a\x96\x1b\xf6\x1c\x6c\x52\x0a\x9d\xbb\xf2\xd0\x71\xa1\xff\xf0\xfa\xf4\x74\xc2\xd6\x1a\xfe\x67\xf9\x19\xfe\x07\xd0\x43\x4f\xc5\xc6\xda\x18\x4c\x07\x84\x6b\xce\xf2\xfe\x99\x78\x0a\x14\xdd\xd7\x20\x00\xaf\x2d\xd3\x37\xa5\x8c\x53\xe1\x53\x20\x2b\x21\x91\x54\x59\xc9\x7a\x74\x8c\xd5\xa5\x56\x60\x8e\x97\x22\xe2\xe6\xe0\x6b\xd4\x8d\xe0\x52\xb5\x2a\x84\x44\x6f\x58\xee\x95\xd8\x38\x7a\xae\xc0\x2c\x06\x28\x24\x2f\x08\x72\x2e\xe0\x8f\x50\x09\x30\x6a\x4f\xea\x1f\xb1\x9d\x2a\x89\x1c\x1a\x28\x4f\x63\x11\xa5\xc0\xc0\x6f\x69\x5f\x58\x2e\x8a\x32\x97\x8c\xb3\x8c\xcb\x98\x6b\x58\x81\xab\x1c\xa2\x9d\x39\xe3\xcd\x86\x4e\x10\x8e\xab\xca\x02\xc8\x8c\x10\x59\x10\x8e\x04\xb2\x77\x07\x6d\x9e\x04\x8d\xc0\x3b\x01\x48\x84\x1b\x3f\x9c\xce\xa5\xd5\x0a\x23\x2c\x1c\x7a\xca\x22\x95\xed\x88\xaa\xa6\x3e\xe8\x89\xf5\x9c\xd1\x70\x4f\x3c\xde\xa4\xfe\xdd\x09\x4b\xaa\xa1\x35\x20\x0a\x2f\x02\xb5\x63\xab\x17\xfd\x52\xc8\x48\xc5\x22\xd7\xaf\xcc\x36\x4c\xdc\xbb\x03\xed\x87\x44\xfb\xc9\x80\x53\xca\x5c\x6e\xe4\x2d\x34\xc5\x3b\x45\x1d\x33\x3a\x15\x6a\xe9\x36\x3b\x67\xff\x56\xf9\xde\x51\x30\x6d\xed\xa5\xff\xfc\xaa\x88\x98\x10\xd7\x69\xdf\x9c\x87\xbb\x20\x70\xcb\x86\x27\x2e\x16\x8a\x36\x0e\x19\x27\x56\x5a\x36\x29\x40\xbd\x2e\x17\xba\x98\x4b\xba\x81\x27\x6c\x25\xb8\xb1\xf3\x26\x2c\xd2\x0f\x78\x18\xe3\x75\x5f\x3c\x2a\x8f\xc1\xb1\xba\x24\x00\x86\xad\x14\xee\x9d\xc4\xf8\x35\x40\x14\xf0\xa8\x40\x80\x41\xa7\x0a\xb9\x35\x55\x60\xb0\x5a\x0f\xc4\x03\xc6\xc1\xca\x5c\xd4\x25\xa5\x42\x95\x15\x18\x89\x1d\x06\x8a\x59\xbd\x1d\xf8\x81\x39\x78\xb0\x77\x08\x03\x09\x0e\x47\xb0\xb8\x09\x4b\x8b\xfb\xcc\xc7\x70\x43\xae\x71\xf0\xcd\x74\x6d\xaa\x9e\x81\x80\x06\x1c\xe6\xb7\x30\x3f\xdd\xeb\xb0\xd2\x22\xb7\x1a\x1c\xd8\x57\x64\x06\xdc\x24\x79\x7c\x92\xf1\xbc\xd8\xd9\xe5\x9b\x26\x4b\xa0\xee\x4f\x93\x7b\xc1\x66\x79\xae\x1e\x9f\x7a\x14\x3a\x8f\x96\xae\x17\xf6\x31\x48\xf6\xb1\xaf\xfc\x56\x5e\xd0\xba\xbb\xe3\x30\x0e\xd2\x2e\xc7\x47\x6b\x3d\xb9\x28\xf2\xdd\xc2\x2c\xc4\x6d\xd6\x79\x52\x0c\x4a\x9a\x18\x6e\xe4\x8e\xa3\x37\xad\xb9\x30\x3a\xe9\x4d\x2b\xb3\xfa\xfd\xd0\x9b\xb6\x30\x97\x36\xe9\x4d\x2f\x2e\x2f\x6e\x2f\x66\xef\x2f\xfe\x5f\xad\xc4\x5f\x67\x17\xb7\x17\x97\x7f\x5c\xbc\xbb\xba\x5e\x5c\x9f\xdf\x5c\xdd\x5d\x9f\x9d\xf7\xf3\x15\x35\x5b\xef\x4d\xf0\x13\x16\xd6\xf3\x9a\xdd\x06\x40\x0d\x4c\x36\x20\xfb\x9b\xb4\x2b\x61\x55\x99\xcd\x9c\xc8\xf5\x04\x36\xea\x6b\x76\x9e\xe7\x17\x5b\xbe\x16\x1f\xcb\x34\x05\x38\x15\x66\xf6\x9c\xe5\x02\x1e\x9e\x13\xf6\x51\xc5\x17\xc1\xef\x20\x1d\xb1\xb5\x1b\x50\x3f\x8f\xe3\x5c\x68\x8d\xd5\x4f\xa8\xfe\x00\x3c\xe4\x52\x1d\x09\x3c\xc7\x1f\x78\x92\x9a\xf7\xdb\x6b\xf6\x86\x47\xf7\x6a\xb5\xc2\xf4\x99\x89\x4b\x9c\x62\x9f\x4b\x55\x70\x26\xbe\x44\xc0\xd1\xd5\xbe\x4e\xde\xab\xf5\x37\x80\x2a\x0f\x08\x4f\x75\x3c\x52\x40\xa3\x6c\xd1\x7e\x9d\xb7\x1f\x04\xd4\xcb\x0f\xf8\xd3\x77\xf8\xcb\x76\x07\x65\x91\x3e\x41\x7a\xfc\x7b\xb5\x6e\x57\x8c\x01\xeb\x9a\x64\x6e\x28\x90\x10\x11\xd9\x86\x5a\x33\x9d\xc8\xfb\xb9\xfc\x75\x23\x24\x53\x65\x8e\x7f\x82\x67\xbe\x31\x33\xd3\x52\x6f\x04\x48\xc8\x4e\xd8\xa3\x60\x5b\xbe\x43\xb3\x19\xde\x04\x4e\xe6\x02\x96\x0c\xdc\x22\xe6\xd7\x69\x22\xcd\x69\x91\x25\x36\x2f\xa1\x3e\xf5\x4f\xf1\xe2\xb2\x0c\x75\xfc\x78\x02\xd9\xbe\xfb\xb4\x82\xcf\x03\x57\x99\xc7\x4d\x5a\x80\x10\x9d\xdc\xa0\xa2\xa9\xd4\x7d\x99\x79\x2e\xcb\x17\x36\x38\x09\xc3\xfd\xa0\x92\x98\xc5\x65\x96\x26\x91\x3b\x77\x1f\x55\xde\x49\xd8\x8b\x09\x34\xc3\x6f\x9d\x7a\x5a\x58\x5f\xc7\x5a\xb2\x73\x02\x24\x5d\x0f\x75\xef\x33\x93\x17\xb3\x44\x46\x69\x09\xfa\x60\xa5\x16\xf9\x49\x91\x27\xeb\x35\x18\xe0\x36\xd7\xef\xfb\x67\x37\xf6\xec\x89\xc7\xa7\xb5\x85\x49\xe7\xa9\x5a\x27\x11\x4f\x43\x70\xb3\x47\x45\x38\xfa\x54\xbb\xed\x49\x3d\x15\xf2\x20\x6c\x83\x3a\x19\x90\xb2\x5c\x00\x83\xef\x02\x8e\xf2\x05\x1d\x77\xc7\xb4\x7b\xc5\xcc\x03\x1d\xdb\x15\x92\x9b\xda\xf0\x82\xbd\xe1\x7c\xdd\x56\x42\x0b\x4c\x4c\x94\xd7\x66\xea\x51\x8a\x1c\x2c\x58\x80\x7d\x98\x9e\x4a\x05\xb6\x89\x93\xd5\x72\xf8\x64\x2b\x2b\xb7\x72\x40\x6c\xcc\x9c\x5d\x27\x0f\x42\x7e\x7d\x36\xea\xa0\x82\x88\x47\x1b\xb1\xb0\x76\xf9\x53\x1f\x59\xee\x02\x18\x79\x58\x59\x7d\x8b\xf0\x28\x75\xe1\x4d\x78\x3a\x61\x8b\x9b\x67\x17\x06\x12\x7b\x32\xb2\x4c\x23\x16\xb1\x88\xee\xbf\xfa\xd1\xec\x41\x56\xb6\x21\x8c\xb3\xb7\x22\xba\x67\x77\xd7\x17\x98\x0d\x9c\x14\xcc\x1c\x05\x7a\xe3\xf5\x7a\x3a\xdf\x6e\x05\x5f\x3f\x03\xa3\xd3\x50\xc1\x21\xcf\x31\xef\x64\xd6\x4c\x83\x08\x10\x05\xf9\x92\xe6\x90\xa4\x5c\x1a\x00\x82\xf1\xc2\xca\xd0\x80\x23\x9e\xe9\x2d\xa8\xce\x94\x45\x20\xd5\x96\xf2\xa5\x48\x3b\x18\x17\x33\x15\x2f\x6c\x9c\xe4\x58\x30\x4f\xa3\x2c\xeb\xc7\xa0\xa8\xa3\xcd\x63\xe0\xc6\x62\xbd\xa5\x2f\xb2\xfb\x9f\x74\x40\xaf\xa1\x42\xe2\x67\x78\xd7\x73\x0d\xe9\xdd\xab\x64\x6d\xa3\x6d\xc9\x8a\xb4\x71\x30\xa1\x1f\x54\xe0\xcd\x79\x69\x4a\xfa\xa8\x62\x82\xe9\x39\x12\x33\x63\x05\x09\xf2\x9e\x78\x5c\x45\xd8\x04\xa7\x94\xaf\xc1\x37\xa0\x0b\xc1\x63\xa6\x56\xe4\x4d\xcc\xb2\x34\x01\x4a\xdf\x18\xd9\xc3\x81\x3d\x43\x57\xd1\xf1\x61\x69\xb6\xb1\x01\xc9\xc7\x47\x0b\xc4\xeb\x8d\x37\xfa\x20\x97\xe9\x57\xd5\xc9\x5d\xb7\xa9\x8e\x55\x81\x73\xf9\x48\x87\x3e\xa1\xfb\xbd\x69\xeb\x54\x2d\x61\xa0\xba\x41\x71\x3d\x07\xb4\x39\x9d\xf2\x24\x1e\x73\xbd\xdb\x31\xb9\x72\x3f\xed\x6b\xe0\x95\xf5\x74\xb8\x9a\xec\x34\x33\x22\x5c\x0f\x23\xf8\xb5\x34\xf6\x7d\x6f\x6d\x08\x10\x6a\x17\x21\x74\xd6\x78\x41\x72\x03\xb0\x2b\xdc\x71\xdc\xf1\xac\xae\xf6\xe5\xa8\x89\x6e\x12\xa3\xec\x19\x4b\xcf\xa5\xd2\x3f\xc9\x47\xb0\x5b\xe0\xce\x75\x14\x17\x63\x38\xa3\xed\xd4\x21\xc6\xd7\x8c\xa7\x9f\xc4\x4a\xf2\xc5\xa0\x19\xad\x8f\xbb\xdd\xc5\xc7\x0c\xf9\x73\xec\xa8\xb2\x50\xde\xe5\x0f\xfd\xb9\x00\xd2\xd0\x30\xad\x0b\x8e\x8d\x8b\xb8\x03\x2e\x60\x6d\x30\xbb\x35\x47\x80\x50\x47\xc1\x60\xb3\x5c\xd8\xe0\xd1\x4e\x14\x2e\xb9\x3f\xb5\xaa\x60\x10\x1b\x71\xbd\xae\xb2\x9b\x58\x02\x03\xc7\x48\x05\x91\x0c\xb2\xf7\x22\xb5\xcd\x94\x04\x6c\x0a\xa6\x2a\xcd\x25\x15\x6e\xb5\x9d\x5d\x78\xa5\x92\xef\x36\x21\xaf\x16\x66\x4f\x08\xad\xd2\x07\x8a\xa3\x05\x12\x04\xa0\x0a\x67\x1a\x78\x66\x1e\x08\xe6\x39\x0c\x01\x5e\x3a\xde\x01\x0e\x5e\x13\x38\xce\xc5\x3a\xd1\x85\x08\x53\x04\xc3\xdf\x3f\x99\x16\x65\xe5\x05\xdd\x37\xf4\x9d\x5a\x94\xfb\x4c\x61\xb3\x6b\x47\xb4\x67\x97\x89\xf8\xc2\xfd\xae\x7f\x31\xd4\xb2\xb8\xfd\x21\x51\xb9\x05\x70\x0d\xe0\x13\x40\x23\xdf\x93\x76\xe2\x01\x6e\x92\x88\x89\x87\x7b\x54\x9b\x99\xa2\x75\xc9\x73\x2e\x0b\x21\xf4\x5c\x52\xf4\x11\x79\xcb\x42\x6a\x8e\x1a\x1a\xce\x19\xb8\x91\xd2\x05\xd2\x00\xc1\x4f\x56\x3c\x49\xcb\xbc\xf3\xcd\x89\xab\xf2\x20\xee\x81\xbe\x51\x3a\x83\x62\x59\xdb\xa4\xb9\x2c\xd6\x60\x17\x39\xea\x8c\x7a\xec\xb0\x9a\xe4\xd9\xd1\x05\x7b\xe4\x0e\x9f\x6f\xe7\x70\xec\x48\x6c\xfd\x49\x2f\x32\x35\xe2\xc4\xfb\xf3\x4f\xfa\xa3\xea\x48\x09\xd6\x9f\x1b\x8e\xb1\x9e\x18\xfa\xe7\x2e\x39\x05\xae\xef\x21\xfc\xb4\xef\x3d\x3e\x88\x92\x71\x6f\x90\xaa\xf3\xec\x82\x55\xbb\xe1\x32\x4e\xcd\x3b\x95\x17\xb5\x1b\xc8\x83\x7d\x8d\x5d\x5c\xd8\xc3\xb1\x3b\xb3\x0b\x12\x25\x16\x51\x23\xcb\x6e\xdf\x38\xd5\xd2\xf3\x7a\x01\x75\xb5\x5a\xaa\x49\x73\x6d\xc9\x1a\xfe\x66\x27\x11\x53\xb7\x61\xbb\x97\xe0\x2a\x59\x7f\x07\x8f\xac\x0f\xcd\x93\x32\xa2\xad\x48\xf7\x97\x43\x7e\x1f\xb9\x19\x21\xc9\xc6\x1c\x66\x21\x39\xf1\x5c\x92\xc6\x31\x46\x5e\x21\xe4\x86\x5c\x54\x9a\xfd\xe0\x32\x2f\x7f\xf8\x47\xcb\x44\xb4\x63\x2b\x18\x6b\xa0\xfb\x52\x51\x54\xe6\x10\x16\x25\xd7\x0d\x13\x78\x37\xe9\x51\x24\x1b\x70\x23\x3b\x30\x0b\x9a\x4f\x6d\xd6\x83\xf3\xd5\x55\x3a\x75\x0b\x2e\x1a\x54\x6b\x76\x77\x21\x89\xf0\xe4\xba\x60\xba\x10\x59\xeb\xa9\x54\x31\xba\xaa\x82\xe4\x47\x98\x5d\x5e\x0e\x7d\xa0\xad\x3b\xe2\x8c\x9e\x05\xcf\xe9\x3f\xdd\x5c\x5d\xb2\x8c\xef\x00\x17\x56\x28\x52\x92\x07\x32\xc6\xfa\xfe\xdd\x37\x03\xd5\xce\x57\x37\x1b\x8e\xa9\x05\x98\xb6\xfb\x6e\xa9\xc6\xa6\x0d\x05\x6b\x86\x96\xa4\xd9\xca\xb9\x4a\x4f\xb2\x94\xcb\x00\xfa\xab\xa7\xac\x56\x7d\x18\xeb\x75\x51\x1f\x42\xd3\x40\x03\xc0\x9d\x42\x6b\x21\x2f\x5b\xc1\xa1\x55\x8d\xf5\xa3\xc2\xbb\x9d\x67\x44\x2f\xe8\xed\x03\x4a\x1b\xf0\xc8\x6c\x13\x64\x16\xb0\x21\x6b\x87\x7a\xe0\x1a\x00\x89\x23\x26\xaa\x5f\x10\x7e\x2e\xad\xde\xaf\x7a\xd4\x2c\x46\xee\x85\x32\xd1\x1b\xf0\x4f\x62\x40\x00\xc0\x41\x74\xbe\x20\x72\x21\xe7\x52\x9b\x09\x05\x9f\xa6\x78\x10\xe4\xd8\xa8\x04\xe3\x2e\xde\xbe\x77\xf1\x7d\x9c\x24\x92\x64\xeb\x18\xfa\xc0\x30\x3b\xe6\x01\xd3\xaa\x50\xbe\x9f\xd8\xff\x03\xcf\xfa\xb2\xc6\x8e\x2e\x71\xdf\x2c\x39\xe6\x99\xba\xd5\x09\x5a\xad\xa0\xd2\x54\x49\x1d\x0b\x47\xef\x4e\x1e\x79\xfc\xb4\x92\x3f\xef\x17\x15\x18\xfc\x08\x1b\x96\x09\x38\xe2\xec\x09\x48\xe0\x1c\xb6\xc6\xd9\xcb\x66\x97\x83\x06\x0e\xb0\x57\xa1\x27\x6d\xca\x6e\x84\x60\x9f\x60\xa4\x4c\x65\x9f\x48\x63\x0d\xe0\x82\x05\x4f\x5a\x25\x70\xe0\xdb\x17\x72\xa5\x8e\x3b\x0c\xf2\x75\x03\x8e\x76\xd4\xa8\xb4\xb7\xf3\x58\xc0\x1b\xa4\x32\xca\xe7\xcd\xbf\x6f\xed\xd7\x1e\x78\xdb\x47\xff\x26\xa7\xac\x3c\xdb\x52\x73\x3f\xc3\x14\x1f\xc2\xf0\x54\x5b\x24\xa6\x97\x13\x64\x2d\xbe\x97\xea\x51\xa2\x2d\x40\x35\xb1\x97\x66\xff\xc1\x05\x86\x0e\x54\x34\x0b\x4a\x3c\x0d\x5f\x01\x8d\xf2\xcc\xfd\x9b\xdd\x60\xac\x08\xdb\x0c\xe2\x28\x1a\x8c\x1f\x92\x35\x81\xd3\xfc\xe5\x6c\xc2\xde\x4c\xd8\xd9\x84\x4d\xa7\xd3\x57\x13\x54\x6e\xa6\x16\xe1\x4f\x10\x39\x56\xf0\xb5\x29\x9b\xe4\x22\x56\x41\x05\x20\x80\x64\x2e\x2b\xcb\x16\xc6\xfd\xb7\x02\xcf\x83\xed\x02\xe6\x30\x52\xc2\x05\xc5\xd5\xa3\x8d\x4a\x7c\xa3\x00\xa2\x29\x22\x95\x5b\x90\xa7\x2e\x54\x6e\x01\x6b\x0f\x3c\xe7\x89\x84\xd4\x6e\xde\x84\xeb\x52\xcd\x01\xb9\xb3\xf8\xc2\xb7\xd0\xff\x44\x3a\x7e\x4b\x33\x4c\xb7\xae\xfd\xc5\x2e\x23\x87\xf4\x63\x9e\x14\x85\xb9\x9d\xf5\x5c\xde\xb0\xd7\xff\xc2\x66\x59\x96\x0a\x36\x63\x7f\x63\x6f\xb8\xe4\x92\xb3\x37\xec\x6f\xec\x8c\xcb\x82\xa7\xaa\xcc\x04\x3b\x63\x7f\x33\xc3\x66\xca\xbb\x54\xe6\x3a\xdc\x4d\x18\x67\xb2\x4c\xf1\xd6\x7f\x69\xc1\x60\xaf\x5c\xbf\xb8\x9f\x9d\xa5\x28\x1e\x85\x90\x4c\xab\x2d\x5d\x85\x7f\x75\x31\x09\x9d\xc8\x75\x2a\x0a\x5a\x0f\x55\xd8\x1e\x56\x70\x02\x3d\x7d\x3d\x97\xce\x97\xf7\x57\xd3\xe2\xbf\xb2\xbf\xb1\xcb\x32\x4d\x4d\x93\xcc\x41\x63\x16\xd2\x6b\x66\xd3\x28\x84\x9c\x3e\x26\xf7\x49\x26\xe2\x84\x43\x22\x85\xf9\xd7\xe9\x2d\xcc\xf6\xa2\xf4\x9c\x79\xe1\x9e\x76\x82\x33\xc7\x1c\x3d\xcf\x92\x94\xed\xe4\x90\xec\xe4\xf7\xbc\xfc\xaa\x3f\x1d\x6f\x11\x79\xa6\x50\xda\x0f\x64\xb0\xa2\x58\x50\x28\xbb\x74\xd0\x11\x50\xbb\x6c\x6d\x59\x2d\x57\x41\x78\xa9\x1f\x7b\xc8\x82\x86\xd8\x93\xbf\x21\x07\x68\x17\x0d\x3d\x72\x1b\xca\x2b\x95\x34\x77\xb0\x25\x3d\xd5\xcf\xa0\xa8\x90\x53\x5c\xf9\xa5\x2a\xa2\x56\x19\x62\x95\x0c\x52\x9b\xab\x35\xf6\x8e\x7c\x17\x98\xa0\x66\xb6\x69\x92\x9e\x9a\xad\x7a\x7a\xa9\xa4\x79\xb6\xea\x64\x8d\xf4\x44\x00\x23\x42\x99\x36\x6b\x14\xdc\x56\x4d\xd6\x60\x0b\x80\x7d\x60\x9a\x84\xd0\xb6\xc2\x9c\x02\x66\x0a\xd2\xdd\x5c\x9a\x5f\xd0\x8d\x04\x30\xf7\xc4\xb1\xd8\x62\x6d\x56\xd7\x9e\xea\xa2\x03\x39\x28\xbc\x65\x81\xf5\xe5\xd0\x1e\xb1\xe0\x28\x65\xeb\x08\xaf\xf8\x65\xc0\xe0\x46\xa5\x59\x7a\x0f\x8c\x7b\x2e\x45\xaa\xe4\xda\xac\x8a\xae\x43\x40\x6d\x79\x72\x0c\xb0\x24\x6c\x02\x16\xd6\xd9\x02\x73\x59\xd2\x57\x68\x4a\xcc\x3d\x99\xc4\xfe\x7d\xaf\xcb\xa5\xb1\x23\x9c\x47\xd6\xdd\x86\xd4\xb9\xae\x84\xe2\xe3\xe2\xcb\x77\x5a\xe4\x40\xb3\x8c\x08\x07\xe7\xed\xc7\x8b\xd3\x93\x6d\x60\x8f\x86\x6d\xaa\x5e\x60\x6c\xbb\x2b\x84\xa2\x09\x8d\xd4\xea\x01\xeb\xf1\x5b\x62\x64\x87\x20\x67\xdf\xcd\x2e\xde\xd7\xbe\xd7\x44\xce\xb6\xc0\x6b\x6f\x2f\x3e\x9c\xbf\x5d\x5c\xdd\xdd\x36\xbe\x67\x4a\xa3\x3f\x8d\x55\xfa\xb7\xa3\xf7\x14\xf0\xc1\xcf\x28\xf7\xb2\x50\x2b\x9b\x49\x39\xfc\x4e\x6f\x08\xee\x0c\x43\xa9\x84\x24\xdb\xa1\x30\x4d\x73\xe1\x74\xe6\x83\xcb\x05\x45\x24\x86\x35\xb6\x3e\x60\x57\xf2\x1d\xfe\xfc\xa3\x4a\x93\xa8\x1f\xf4\x66\xaf\xab\x8d\x7a\x6c\x41\x11\x2d\x05\xa0\x40\xc9\xff\x43\x8d\x42\x0b\xbd\x10\x51\xe1\x23\x6e\xcd\xce\xfd\xaf\x06\xda\xec\x7f\x83\xa3\x47\xd9\x0d\x1b\x08\x70\xba\x18\x1e\xdc\xad\x40\xb0\x09\xbc\xf2\xe8\x6b\x85\x5c\x3a\x88\x6d\x47\x9c\x5c\xd0\x95\x91\x87\x03\xfa\x71\xa3\x52\xf3\x16\x93\x31\x91\x95\xce\x65\x26\xf2\x48\x01\x40\x05\xf3\xe0\x15\x8b\x36\x49\x1a\x7b\xf1\x96\x97\x80\xe8\x05\xdc\xdd\x2b\x92\xe5\x13\x2e\xc6\x6c\x8b\xef\xb9\x75\xed\xb2\xb3\x7a\xe5\xc7\x79\xa0\x9e\x0e\xa2\xd7\xb7\xec\x7f\x25\x28\x19\x0e\x05\xd1\x0b\xd5\xa2\x85\x66\xd0\x2b\xed\x19\xe5\xe1\x05\x45\xf1\x95\x15\xff\xb4\x0f\xa7\xa2\x36\xaf\xb4\xcc\xea\x43\x09\xa4\xb3\x88\xa2\x42\x00\x89\x16\xd0\x9c\xad\xe0\x68\x8b\x79\x0a\x48\x9a\xd4\xb9\xf4\xf1\xd1\x17\x3a\xb4\xcb\x5a\xe7\x19\x39\x55\x2d\x08\x70\xc2\x5e\x54\x3a\xfa\x02\x48\x49\xa5\x82\xfa\x28\x86\x55\x19\x1a\x58\xae\x13\x96\x14\x73\x99\x68\x5c\x99\xb9\x48\xc5\x83\x69\x5d\xe8\x2c\x26\xac\x8b\x7d\x3b\xdb\x6e\x03\x8e\x9c\xdb\xf4\x63\x27\x73\x0f\x9b\x30\x0f\xc9\x2d\x39\x38\xa6\x63\xa1\x8d\xdd\x08\xb2\x1c\xe2\x8b\xd9\x00\x09\xc4\x42\x10\xfe\x11\x0b\x69\xdb\x07\xa8\x10\xd4\xb2\x9d\xcb\x8b\x15\xe4\x80\x42\xe6\x69\x1c\xe3\x2b\xd4\x0a\x35\x38\xa6\xb1\x84\x9c\xc3\x8a\xde\xe4\x76\x22\x48\x55\x12\x77\x92\x78\x10\xf9\xae\x00\xa7\x2e\x8c\xab\x14\xbc\xd8\xb0\xa4\x98\x00\x45\x9c\x3d\x29\xe7\x92\xc7\xa4\x54\x4e\xc5\x99\xa1\x81\x75\xdf\x33\xcf\xf4\xf9\x52\x3d\xf4\x19\xb6\xc7\xa2\xbe\x70\x57\x67\x29\x97\x0b\xbc\x41\xbe\x01\xee\x2b\x10\xfc\xec\x0a\x75\x96\xcb\x85\xa3\xb5\x79\x92\x76\x06\x9a\xd0\xa1\x0c\xaf\xb1\x63\x6d\x45\x13\x5c\x0c\x9e\xd6\xda\x3e\x4f\x9c\x9f\x86\xd0\x05\x39\xb3\x11\xd8\xe1\xa7\x80\x87\x84\xf1\x1a\x12\xc1\xae\xd6\x7d\x98\x30\xbb\x02\xbe\x57\x7c\xd2\x90\x99\xaf\xdd\x21\xf5\x69\x1f\x0f\x8d\x69\x58\x88\x07\xc1\x63\xf6\x34\xeb\x79\x21\x32\x9d\x7e\x94\x26\x54\xc6\xf6\x36\x08\xf7\x61\xf2\x84\x40\x3f\x9c\x73\xf3\xb4\x0b\xba\x86\xef\x30\xd5\x82\xad\x7c\x4a\x1f\x35\x9c\x53\x43\x3d\x25\x3e\xf7\x1a\xda\x35\x65\x17\x92\x59\x73\x6f\xc2\x5e\xe0\xc2\xd2\x2f\xc8\x05\x49\xaa\xc0\x14\x3b\x8f\x69\xf7\x50\xb6\x6a\x1d\x8a\x81\x39\x03\x7e\xbb\x61\x24\xa8\x97\xda\xf0\x59\xc7\xe5\x4d\x02\x39\x0b\x87\xa4\xa5\x63\x14\x71\x89\x05\xd0\x25\x89\xcf\xee\x1d\x1a\xed\xca\x7b\xb3\x7d\x87\x6d\xbc\x8b\xbd\xb1\x3f\x34\x43\x94\x95\x74\x9f\xda\xcf\x99\xca\xe7\xd2\x96\x46\x2e\x49\x8d\x5a\x4a\xf5\xa2\x02\x08\x35\xd9\xfc\xc1\x4a\x05\x10\x83\x95\xcf\x02\x55\x36\xcf\xbf\x5a\x3f\x05\x00\x14\xb1\x14\x5e\xcf\x7c\xca\x66\xbe\x36\x63\x78\x98\x05\xbe\xc5\x6b\xbe\xce\xd1\x98\xa6\x66\x50\x92\xc2\x52\x42\x06\xe9\x0d\xba\x04\x62\xd3\x55\x69\x0e\xa3\x80\xfd\x75\x2e\xcd\xe0\xb1\x55\x02\xb8\x5f\x1a\x97\xb9\xfc\xa0\xb4\xcd\xa6\xd7\x7e\x3c\x2c\x86\x94\x86\xed\x85\x53\x11\xa3\x3f\xbc\x85\x4b\x9b\x7c\xfe\x35\x89\x7f\xc8\x6b\x21\x4a\x8c\x9d\x2a\x73\xdf\xa9\x88\xcb\xb9\xfc\x0f\x33\x3c\xa8\x64\xed\x64\xe0\xd5\x0a\xb7\x30\xcc\x20\x04\x4b\x3e\x61\xa1\x2f\xff\xf1\xd5\xa7\x57\x88\x43\x2f\x35\x08\x37\x4e\xaa\x17\x88\x23\x02\x2f\xd3\x14\x22\xd1\xb6\x07\x8e\x8c\xc2\x57\xc1\xfb\x60\x39\xf4\xa8\x5b\xc8\xaa\x89\x31\x64\xa3\xf7\xad\x60\xef\x7c\x9e\xb1\x88\x17\xd1\xe6\xc4\xda\x72\x74\x8c\xd9\xdb\x8f\xa6\x0f\x15\xdc\x8c\xa5\xd5\xce\x85\x6d\x1e\x9c\xf9\xd6\xb1\xf3\x55\xd6\x8b\xe9\x02\x00\x6b\x6e\xeb\xc2\x30\x8e\x3c\x14\x17\xa7\x17\x62\xf7\x76\x9e\xfb\xba\x95\x65\xf3\x2f\x4e\xf2\x92\x4b\xbe\x15\x31\x7b\x01\x19\x53\x2f\xec\xe4\xcf\x65\xb6\x9c\xa6\xbb\x55\x41\x14\x4f\x66\x50\xa6\x20\x60\xb4\xe7\x96\x5b\xc4\xcd\x67\xd2\x9e\xc1\xee\x7c\x68\xb5\xdb\x3a\x6e\x6c\x5c\x4d\xc3\x0d\x16\xf4\x71\xb9\xd1\xb9\xa9\x42\x84\xaa\x4c\xea\x5c\xdf\x4f\xd8\x32\xe7\x12\xb4\x27\xe2\xd0\xa8\xf2\xbb\x13\x1e\xcf\xc8\x9f\x64\x53\x28\x24\x4f\x77\x80\x1d\x9f\xcc\x25\x92\x4d\x01\x2b\xf1\x2e\x4a\x93\x88\xad\x73\x9e\x6d\x6a\x76\x90\x78\x10\xb2\x00\x09\xd3\x6b\xc1\xf5\x71\xd1\xfa\xbc\x5e\x02\x1b\x1c\x4f\x99\x49\x78\x7d\x70\x59\xa3\x07\x85\xe6\x75\x5c\x2d\x00\xd7\x12\xf1\x62\x1c\x35\xc8\x5e\x02\xcb\x0a\x2d\x1a\x71\xf4\x40\x04\xd2\x74\x8e\xd9\x5a\xf7\x85\xbf\x71\x5c\x89\xb5\xc2\x82\xff\x8f\x0d\xd9\x3b\x16\x8c\xa3\xb8\x0c\x2f\xaa\x56\x24\xf7\xf4\x1e\xde\x73\x8d\x09\x40\xe4\xa9\xb0\x60\x61\x77\x70\x4c\x48\x62\x0e\x78\xce\xd8\x5f\xca\xa5\x4a\x2d\x51\xdc\xc5\x5b\xa6\x72\xd0\x68\x28\x14\xfd\x29\x89\xbb\xac\x83\x44\xc6\xe2\xcb\x51\x6c\x0d\xfd\x17\xbd\x35\x9b\x4d\x35\x81\x14\x40\xbd\xb3\x70\x3a\xe5\xc2\x5c\xc2\x85\x7d\x19\x37\xbe\xa5\xeb\xc8\xc5\x59\x5a\x6c\x00\x4e\x88\x40\x76\x3f\xa8\x5b\xbe\x63\xd1\x86\xcb\x75\xe0\x9a\x00\x74\x97\xc8\x54\x8e\x5a\x86\x0f\x40\x8b\xa6\x72\x9b\x0d\x4b\x39\x9e\x84\xa6\x77\x81\x04\x04\xb1\x2a\x9b\xc8\xc9\xd7\xeb\x5c\xac\x81\xa0\x60\x2e\x2b\x59\xea\x40\x09\x67\x65\x14\xb0\x9e\xbe\x24\xdf\xa7\x61\xca\xe8\x7a\x0d\x16\xf9\xce\xa5\x48\x92\x10\xa8\xdf\xcf\xf5\x61\x9d\xb0\x44\x4c\x27\xec\x47\x0f\xdc\x15\x91\x92\x2e\xc7\xb2\x23\xc1\xae\xe6\xf2\x67\x7b\x9e\x0e\x4d\x4a\x8d\xf6\xb6\xc3\x67\x0d\x39\xd1\xd6\x45\xd3\x9b\xa4\x5a\xf0\xa2\x1c\x71\x07\x91\x64\xf4\x99\xf9\xf1\x0d\xfe\xb6\x17\xdb\xce\x33\x73\x6d\x58\x3a\x23\xf3\x7d\x73\x73\x9a\xba\x3d\xdd\x71\xdb\x58\xef\x75\x20\xa7\xaa\xdb\x81\xfc\x14\xa6\xba\xe5\xac\xd8\xef\x43\x4e\x3b\x78\x18\x7a\xfa\x34\xd6\x45\x6c\x41\xbe\x04\xdf\xd7\xf5\x67\x6c\xcb\x09\x90\xe5\x2a\x2e\x23\x11\x9b\x9d\x0b\xef\x21\x44\xc4\x38\x3a\x88\xca\x21\xd9\x76\xd1\x56\x38\x6d\xe0\xd6\xfd\x5a\x3e\x87\x41\x34\xc2\x6e\xf8\xef\x3a\xfc\x0d\xd6\xe2\x6b\x1b\xf4\x70\x7f\xe2\x38\xe5\x23\xef\x29\x57\x7d\x95\xfc\x57\xe5\xc9\x3a\x91\xbc\x50\x39\x7b\xe9\x92\x3e\x5f\x39\xc5\xa0\x6e\x0b\x61\xe4\x31\x51\x19\x22\x3c\x26\xbe\xaa\xe1\xd1\xb6\x48\xcd\xb7\x74\xc1\xb7\x59\x48\xa7\xe9\xf4\x98\x69\x64\x52\x1c\x04\x67\x9b\x80\xef\x34\xd1\x3e\xb7\x6d\x2e\x29\xe2\x80\xf3\xa6\xf2\x90\x0f\xba\xf3\x6e\xce\xca\x62\x71\x20\x45\x0c\xfe\x78\x9c\xe3\x89\x60\x08\x1f\x78\xd6\x4f\xba\xc1\xc9\xe5\x80\xc9\x3d\x4e\xa4\xda\x5a\x2a\xd5\xf5\xd9\xaf\xb8\x30\x92\x04\xb4\x1e\x3a\xbf\x7e\x6f\x03\x45\xfe\x3d\x58\x79\x60\xc1\x44\x20\xfb\xa0\x06\xab\x02\x9f\xf6\xee\x58\x33\xb7\xb8\x65\xea\x38\x4b\x55\x19\x33\x3a\xd4\x28\x0c\x9f\x4f\xf1\x76\x04\x3a\xd0\xe9\xb4\x8b\x1b\x6d\xa4\x12\xac\x3b\x7f\xe0\x77\xed\x3b\x10\x3e\xeb\x38\x81\x7b\xb7\x3e\x8d\xec\xb3\x4d\x3d\x8d\x34\xcc\xbd\x3b\x8e\x47\xcd\xbd\xf3\x82\x03\x37\xd9\x38\x07\x29\xbc\x47\x93\x38\x85\xfd\x16\x06\x10\x5a\xd8\x53\x2b\x81\x59\x7d\x7f\x74\x75\x36\x83\xb9\xbf\xaa\x8c\xe7\x42\x16\x0b\xa8\x71\x5c\x65\x50\xc9\x47\xf8\x79\xc5\x60\x1a\xe4\x08\xfe\xb7\x5b\x85\xfe\x7d\x4b\x44\xf2\xef\xec\x86\x7c\x5a\xe6\xbc\x4a\x00\x44\xaa\xef\xd9\xcb\x04\x30\x47\x41\x2c\xd4\x4d\x5c\xc7\x74\x51\x87\x0e\x18\xbd\xa0\x43\x95\xa3\x7d\x50\x87\x7c\xeb\x21\x54\x0d\xa5\x90\x7b\x8f\x32\x67\xcd\x51\x6b\xff\x16\x90\x93\x5f\x56\xfe\x0d\x44\x92\x66\xfe\x52\xf6\x9f\x22\x57\x3e\x1d\x04\x9d\x55\x61\xc1\xbd\xf6\xfa\xe1\xba\xaa\x68\x8f\xa3\xa2\x67\x28\x69\x07\x7f\x21\xae\x17\xf4\x28\x2c\x77\xf6\x39\xd2\x45\x53\x2f\xa2\x45\x87\x7e\xc1\xa0\xa6\x04\x0f\xcf\x50\x8f\x20\xa9\x5d\x66\x76\x83\x9e\x82\xbf\x82\xd8\x14\xb7\x3c\x23\x7c\x1f\x41\x89\xeb\xc1\x9b\x29\x74\xe2\xdf\xfe\xfa\xef\xd3\x2e\xc5\x6c\x68\xfa\x58\xb8\x94\x6b\xfc\xbb\x3c\x11\x32\x86\x60\x2c\x8f\x9b\xd2\x3a\xb2\xe2\x9d\xaf\x1c\xcf\x66\x19\x3e\x49\xd6\x64\xfb\x55\xab\x17\xb8\x88\xbe\x42\x44\xdf\x1f\xb2\x6e\xfb\x56\xe2\x7d\x5d\xa6\x84\x5e\xc4\x3b\xc9\xb7\x4d\x8d\xf1\x67\x6d\xe3\x2e\x11\x69\x0c\x4d\xa4\xda\xf7\x45\xa5\x62\x11\xdd\x8f\xb5\x09\x0e\x26\x06\x17\xd1\x3d\xfb\xf9\xf6\xc3\x7b\xd4\x81\x4c\xf4\x5c\x5e\xf2\x22\x79\x10\x77\x79\xea\xc2\x01\xc4\xf4\x92\xa7\x76\x8f\x54\x89\x6a\x03\x52\x14\xcb\x6a\x6b\x0d\x87\x90\x47\x7c\xbb\x3b\x59\x96\xd1\xbd\x28\x4e\x73\x2e\x63\xb5\xc5\x6e\x9c\xea\x72\xb5\x4a\xbe\x4c\x0b\x9e\x77\x90\x8a\xa3\x1f\xe1\x1b\xda\xb9\x5e\x2a\xa6\xf0\x36\x2f\x9a\xba\x8f\x90\xf5\x49\x02\xc4\x15\xe3\x16\x9e\xc0\x39\xdf\x0a\x60\x85\x63\x55\x42\x7e\x28\x05\x13\x29\x41\xb7\x4e\x6b\x42\xd0\x2b\x52\xc5\xfd\x14\x18\xf7\x9f\x82\x56\x55\x95\x99\x6d\xa3\xbc\x16\xdc\x96\xdf\xe3\xfb\x70\x9d\x0b\xad\x27\x4c\x2b\x68\xf1\x5c\x5a\x2c\xba\xcd\x97\x02\xdc\x0b\xf0\x4a\xa6\x3b\x16\xa9\x2c\x01\xe9\x3c\xd7\xaf\x8d\x7a\x04\x3f\x7d\x98\x36\x08\x6a\xa7\xa5\x2c\x92\x94\xf1\x55\x41\x4e\x7c\x20\xd1\xb6\xa2\x39\x7a\x3a\x97\x10\x8a\x8d\xa0\xfb\x00\x91\x70\xe1\x17\xd7\x09\xcd\x56\x3c\x4a\xd2\xa4\x20\x6a\x1f\x48\x32\xe2\xa6\xbf\xe6\x3e\x30\x63\x99\xf3\x1d\x4f\xfd\xc3\x8a\xa7\xa5\xcf\x94\x3c\xd1\xa2\x87\x3a\x2e\xd1\x0b\x74\x10\x7c\x0d\xae\xaa\x24\x0c\x3e\x20\xcd\xee\xcc\x54\x7e\x59\xbb\x45\x7f\x17\xfe\x6f\xe5\x1d\xde\x67\x15\x1c\xf1\x20\x3f\xe6\x72\x6c\x3e\xb9\x9d\xd2\xac\xb7\x33\x92\xd8\xe2\x83\x2b\xa6\xb8\x67\x1c\x76\xd7\x23\xc4\x4c\x3a\x1e\xfd\x53\xab\x0f\xd4\xac\x61\xc4\xe8\xb5\x1b\x89\x5f\xc9\x9d\xd1\xc5\x7d\x3c\xa4\xf9\xd6\x1b\xff\x51\xa9\xf4\x58\x8f\x3c\x4f\xed\x81\xbf\x00\xc9\xcc\x63\x9e\x93\xb8\x00\x9c\x63\xeb\xe2\xad\x8b\xb9\x3b\x32\xe1\xaa\xd0\x0e\xc1\xc1\xa8\x09\x70\x90\x41\x23\x7a\x90\xe2\x3a\x6b\x01\x5d\x8c\x44\xbc\x43\x19\x88\xd6\xb2\xa6\x7d\x33\x44\x10\x70\x18\x70\xdf\x46\x20\x5c\xac\xb5\x70\x94\xb3\x0e\x05\x2e\x6b\x55\x39\xc7\x5d\x48\xcc\xea\xc6\x31\xa8\xdb\x8e\x27\xca\x6a\x9b\xfb\x93\xac\xf8\xb9\x0c\x2c\x76\x64\x53\xb2\x29\x05\x6e\xd4\xda\xfc\x79\x95\x65\x78\xb4\x3f\xef\x18\xf6\xed\xde\x93\xf3\x6d\xa8\xa3\x05\x58\x90\x48\x6d\x97\x89\xb4\x29\xea\xe4\xe4\x86\xa7\xc6\xcc\x92\x1b\xba\x80\x84\x7d\x32\xa0\xba\x42\x6d\xec\x9d\x99\x13\xf2\x44\x86\x47\xd6\xbe\xe7\x78\xf8\xbe\x7b\x5a\xa2\xf0\x8e\x48\x63\xbd\x07\xe6\x02\x49\x1f\xf9\x4e\x83\xd6\xac\x30\xa7\xe2\x0a\x1d\xbb\xd5\xf6\x4f\x02\xf3\xc3\x12\x67\x92\x70\x7b\x49\x12\xd4\xd4\x97\x04\x79\x3b\x44\x6a\x55\x75\x3d\x1f\xd2\x0b\xdd\x3e\x38\xdf\x26\x56\x93\xf7\xc6\x6a\x30\x08\xfd\x3f\x23\x3c\xd3\xe3\x04\x3e\xd2\x17\x1d\x5c\x93\x68\x31\x12\x4c\x08\x12\xb7\x5c\x88\x7a\xc2\xb6\x3c\x91\xb4\x0d\x50\xb9\x2c\x16\xcb\x72\xbd\xee\x74\x91\x7e\xff\xb1\x96\xea\x3e\xf9\xbb\xf7\x85\xf7\x32\x7a\x3d\x85\xb7\xf8\xc2\xd6\x84\xee\x6b\xf3\xee\xfb\x3a\x0e\xe2\x6f\xe8\x8d\x6f\x0d\x89\x35\x16\xd1\xd3\x78\xe3\x2f\x86\x78\xe3\x2d\xb6\x0b\x52\xec\xe8\x39\x6d\xf1\x37\xbf\xb9\xe9\xbf\x8e\x9b\x7e\xd0\xa2\x40\x5a\x97\x45\x52\x35\xd0\x7b\x5a\x78\x20\x3b\x9c\xa3\x11\x85\x56\x91\x0a\xbc\x16\x32\xd6\x6c\xc9\xa3\x67\xa0\x8b\x83\xdb\xf1\x78\x7f\xe0\x1e\xf0\xcb\x8d\xda\x0a\x06\x55\x69\xd4\xbc\x60\x94\xc5\x38\x01\xb4\xaa\xe9\xa0\x47\x8c\x10\x1e\x05\xae\x53\x44\xae\xc4\xde\xa8\x7e\x29\xc5\x23\x33\xb7\xd5\x24\x84\xef\x05\xd3\x03\x62\x48\xaf\x8c\x75\x58\xc1\xfa\x3b\xca\x86\x5c\xac\x79\x1e\x43\x86\x09\x6d\xc9\x94\x47\xf7\xe6\xbf\xa1\x7d\x54\x23\x41\x0c\x2d\x2f\x3b\xc2\x5e\x7d\x69\x89\x8c\x72\x20\xe3\x22\x34\xa3\x6f\x1f\xfe\x5c\x33\x1e\xe5\x4a\xa3\xd3\xc8\x69\x88\x42\x86\x33\x18\xb0\x0f\x49\x5c\xf2\x14\x6b\xec\xf4\xb4\x8f\x85\xaf\xd5\x01\x47\x81\xdc\x4f\x13\xcd\x46\xd3\x81\x1c\x45\x30\x8c\xd3\xb9\x7c\xeb\x02\x26\xaf\xd9\x9d\x16\x84\x32\xd3\x96\x30\xb9\xb7\xa5\xcf\x66\x3e\x34\x30\x81\x9d\x36\x44\xcf\x00\x58\x90\x75\x30\x10\xba\x7b\x24\xf6\x90\x0e\x1e\x33\x29\xa3\xc9\x53\x2f\x02\xcd\x61\x3f\x2c\xf8\x4e\xc8\x05\x8f\x77\x21\x35\x5b\x22\x19\x44\xe9\x18\x8f\xb7\x89\x34\x9b\xc0\xea\xda\xb9\x9b\xc6\x52\x5c\x23\xe4\x18\xe4\x5f\xd2\xb4\x76\x08\x6a\x26\x85\x31\x2e\x79\x9e\xa4\x3b\x78\x4f\x64\xb9\x38\x09\xea\x09\xe6\x87\x32\x9e\x80\xac\x9b\x68\x44\x4a\x2d\x56\x65\x8a\xaf\x0e\x78\x97\xbb\x0e\xd0\x89\x74\x77\x31\x31\x06\x47\x41\xa2\x0b\x41\xc5\x28\x65\xf6\x14\xd9\x23\x8d\x68\xe5\xb8\x88\x9b\xa7\x0e\xcc\x01\xe4\xbe\x51\x8f\x36\xd5\xed\x91\x7b\x2c\x73\xd7\xed\xfa\x64\x51\x96\x7e\x3b\xd4\xbe\x00\xed\x39\x85\x83\x1f\x57\x42\x6b\xf4\x99\x88\xdd\xd9\x94\x48\xe8\x0e\xa9\x81\x7a\xcf\x75\xa9\x31\x63\xce\xcc\x25\xdc\x5f\xd6\xd1\x51\x75\x5c\x33\xd7\xbb\x44\x2b\xc9\xe6\xe5\xef\x7f\xff\x07\xc1\x7e\x0f\x29\x84\xf4\x1e\xc1\xf8\x18\x90\x07\x62\xe9\x70\x64\xbb\x0a\x04\x32\x0b\x36\x66\x84\xb5\x41\x54\x6d\xbe\x3e\x80\x3c\x79\xb4\x61\xba\x5c\x22\x82\x91\x53\x88\x85\x4b\xc7\xcd\xfb\x5e\x01\x18\x11\x6f\x76\xdb\xfa\xff\x25\x01\x05\xe4\xc7\x9f\xcb\x4c\x21\x7d\x34\x40\x3f\x97\x82\x6d\x79\x7e\x0f\x72\x87\xa4\x69\xcf\x78\xc1\x5e\x26\x62\x5a\x0d\x2f\xbc\xaa\xb4\x87\x02\x3a\x48\x0b\xcb\xf2\x52\x4a\xab\xdf\xc2\x8c\x61\xea\x7d\xfd\x93\xb9\x5c\x96\xe1\xdb\xb3\x12\x2c\xf0\x4b\x0b\x02\x06\x70\xd8\x2a\xe0\x0a\xa1\x46\x71\x1d\x68\xed\xb3\x01\x51\x83\xb9\x7c\xe2\xb0\xc1\x3e\x87\xdf\x47\xb2\xc1\xac\x33\x2f\xc8\x57\x80\xee\x86\x12\xa3\x30\x1d\xb8\xec\xc1\xc8\xf9\x08\x3a\xa3\x13\xf6\x73\xf2\x20\x26\xec\x26\xe3\xf9\xfd\x84\xbd\xc5\xf0\xdf\x9f\xd4\xb2\xcd\x87\xd7\x20\x94\x38\xda\x8f\x77\x98\x1b\xab\x8f\x68\xa5\xdd\xfa\xff\xb5\x41\x0c\xc0\xba\x62\xdf\xff\x33\x11\x79\x1d\x5c\x1f\x7f\xef\x9e\x88\x3d\x61\xea\xdf\xc0\x6b\x7f\x97\xaf\xe2\x7e\x9a\x8f\xdf\x85\xff\x6b\xcf\x2f\x6b\x71\x81\xed\x49\xa7\x5c\x2b\x2a\xed\xfb\x4a\x6c\x4e\xe2\xfa\xa5\xdc\xcc\x6f\x1e\xb6\x15\x28\x7d\x3c\x76\xa9\xed\x23\x40\xf7\xf4\x53\x3b\x5e\x67\xa9\xd2\x65\xde\xbf\xf9\xaf\xab\xad\xb6\xb5\xb7\x50\x7d\xc2\x62\xdb\x2e\x05\xb0\x16\x0c\x85\x9f\xe0\xd7\x16\xff\xa1\x96\x0b\xc0\x5a\x1d\xb7\xc3\xdb\x8a\xb3\xe2\x41\x2e\x22\x46\x4d\xf5\x37\xe4\x4d\x26\x80\x71\xca\x9b\xa2\x3e\x20\x50\x5b\x61\xce\x35\x32\x97\x96\x80\x1b\x33\x66\xf3\x5c\x00\x53\x70\x2e\x40\x13\x8b\x65\x3c\x77\x80\x07\x6b\x11\x05\x2f\x1f\x0f\x8a\x09\xb3\xdc\x20\x59\x95\xde\x5b\x4b\x21\xa4\x1b\xed\x31\xa6\x84\xb1\x0e\xea\xa3\x4f\x68\xb7\x47\xc1\x22\x4c\xa4\xe8\xd0\xef\x6b\xfc\x2e\x78\x0b\x82\xc9\xbd\x16\x45\x70\x9a\xd7\x4c\x8b\xca\xd6\xac\x44\xa8\xbe\x2b\xc4\x7f\x6b\x0c\xba\x46\xce\x55\x71\xa0\x0c\x8a\xe9\x3d\x85\xbf\xfc\x23\x2f\x36\xf8\xa0\xdd\xaa\x42\xe0\x99\x89\x2c\x41\xb8\x5e\xd0\xeb\xbc\x4c\xd5\x12\xc4\xa8\xcc\x27\x5d\x6f\xc3\x88\xb6\xf6\xa0\xa1\x6b\x4e\xd8\x90\x93\xc1\x9c\x26\x90\x69\x9b\x0b\x0d\x84\x2b\xcd\x28\xd5\x50\x7c\xf2\xb8\x47\x77\xb3\xb9\xe6\xd0\x7f\xdb\x78\x6c\x37\x19\xfa\xcd\xb6\x06\xb0\xea\xf9\x01\x19\x34\xe7\x61\xb6\xaa\x31\x0e\x88\xac\x98\xc2\xc0\xc8\x57\x5a\xeb\xaf\xd5\x3c\x9e\xcb\x19\x7e\x12\x5c\x02\xdc\x2b\xd1\x38\x3c\x28\xc9\x5b\xba\xfd\x87\xe9\xab\x6c\x16\x22\x10\xc9\x43\x30\xf1\xbe\x4c\x78\x0c\x4c\x20\xab\x51\x16\x49\x2e\x98\x04\x14\xc2\x5c\xea\x72\x79\xe2\x89\x49\xcc\x2b\xee\x01\xc8\x74\xb4\xc8\x38\x3c\x65\x80\xaf\xe8\xa4\xe5\x1a\x46\xcf\xa4\x57\x94\xb0\x04\x7e\x3c\xa5\xc3\x1f\x72\x25\x31\x33\xde\xf5\xdd\x95\x63\x1e\x6b\xf0\x8a\xb6\x70\x25\xbc\xec\xfa\xce\x0b\xd0\xbc\x81\x0c\xcc\x6b\x44\x51\x7c\xeb\x0b\x3c\x8c\x86\x0e\xbd\xba\x21\x9e\x36\x97\xff\xd7\xde\x0d\xdd\xa0\xe2\x11\x2b\xdd\x8c\x8c\xb9\xa2\x3a\xc1\xce\x95\xb6\xd9\x27\x64\x60\x04\x76\x37\xaa\xb1\xe4\xdb\x4a\xe5\x16\xd7\x12\x2a\x3c\x28\x4a\x97\x85\x4f\x1f\x12\x1d\xd0\x7d\x43\x6d\x37\x42\xb0\xd7\xb9\x58\xbd\xfe\x94\x8b\xd5\xc2\xce\xf4\x14\x3a\x34\x35\x3d\x6a\x92\x7e\x0f\x5c\x1c\x3a\x53\xb2\x9d\xfc\x70\x0f\x39\x69\xad\x4b\x58\x4e\xd0\xa7\x64\xc5\xbc\x10\xa0\xe9\x0f\x30\x40\x88\xb8\xce\x46\xde\x68\xd9\x57\xbf\xe6\xba\x90\x60\x03\xa0\x56\x1d\x02\x7a\x7f\xff\xd7\x5b\x65\xcc\x86\x5c\x6f\xb7\x55\xc8\x8c\x3d\xec\xb9\x74\x17\x5e\x37\x2e\xf4\xeb\xa2\xd3\x61\x02\x75\xc6\x1f\x25\xf1\xd8\x8c\x72\x3d\x0d\xbb\xd6\xda\xa5\xfc\xcd\xb5\xd6\xc0\xc0\xf9\x5d\x26\xad\xa7\x2f\x71\x6a\x73\x93\x40\xa8\x99\xa7\x69\xa8\xa9\xe0\x23\x6d\x73\xe9\xf3\x52\x8d\xd5\x9a\xa6\xd6\x85\x57\xb1\x37\x88\xb5\x28\x86\x84\x60\x31\xb1\xa4\x2b\x44\x57\x48\xf1\xb0\x93\x25\x07\x15\x50\xa7\x34\xb4\x6f\x37\x3f\xd5\x23\xf2\x3b\xcb\x8b\xde\x13\x79\xc6\x6a\x17\xf7\xa2\x01\x67\xde\xdb\xd6\xf6\x48\x47\x40\x29\x01\x9b\xd9\x9e\xb2\x11\xcf\x73\x8b\xf2\xa7\x5a\x99\x79\x2b\xad\x78\x54\x71\x73\x76\xb4\x73\x23\xa2\xfb\x4c\x25\x72\xf4\x59\x54\xa1\xb8\x80\xc5\x5e\x30\x5f\x9a\x7b\x1d\x0e\xba\x1c\x2b\xf6\x24\x76\x44\x03\xbc\xc2\x42\x43\x3d\x19\x1b\x67\x4e\x65\xb5\x7b\xd9\x3d\xb5\xff\x42\xf8\xbb\xe1\x19\x7c\xb1\x2d\xf1\xa1\xda\xad\xc2\x5b\x1c\x3b\x15\x26\x50\xde\xc8\xfe\x1a\x38\xd8\x9c\x55\x28\x0c\x5b\x87\x14\x5c\x90\xbf\x79\x86\x7e\xf3\x0c\xfd\x0f\xf7\x0c\x7d\x4d\xb7\x10\x60\x63\x9e\xd3\x27\xd4\x13\x20\x3f\x62\x3b\xba\x5a\x8f\x95\x1c\x47\xeb\x78\x12\x48\xe3\x06\x99\x8e\x4d\xa0\xbf\x25\xc2\x28\x40\x58\x3c\xba\x17\xb2\x33\x46\x6f\xe9\x8b\x3a\xf5\x93\x9f\x16\xc1\xd2\xc6\xbe\x14\xfc\xba\x1f\xca\xe2\xa1\x4e\x44\x1a\xdc\x46\x08\x62\xf6\x09\xd8\x9e\xa6\xe3\x27\x00\x1a\x53\xb9\x23\xb6\xd6\x94\x85\x87\xc1\x48\xa4\x49\x42\xb0\x54\x8d\x0a\x7a\x28\x26\xce\x56\xbc\xc8\x94\x4a\x5b\xa1\x71\x4f\x3a\x80\x8d\x44\x99\xa1\x83\x77\x81\xc6\xa8\x0e\x01\x63\x76\x14\x7d\xd2\x85\x4f\xd1\xc0\x7c\x0c\xd0\xc2\x80\xd5\x14\x97\x90\x4b\xe9\x87\x63\xea\x33\x9c\xb8\x73\xb8\x10\x46\x6c\x29\x22\x0e\x3a\x90\x16\xbc\x17\x71\x97\x7d\x12\x92\x22\x35\xd2\x41\x74\xb3\x9e\x8e\xa8\x25\x94\xbb\x48\xda\x84\x2f\xc6\x6e\xae\x9a\x85\x60\xa1\xe5\xd8\x72\x8b\x24\xb1\xb4\x8b\xfb\x64\x3f\x2d\xc7\xf4\x22\x4a\xb9\x1e\x68\x58\xb7\x9e\x3b\x17\x54\xd0\x19\x94\x33\xfc\x20\xfd\x19\xd2\x71\xb6\x03\x91\x3b\x73\x39\x73\xb2\x97\x1e\xfb\xe5\x90\x7b\x18\x2e\x45\xcc\x62\x63\x6a\x90\xcb\xd1\xbf\x5c\x26\x4c\x97\xd1\x06\xd8\x2a\xab\xe7\x54\x78\x6e\x35\x77\xec\x64\x2e\xcd\x83\x08\x5c\x2d\x5b\x0e\x79\xf1\x8f\xc6\x58\xd5\xc9\x7f\x0a\x07\xcf\x22\xf2\xae\x10\x91\x85\x0f\x27\x25\x5b\xd1\x6b\x96\x38\x14\x01\x16\x1e\x53\x52\x66\x31\x2f\xc4\x74\xee\xd1\x36\x09\x7a\x3a\x2d\xca\x83\x4c\x66\x1d\x76\x2c\xc4\x31\xd6\x4e\xda\x34\x59\x89\x68\x17\x35\x74\x80\xfa\x69\x22\x7e\x7b\xb6\x7d\x5f\xcf\x36\x64\xd9\xc5\x9c\xc1\x31\x43\x4b\x4d\xbd\xf6\x3f\x3f\x6e\x70\x05\x0b\x5a\xa2\x47\x8c\xf3\x57\x7c\x76\xb6\xd8\xc0\xe3\xec\xf9\xc1\xef\xa0\xfe\xeb\xcc\x3f\x6c\xfd\x65\x1d\x50\x20\x34\xcc\xc2\x30\xb8\x58\x84\x4b\xc7\x18\xb4\x83\xc3\xfa\xdd\x2c\x33\xdf\x15\x38\x69\xc8\xc3\xd5\x58\xdc\x0e\xae\x74\x69\x2d\x6d\x29\xf0\xbe\xeb\xb1\xb8\x03\x56\x77\x5e\xbc\xd0\x6e\xd4\xab\x27\xa0\xc5\xfe\xbf\x4f\x74\xf1\x4b\x4d\x33\xf4\x30\xd1\xd1\x67\x33\x4d\x6d\x53\xb1\x99\x43\x2d\xaa\xeb\xaa\xcd\xa3\x56\x76\xcd\xc1\xe3\xc9\xaa\xc4\x99\x76\x8f\x79\x07\x7d\x72\xe3\xf5\x09\xaf\xa6\xc7\x9c\x67\x99\xc8\xed\x45\xde\xb0\xb5\x40\x72\x0d\x6a\x01\xcd\xc4\x8d\x40\xe1\xe6\xda\x2b\xd7\x1c\x25\xb5\xa2\xe1\x6b\x30\x74\xd3\xf6\x99\xbb\x2c\xd3\xb4\x73\xe6\xf6\x2b\x39\x5d\xde\xbd\x7f\xbf\xf8\x65\xf6\xfe\xee\xdc\x76\xbf\x55\x19\x29\xf8\x5a\xe7\x98\xb8\x96\xd0\x98\x78\xed\x45\x53\xad\xb0\xe2\xd1\xca\xf7\x1a\x9d\x5c\x65\x9a\x56\x55\xb3\xe6\xf2\x13\x95\x03\x30\x6d\x54\x04\x35\xe3\xc6\x7a\x07\xae\x5a\x3f\x7c\xed\x93\x29\xfc\x13\xfe\xf6\x84\xf9\x4e\xbc\x06\x6d\x47\xd2\x8c\x6b\x1f\x57\xca\x86\x39\x62\x3b\x20\x18\xb8\x6b\x3b\x3c\xb5\x2e\xe0\x61\xdb\xe3\x4e\x02\x23\xb9\x88\xad\x9c\xdf\x93\xec\x0e\x1c\xbb\x4f\xd5\x38\xb5\x3b\xcb\x63\x7c\xd2\x40\xb9\x13\x54\x73\x03\x8d\x6a\x2f\x78\x36\x97\xe8\x03\x35\x6d\x2a\x54\x77\x9b\xd8\x05\x99\xb7\x29\x97\xeb\x92\xaf\x8d\x75\x6b\x2b\x9f\xcb\x6d\xb2\xde\x20\x0f\x48\x99\x79\x7c\x32\x67\x12\xe8\x62\x6a\x4b\xa8\x86\x4f\x4e\xe4\x5c\x52\x9f\xe4\xda\x17\x8f\x58\xd9\x3f\xdd\xb8\xee\x10\x28\x1d\x0b\x22\x41\x3a\x39\x97\x38\xb9\xc8\x4f\x62\x23\x21\xf0\x62\xe1\x45\x7d\xe9\x72\x88\x5d\xa2\x68\xbb\x39\xd3\xd7\x10\x93\x99\x4b\x97\xa2\x8b\x9e\x23\xea\x43\x20\x5c\x82\x4d\xda\x7f\x9e\xd8\xc9\xb0\x7b\x82\xda\xd6\xbe\xea\x8f\xbe\x03\xcc\x86\x5b\x8c\x50\x9f\x6e\x1e\x63\x03\xbd\x85\x3c\x38\x38\xba\x78\x1b\x20\x2f\xbb\xbd\x35\xb6\x5f\xf8\x9d\x4e\x68\xab\x2a\x97\xe9\x88\x26\xe1\xf7\x7b\x1b\x85\x47\x72\x7f\xa3\x06\x3c\x87\xaf\x6b\x5b\xcb\x2c\xd3\xbe\x6a\x97\x4a\x75\xcc\xcb\x13\x06\x14\x2b\x8d\xa2\x1f\xec\x1b\x8c\x32\x2a\x0e\x59\x2f\x03\x92\x15\xeb\x43\x64\x4f\x9f\xbe\x06\xa5\x89\x3e\xa8\x39\xde\x7e\x1a\xdc\x22\x67\x21\xd0\x65\x37\xea\x84\xa5\x7b\xae\x72\xc0\x76\x1c\x93\x14\x4f\xb2\x32\xce\x09\x1e\x2f\x66\xf3\xa0\xc6\xb2\x59\xff\x13\xb7\x88\x26\x7e\xe6\x26\xd0\xc8\xa8\xcc\xb5\x39\x2e\xe9\xbc\xa3\x53\x5b\xe5\x8c\xcf\xa5\x4d\x55\xb3\xc7\xf1\xcc\x82\x73\x73\xf7\x57\x4c\x00\xcd\x90\x4f\x1f\x2c\xd6\x82\x29\x29\xec\x69\x38\x97\x56\xfb\x7b\xc2\xf8\x52\x5b\x49\x6d\x2e\x77\x4e\xe7\x3a\x71\xf4\x45\x5c\x32\x40\x3d\xef\x3f\xf3\x6a\x66\x40\xe5\x9e\xff\x9d\xf9\xbf\xff\xfe\xdd\xff\x0f\x00\x00\xff\xff\x7a\x2f\x54\xf3\xe3\xad\x04\x00") +var _adminSwaggerJson = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xfd\x7b\x73\xeb\xb8\x95\x2f\x0c\xff\x3f\x9f\x02\x67\xcf\xa9\xea\xee\x44\xb6\x3b\xc9\x4c\xde\x94\xa7\x4e\xbd\x8f\xda\xd6\xde\xad\xd3\xbe\xc5\x97\xee\xe9\x67\x94\x52\x43\x24\x24\x21\x26\x01\x06\x00\xed\xad\x4e\xe5\xbb\x3f\x85\x85\x0b\x41\x8a\x94\xa8\x8b\x6d\x79\x37\x67\xaa\xd2\xde\x22\x89\xeb\xc2\xc2\xba\xfe\xd6\x3f\xff\x0d\xa1\x0f\xf2\x19\xcf\x66\x44\x7c\x38\x45\x1f\xfe\x78\xfc\xed\x87\x9e\xfe\x8d\xb2\x29\xff\x70\x8a\xf4\x73\x84\x3e\x28\xaa\x12\xa2\x9f\x4f\x93\x85\x22\x34\x4e\x4e\x24\x11\x4f\x34\x22\x27\x38\x4e\x29\x3b\xce\x04\x57\x1c\x3e\x44\xe8\xc3\x13\x11\x92\x72\xa6\x5f\xb7\x7f\x22\xc6\x15\x92\x44\x7d\xf8\x37\x84\xfe\x05\xcd\xcb\x68\x4e\x52\x22\x3f\x9c\xa2\xff\x31\x1f\xcd\x95\xca\x5c\x03\xfa\x6f\xa9\xdf\xfd\x1b\xbc\x1b\x71\x26\xf3\xd2\xcb\x38\xcb\x12\x1a\x61\x45\x39\x3b\xf9\xbb\xe4\xac\x78\x37\x13\x3c\xce\xa3\x96\xef\x62\x35\x97\xc5\x1c\x4f\x70\x46\x4f\x9e\xfe\x70\x82\x23\x45\x9f\xc8\x38\xc1\x39\x8b\xe6\xe3\x2c\xc1\x4c\x9e\xfc\x93\xc6\x7a\x8e\x7f\x27\x91\xfa\x17\xfc\x23\xe6\x29\xa6\xcc\xfc\xcd\x70\x4a\xfe\xe5\xdb\x41\xe8\xc3\x8c\xa8\xe0\x9f\x7a\xb6\x79\x9a\x62\xb1\xd0\x2b\xf2\x91\xa8\x68\x8e\xd4\x9c\x20\xd3\x0f\x72\x4b\xc4\xa7\x08\xa3\x53\x41\xa6\xa7\xbf\x08\x32\x1d\xbb\x85\x3e\x36\x0b\x7c\x01\xa3\xb9\x49\x30\xfb\xe5\xd8\x2e\x13\xb4\xcc\x33\x22\x60\x6e\xc3\x58\xb7\xfe\x89\xa8\x3e\x34\x5b\xbc\x1f\xbe\x2d\x88\xcc\x38\x93\x44\x96\x86\x87\xd0\x87\x3f\x7e\xfb\x6d\xe5\x27\x84\x3e\xc4\x44\x46\x82\x66\xca\xee\x65\x1f\xc9\x3c\x8a\x88\x94\xd3\x3c\x41\xae\xa5\x70\x30\x66\xaa\x7a\x63\xf1\x52\x63\x08\x7d\xf8\xdf\x82\x4c\x75\x3b\xff\x7e\x12\x93\x29\x65\x54\xb7\x2b\x0d\xfd\x04\xa3\x2d\x7d\xf5\xaf\x7f\xab\xfb\xfb\x5f\xc1\x8c\x32\x2c\x70\x4a\x14\x11\xc5\x8e\x9b\xff\xab\xcc\x45\xef\x91\xee\xbc\xd8\xc7\xea\xc0\x2b\xb3\xbd\xc2\x29\xd1\x7b\xa2\x77\xca\x7e\x01\x7f\x0b\x22\x79\x2e\x22\x82\x26\x24\xe1\x6c\x26\x91\xe2\x4b\x6b\x40\xa1\x05\x4d\x5e\xd5\x27\x82\xfc\x23\xa7\x82\xe8\xbd\x52\x22\x27\x95\xa7\x6a\x91\xc1\x20\xa5\x12\x94\xcd\xc2\xa5\xf8\x57\xaf\xd5\xd4\x0c\x55\x6e\x30\x33\xf3\x41\xe3\xc4\x46\xac\xef\x5e\x89\x30\x43\x13\x82\xf4\x59\xa4\x31\x11\x24\x46\x58\x22\x8c\x64\x3e\x91\x44\xa1\x67\xaa\xe6\x94\xe9\x7f\x67\x24\xa2\x53\x1a\xb9\x35\x3b\x9c\xb5\x81\x3f\x57\xaf\xcc\x83\x24\x42\x0f\xfc\x89\xc6\x24\x46\x4f\x38\xc9\x09\x9a\x72\x51\x5a\x9e\xe3\x11\xbb\x9f\xeb\x75\x48\x27\x94\xc1\xc9\xd3\x6b\xe9\x28\xe4\xf7\x6e\xb9\x7e\x8f\x74\x7f\x28\x67\xf4\x1f\x39\x49\x16\x88\xc6\x84\x29\x3a\xa5\x44\x56\x5b\xfb\x3d\x87\xfe\x71\x82\x8e\x90\x5e\x67\x22\x14\xac\x37\x67\x8a\x7c\x56\x12\x1d\xa1\x84\x3e\x12\xf4\xd5\x05\x95\x0a\xf5\x6f\x86\x5f\xf5\xd0\x57\xe6\xbc\x20\xe0\x4d\x5f\xbd\xc2\x0a\xfb\xbf\xff\x16\x1c\x3d\x85\x67\xd5\x43\xf7\xa1\xaf\x4f\xf3\x9d\xb9\x1a\x8a\x16\xfe\xf6\x6f\x61\x3b\x76\xbf\x56\xf3\xdb\x82\xd9\x5a\x4e\xdb\x96\xbf\xc2\x32\x95\x59\xab\xd4\x3b\xb4\x2b\x67\xd5\xed\x56\x59\xab\x7c\x5f\xbc\x55\x4f\xe1\xa5\xf9\xeb\x2e\xcc\x15\x2b\xa0\x7a\x4c\x99\x39\x24\xfe\xcc\x08\xa9\xcf\x89\xa3\xde\x03\x61\x29\xbb\xf0\xda\x60\x66\x01\xbb\x75\x5c\x34\x58\x95\x03\x9c\x77\x42\x53\xba\x6e\x7f\x87\x2c\xd6\x22\x97\x65\x76\x2c\x4f\x27\x44\xe8\x65\x70\x6c\x0f\x66\x3b\xd1\x6c\x50\xe5\x82\x91\xb8\xc5\x34\xff\x91\x13\xb1\x58\x31\xcf\x29\x4e\x64\xd3\x44\x29\x53\x44\xcb\xb7\x95\xc7\x53\x2e\x52\xac\xec\x0b\x7f\xfe\x8f\x4d\x17\x42\xf1\x47\xb2\x6e\xff\x87\x66\x37\x23\x2c\x81\x0c\xd2\x3c\x51\x34\x4b\x08\xca\xf0\x8c\x48\xbb\x22\x79\xa2\x64\x0f\x5e\xd3\x32\x35\x11\x47\xfe\x06\x82\x1e\xdc\xcd\x9b\x4b\xf8\x05\x4d\xbd\x00\xc9\xc8\x67\x05\x2d\x8d\x18\xdc\xbd\xb0\x44\xe1\x8d\xf2\x02\x4b\xb9\x1d\xcd\x48\x2e\xd4\x78\xb2\x38\x7e\x24\x4b\xfd\x36\x52\x0e\x66\x08\x2b\x25\xe8\x24\x57\x44\xcf\x5b\xb7\xe1\xee\x4e\x60\x8f\xe6\x82\x6e\xc3\x1a\xde\x6e\xc2\x31\x15\x24\x82\xb9\x6d\x72\x60\xfc\x57\x7a\xde\x5a\x7f\x59\x98\xd9\x3f\x92\x05\xc8\x23\x35\x2b\xe0\xb7\x7c\xc4\x46\x0c\x1d\xa1\xf3\xc1\xdd\xd9\xe0\xea\x7c\x78\xf5\xe9\x14\x7d\xb7\x40\x31\x99\xe2\x3c\x51\x3d\x34\xa5\x24\x89\x25\xc2\x82\x40\x93\x24\xd6\x32\x87\x1e\x0c\x61\x31\x65\x33\xc4\x45\x4c\xc4\xcb\x2d\x63\xe5\x29\x61\x79\x5a\xb9\x57\xe0\xf7\x62\xf4\x95\x2f\xb4\x88\xe1\x1f\x95\x9e\xfc\x6d\x69\x81\x61\xc6\xba\xef\xa0\xb5\x57\x13\x6a\xa2\x39\x4d\x62\x41\xd8\x89\xc2\xf2\x71\x4c\x3e\x93\x28\x37\x77\xf2\x3f\xcb\x3f\x8c\xb5\x64\xca\x63\x52\xfe\xa5\xf4\x8f\x42\x14\xda\xf8\x53\xaf\xa5\x6e\xfc\x25\xe8\xb4\xed\xbe\x83\x5f\x68\x5c\xfb\x36\xfc\xb2\x66\x0e\xee\x9d\x15\x83\x75\xaf\x34\x8e\xca\xbd\x60\x25\xbe\xda\x77\x04\x51\x62\x31\xc6\x4a\x91\x34\x53\x1b\xea\xeb\x18\x25\x5a\xae\x5c\x25\x47\x5e\xf1\x98\x0c\x5c\x7f\xbf\x20\x23\xce\x92\x18\x4d\x16\x96\x6b\x4d\x89\x20\x2c\x22\xcd\x2d\xdc\x63\xf9\x58\xb4\xb0\x4e\x18\x2d\xf5\x27\x3f\x72\xa1\x3f\x7f\x0f\x02\x69\x69\xe0\xaf\x21\x93\x6e\x7b\xe2\xbe\x38\x0b\xc1\x96\xfc\xa3\xb3\x27\xec\xbe\x92\x6d\xad\x0f\x5c\x20\xb9\x90\x8a\xa4\x6b\xed\x10\xef\x67\x21\xec\x05\x71\xa8\x03\xae\xdc\x51\xbf\x81\x53\x5f\xbe\x71\xbb\xe3\xbd\xc1\x92\xed\xcb\x8a\x78\xe8\xf3\x74\x3e\x9c\xd5\x53\xbd\x73\xdb\x17\x38\x31\xde\xc5\x34\x4b\xb2\xe0\xbe\x07\xf9\x42\xe6\x86\xc6\xbd\x72\xab\x3d\x86\x01\xac\x51\x34\xcb\x76\x68\x7f\xfe\xf4\xa7\xa1\x85\xc6\x98\xe3\xd4\x9c\xca\xc0\x58\x85\x22\x2e\x8c\x2c\x18\xdb\xf3\x6e\x74\xcd\xfe\x7d\xff\x6e\x70\x7f\x8a\xfa\x28\xc6\x0a\xeb\x03\x2e\x48\x26\x88\x24\x4c\x81\x1e\xaf\xbf\x57\x0b\x94\xf2\x98\x24\x46\xe3\xfc\xa8\x25\x5f\x74\x8e\x15\x3e\xc3\x0a\x27\x7c\x76\x8c\xfa\xf0\x4f\xfd\x31\x95\x08\x27\x92\x23\xec\xc8\x8a\xc4\xae\x09\xcc\x62\xc7\x5a\x30\x8a\x78\x9a\xd1\xc4\xdb\xe0\xbd\x71\x85\xb2\x98\x3e\xd1\x38\xc7\x09\xe2\x13\xcd\x55\xb4\x86\x3c\x78\x22\x4c\xe5\x38\x49\x16\x08\x27\x09\xb2\xdd\xba\x17\x90\x9c\xf3\x3c\x89\x75\xbb\x6e\x94\x92\xa6\x34\xc1\x42\xab\xe0\x66\xb4\xd7\xb6\x2d\x74\x3f\x27\x7e\xac\x30\x2e\xbd\x9a\x29\x7e\x24\x12\x51\x85\x32\x2e\x25\x9d\x24\xc5\x99\x7f\x18\x22\x18\xf7\xd9\xc5\x10\xf4\xf9\x48\x21\x6e\x78\xa8\xeb\xdc\xda\x6f\x5c\x8f\x29\x66\x8c\x40\xc7\x5c\xcd\x89\xb0\xdd\xdb\x97\xdf\x5a\x35\x7f\xb8\xba\xbb\x19\x9c\x0d\x3f\x0e\x07\xe7\xcb\xba\xf9\x7d\xff\xee\x87\xe5\x5f\x7f\xba\xbe\xfd\xe1\xe3\xc5\xf5\x4f\xcb\x4f\x2e\xfa\x0f\x57\x67\xdf\x8f\x6f\x2e\xfa\x57\xcb\x0f\x2d\x59\xb5\x56\xf3\xc3\x91\x6d\x78\xb6\x3a\x9b\xe6\x4b\xd9\x34\x7b\x5f\xae\x51\x73\x4a\x13\xd0\x41\x5b\x1b\x34\xbd\x0d\xc1\x7e\x89\x32\x2c\xa5\x91\x8c\xcc\x08\x8e\x47\xec\x92\x0b\xcd\xc0\xa6\x5c\xf3\x08\x2d\x3d\x29\x91\x47\x8a\xb2\x99\xff\xe8\x14\x8d\xf2\x6f\xbf\xfd\x53\x74\x41\xd9\x23\xfc\x45\x0e\x71\x71\x3a\x8b\x6f\x67\xf1\xfd\x6d\x59\x7c\xb5\xe8\x73\x12\x1a\x7a\xf7\x1b\x32\xa4\x85\x0b\x96\xe5\x0a\x44\x09\x9e\x2b\xfd\xa7\xee\x12\xc8\x63\x45\xe0\x50\x3b\x83\xe2\x27\xa2\xfc\x8b\x5a\xb4\x79\x0f\x76\xc4\x9f\xb8\x78\x9c\x26\xfc\xd9\x0f\xfc\x13\x51\x7a\xec\xb7\xb6\x97\x2e\x94\xa8\x0b\x25\x7a\xdb\x50\xa2\x83\x32\xe6\xbd\x3c\xf3\x2b\x5b\xfe\x0c\x07\x6c\xf0\x64\x35\x3a\xaa\x1a\xfc\x50\x81\x9b\xe9\x55\xb8\x66\xd9\x99\xb3\x86\x73\x96\x5e\x7e\x2f\xdc\xb3\x34\xe8\xd7\xe7\x9c\xbf\x09\x7f\x4b\xe7\x4e\xd9\x72\xa1\xde\x25\x83\x6d\x79\x77\xbc\x9a\x33\xe4\xe5\x19\xfe\x52\x6c\xc3\x26\xc1\x0c\x1b\x44\x2f\xb4\x0e\x57\x58\x13\x9f\x50\x1b\x90\x50\x17\x81\xb0\x1c\x72\x50\x1b\x63\xb0\x5b\x50\xc1\xb6\x77\x53\xfb\x30\x81\x4f\x44\x95\x5e\x7e\x2f\x77\x53\x69\xd0\xaf\x7f\x37\xfd\x46\xa3\x03\xba\x70\x80\x17\x5c\xba\x2f\xfd\x46\x3b\x5c\x87\xff\x6f\xc0\xc3\xdf\xb9\xf4\x37\x5a\xa3\x2f\xcb\x87\xff\xa5\x3a\xed\xdf\xa7\x97\xbe\x73\xcb\x77\x6e\xf9\xb7\xf0\x9f\xbc\x3f\xb7\xfc\xcb\xaa\xa7\xc5\xf1\x1a\x3b\x5a\xb0\xfa\x5a\x70\x28\xff\xd5\xc2\x49\x03\x7f\x39\x95\x6f\xd3\xa0\xf1\x46\x1d\xee\xbc\x18\xdf\x00\x8e\xd0\x2f\x96\x90\xd6\xa8\x73\x4b\xdf\xbd\x07\x75\x6e\x79\xd0\x2f\xaf\xc3\xbd\x19\xf3\x7d\xa1\xcb\xf3\x9d\xb0\x81\xcd\x6f\xcb\x2f\x58\x26\xef\x64\xf1\x97\xcf\xc6\x3f\x98\x09\xbd\x1f\xd9\xfb\x0d\x2e\xde\x96\xb7\xee\xde\x73\xb2\x6a\xae\xd9\xe0\x76\x5a\x97\x61\x55\xfd\x9a\x12\xf9\xc7\x77\x79\xdf\xbe\x46\x92\x55\x77\xe1\x76\x17\xae\x6d\xaa\xbb\x70\xbf\xe0\x0b\xf7\xe0\xe0\x6f\x0e\x26\x02\xb4\x0b\x22\xef\x80\x31\xba\x18\xf2\x3d\x2e\x4e\x17\x43\xde\xc5\x90\xff\xc6\x62\xc8\x77\xd1\x9e\xb6\xc5\xa2\x7c\x0b\x3d\xaa\x53\xa3\x3a\x35\x2a\xfc\xbd\x53\xa3\x3a\x35\xaa\x53\xa3\xbe\x70\x14\xd1\x4e\x87\x6a\xbf\x10\x9d\x0e\xd5\x7a\xa9\x3a\x1d\x6a\xc5\xe2\x74\x3a\x54\xa7\x43\xfd\xb6\x74\x28\xf2\x44\x98\x92\x90\x8c\x16\x6a\x14\x1f\x32\x2e\x9b\x35\xa1\x90\x3b\xd4\x68\x41\xd0\x66\x39\x29\x0c\x02\x97\x7e\x41\x73\x2c\x11\x8f\xa2\x5c\x54\xce\x40\x55\x0f\x3a\x13\x04\x2b\x02\x2d\xe8\x0f\xdf\x83\xfe\xb3\x3c\xdd\xd7\x8a\xc1\x9f\xf0\x78\x89\xda\xcd\x41\xa8\x7b\xb2\x5a\x1e\xd9\xdb\xd4\xff\x91\x93\x76\xea\xdf\x0b\x12\xb5\xc2\xf2\x71\xcf\x44\x5d\xca\xb5\xd8\x8a\xa8\xa1\x85\xf7\x42\xd4\xcb\xd3\xfd\xcd\x10\x75\xdd\xd4\x0f\x81\xa8\x9f\x6d\x1e\xff\x9e\x09\x7b\x09\x1e\x60\x2b\xe2\xf6\xad\xbc\x17\x02\xaf\x9f\xf6\x6f\x86\xc8\x9b\xa6\xff\xb6\x84\xee\x53\x24\x5b\x93\xf8\xbd\xa0\xb3\x99\x56\x33\x40\xc3\xd3\xa4\xb8\xbe\x46\x50\x91\x14\xb8\x96\xac\xfd\xab\xef\x81\xa4\xfd\x60\xcd\xd8\x7f\x33\xb4\xbc\x34\xef\x03\x21\xe2\x13\x41\x22\xfe\x04\xf5\xc2\xda\x11\xf3\x2d\x01\x0a\x06\x7e\x9d\x09\xf2\x44\x79\x2e\x93\xc5\x91\xc8\x19\x72\xcc\x1f\xf9\xe6\x8d\xb5\xfa\x99\x26\x09\xe2\x4c\xeb\x5f\x0a\x0b\xe5\x1e\x6b\xfd\x5b\xf0\x14\x4e\x45\x82\xa5\x42\x8f\x8c\x3f\x33\x34\xc5\x34\xc9\x05\x41\x19\xa7\x4c\x1d\x8f\xd8\x90\xa1\x5b\x33\x46\xc8\x1b\xe8\xa1\x5c\xea\xb3\x14\x61\xc6\xb8\x42\xd1\x1c\xb3\x19\x41\x98\x2d\x6c\x02\x6e\x41\x19\x88\x0b\x94\x67\x31\xd6\x8a\xef\x9c\x54\xa3\xf4\xfc\x18\xc1\x7c\x47\x25\xa2\x12\x91\xcf\x4a\x90\x94\x24\x0b\xdd\x87\xa6\x7d\xc5\x91\x5d\x1f\x33\x54\x9b\xce\x47\x84\xe0\x42\x42\xc6\xc1\x64\xf1\x2b\x66\x8a\x32\x82\x40\x51\x92\xc6\x34\x77\x84\x2e\xb8\x04\xb3\xcd\x0f\x7f\x91\x28\x4a\x72\xa9\x88\xe8\xa1\x49\x3e\x93\x5a\x53\xcc\x12\xac\xa6\x5c\xa4\x7a\x84\x94\x49\x85\x27\x34\xa1\x6a\xd1\x43\x29\x8e\xe6\xa6\x2d\x58\x03\xd9\x1b\xb1\x98\x3f\x33\xa9\x04\xc1\xbe\x77\xf7\x10\x7d\x1d\x3e\x33\x04\x20\xbf\xe9\x41\xda\x21\x4d\xb5\xba\x1b\x0c\xbf\xd8\x71\xb3\x27\xba\x11\x12\xa3\x09\x89\x70\x2e\xad\x87\x41\x89\x05\x22\x9f\xe7\x38\x97\xb0\x77\x7a\x7a\x36\x67\x23\xe2\x69\x96\x10\x45\x10\x9d\x22\x25\x28\x89\x11\x9e\x61\xaa\x97\xee\x8e\xac\x00\x41\xf7\x44\x6f\x37\xd0\x52\xfd\x2f\xa0\x7e\xa7\x5c\x10\x14\x13\x85\x69\xb2\xd2\xeb\x64\xbf\xed\xb8\xdc\x7b\xe2\x72\xe5\x0d\x3f\x08\x36\x67\x40\xfc\xf7\x70\x69\x33\x6b\xba\x8f\x70\xb2\xe3\xfd\x7d\x6b\x07\xd5\xd1\xf6\xfb\xa2\x6d\xb3\x6b\x87\x43\xdc\xaf\x16\x83\xdd\xbe\xa2\x45\x51\xcd\xe2\x5d\xd1\xf4\x6b\x84\x05\x74\x0e\xe7\xce\xe1\xdc\xb8\x32\xef\xd3\xe1\x7c\x30\x1e\xa3\xce\xe7\xfc\x42\x3e\x67\x2a\x3b\xa7\x73\xe7\x74\x6e\xbb\x40\x9d\xd3\xb9\x73\x3a\xbf\x5f\xa7\xf3\x4b\xe2\x3e\xef\x15\xdd\xf9\x5d\x89\xd6\x9d\x58\xdd\x89\xd5\x1d\x84\xb3\x9f\xda\xbe\x58\x98\xfb\xfa\x43\x4c\x12\xa2\x48\xb3\x3d\x8b\x88\x54\x6b\x0b\xe6\x7a\xa6\x4c\xcb\x71\x33\x41\xa4\xdc\x95\x21\xf9\x86\xdf\x27\x5b\xf2\xc3\xef\xa0\xe6\x3b\x3e\xd5\xf1\xa9\x6d\xa6\x76\x38\xa6\xd9\xe0\x30\xbf\x96\x6d\xd6\xf3\xdf\x2c\x6f\x96\xfe\x1e\x8c\x1b\xb2\xf0\x8b\x1a\x0a\xd7\x52\xbb\xe2\xfe\x70\x5b\x3a\xdf\x91\x1f\x9b\xbe\xde\x27\x33\x36\x63\xef\x38\x71\xc7\x89\x3b\x4e\xfc\xbe\x39\xb1\x3b\xc9\x6f\xea\x22\x33\x7e\xba\x71\x96\x60\x36\xa6\xb1\x3c\xf9\x67\xa1\xcb\xbf\x94\x87\x4c\x1f\xa8\xd8\xa4\x98\xfa\x94\x4e\xf1\x8b\xfe\x24\x29\x0c\xe6\x1e\x33\x73\x8d\x13\xcd\xd8\xd8\x6f\x12\xcc\x86\xf1\xbb\xf0\xa3\xd5\xce\xfe\x35\x7c\x6a\xbb\xf0\x71\xac\xc0\xd3\x81\x29\x33\x26\xbc\x22\xaf\xb6\x64\xa0\x3c\x8c\x13\xbe\x0b\x57\x0f\x26\x16\x30\x76\xc7\xaf\x83\x45\x39\xbc\x69\x77\x7e\x9d\x2e\x97\xb0\xf3\x5c\xb4\x9c\x70\xe7\xb9\x38\x5c\xcf\xc5\x5b\xb9\x23\x5f\xf9\x78\xbe\x96\x58\xd7\x3e\x08\xdf\x44\xab\x41\x50\x6b\x9e\x25\x1c\xc7\xab\x5c\x31\x85\xe0\x15\x82\xa3\xac\x8d\xc4\x2f\x3e\x7b\x0f\xc2\x5a\x31\xda\xdf\x58\x24\xdf\xf2\xc4\x0f\x45\x4b\x79\xc5\x50\xbe\x7a\x12\xdf\x40\x25\x79\x1f\xf8\xa9\xc5\x78\xbb\xd0\xbe\xce\xa2\xf4\xf6\x16\xa5\x2e\xb4\xaf\x53\x01\x0f\x4c\x05\xec\x42\xfb\xba\xd0\xbe\x4e\x41\x5e\x3d\xed\x4e\x41\xfe\x22\x42\xfb\x5a\x89\xda\x2f\x88\xbd\xb9\xbb\xd0\xdd\xc9\xdc\xee\xbd\x4e\xe6\xee\x64\xee\x2f\x54\xe6\x3e\x8c\x15\xee\x04\xee\x4e\xe0\xee\x04\xee\x4e\xe0\xee\x04\xee\xbd\x2f\x63\x27\x70\xbf\x66\x81\xce\x7a\xa9\x7b\x4d\x92\xcd\x7b\xf5\xe5\x74\xe2\x76\x27\x6e\x1f\xb6\xb8\x7d\x30\x13\x7a\x3f\x65\x1e\xdb\xcd\xa7\x2b\x52\xde\x15\x29\xef\x8a\x94\xbf\x78\x91\x72\xf7\x75\x8b\x8c\x0f\x7b\xb8\x14\x56\xb9\x34\x80\x8f\x82\xcc\xa8\x54\xc0\xfe\xdb\xc8\x2b\xeb\x13\x3d\xde\xab\x9c\xd2\xa5\x7a\xf8\xa7\x9d\xd4\xd2\x49\x2d\xbf\x51\xa9\xe5\x80\x62\xc1\x0e\x22\x63\x25\xc5\x2a\x9a\xe3\x49\x42\xc6\xde\xe8\x23\xdb\xea\xc1\x17\x54\x2a\x89\xa2\x5c\x2a\x9e\x36\x5f\x2e\x97\xae\x87\xbe\xef\xe0\x8c\xb3\x29\x9d\xe5\xe6\x6e\x31\xe0\x9c\xc1\x89\x2e\x24\xc1\x45\x46\xd6\x79\xaa\x6a\x5a\x7f\x17\xd7\x52\xfd\xd0\x5f\xeb\x76\xda\x44\x70\x2f\x8c\x7c\x56\xea\xd6\xb2\xd6\xf8\x76\x70\x77\xfd\x70\x7b\x36\x38\x45\xfd\x2c\x4b\xa8\xb1\xbb\x1b\x52\xa0\xbf\xea\x49\x21\x85\xe5\x63\xb1\x97\xc2\x90\xb9\xc1\xb0\x05\x43\xbf\x96\x8d\xd1\x11\x3a\xbb\x78\xb8\xbb\x1f\xdc\x36\x34\x68\x09\x05\xf2\x56\x49\x9a\x25\x58\x91\x18\x3d\xe6\x13\x22\x18\xd1\xd2\x8e\x45\xba\x2d\xcc\xff\xa6\xd1\xc1\x7f\x0f\xce\x1e\xee\x87\xd7\x57\xe3\xbf\x3e\x0c\x1e\x06\xa7\xc8\x51\x9c\x6e\x56\x8f\x4b\x8f\x22\x5e\x30\x9c\x6a\x0d\x44\xff\x50\x64\xca\xfe\x23\x27\x39\x41\x58\x4a\x3a\x63\x29\x01\x44\xe0\x52\x8b\x6e\xc0\x17\xfd\xef\x06\x17\xe5\x96\xe7\x24\x84\xdf\x45\x09\x9e\x90\xc4\xfa\x23\xc0\xc4\xae\x09\x3d\x80\x2a\x36\x8e\x8a\xdc\xac\xea\x5f\x1f\xfa\x17\xc3\xfb\x9f\xc7\xd7\x1f\xc7\x77\x83\xdb\x1f\x87\x67\x83\xb1\x95\x2a\xcf\xfa\xba\xdf\x52\x4f\x56\xf8\x44\xff\xc8\x71\xa2\xb5\x13\x3e\x75\x78\xbc\xe8\x79\x4e\x18\xca\x19\x50\x9c\x51\x79\xb4\x1e\xe4\x3b\xd5\xa7\xcc\xcc\xe8\xe6\xe2\xe1\xd3\xf0\x6a\x7c\xfd\xe3\xe0\xf6\x76\x78\x3e\x38\x45\x77\x24\x01\xa5\xc0\x2d\x3a\xec\x62\x96\xe4\x33\xca\x10\x4d\xb3\x84\xe8\xd5\xc0\x36\x9b\x78\x8e\x9f\x28\x17\xf6\xe8\xce\xe8\x13\x61\x66\x1d\xe1\xcc\x42\xfb\x4e\xf8\x1e\x07\x4b\x77\x7d\xf5\x71\xf8\xe9\x14\xf5\xe3\xd8\xcf\x41\x42\x1b\x25\xca\x71\xb0\xce\x47\xe5\x61\x6b\xe6\x00\xdd\x1b\x22\xe2\x4f\x44\x08\x1a\x93\x0a\x1d\xf5\xef\xee\x86\x9f\xae\x2e\x07\x57\xf7\xb0\x62\x4a\xf0\x44\xa2\x39\x7f\x06\x53\x36\xcc\x10\x2c\xdc\x4f\x98\x26\xd0\x99\xdb\x2c\xce\xd0\xf3\x9c\x82\xfb\x03\x80\x99\x7d\xcf\x46\x3f\x13\x39\x7b\x73\xeb\x6c\xe9\xe0\x2d\xab\x2d\xd5\x93\xb4\xfc\x46\xe5\x58\xac\x7a\xa1\x44\xe5\xcb\x2f\xae\xa3\xd6\xe5\x2f\x2a\xe4\xd6\xac\xac\x2d\xd1\x4b\xf3\x4c\x8b\xbd\x6e\xad\xab\x95\xd7\xf0\xf5\xae\x59\xa2\x04\x8d\xe4\xcb\x42\x3d\x89\x9c\x29\x9a\x12\x64\x3b\xb3\x87\x73\x8f\xf0\x4f\x97\xa6\xe1\xf7\x70\xc1\x2e\x95\x72\xf8\x44\x94\x1d\x7e\xa7\x02\x76\x2a\xe0\x61\xa8\x80\xef\x2d\xdb\x3f\x26\xd9\x72\x87\x95\x89\xc1\x3b\xc6\xeb\xb5\x14\xa4\x61\xec\x89\xd6\xa2\x9a\x90\x27\x92\x80\x94\xa7\x04\xd6\x4a\xa3\x95\x5d\x26\x82\xe0\x47\x2d\xf0\xc5\xfc\x39\x94\x5c\x6a\x90\xfb\xd1\x7e\x6e\xe1\x36\x41\x1c\x7f\xfa\xe3\xeb\x5d\x16\x7a\xb9\xe3\xd7\x28\xe1\x7d\x0b\x41\x32\x2b\x31\x02\x83\x04\xfb\x5f\xac\x25\x78\xcd\x6d\x11\x7c\xf1\x1e\x2e\x8a\x70\xb8\x07\xa4\x75\xdd\x86\x4a\xb0\x63\xa1\x29\x51\x38\xc6\x0a\xeb\x43\x33\x23\xea\x18\x5d\x33\x78\x76\x8f\xe5\x63\x0f\xb9\x2b\x4f\xb3\x95\xc2\xca\xf0\x0a\xa9\xf5\xef\xc4\x80\xbf\x39\x1f\xef\xae\xef\xee\xfa\xae\x5f\x99\x2e\xcc\xb3\x61\x85\xf7\x75\x31\x6e\xe4\xf3\xda\xdf\xfd\x65\x5a\x7c\xbf\x57\xd8\xeb\x3a\xb9\xf6\x7a\xa1\x99\xca\x59\xdd\x6d\x65\xfe\xaf\xbb\xad\xba\xdb\xaa\xbb\xad\x0e\x60\x85\xdf\xdc\x61\x58\xc3\xdd\xdf\xd4\x63\xb8\x4e\x3b\xdd\x1a\xf2\xae\xd0\x46\x37\x01\xbd\xfb\xa5\x2d\xb6\x5d\xf1\x0d\x7d\x1f\x3e\xc2\x60\x92\xaf\x91\xd6\xb6\xd7\xcb\xdc\xe4\x8b\x74\xfa\xe9\x0b\xde\xf8\x1d\x02\xe1\x5e\x11\x08\x0f\x63\xae\x2f\x92\x02\xf7\x36\x16\xd3\xb7\x4f\x7b\xeb\xa0\x06\xbb\xc4\xae\x2e\xb1\x0b\x7e\xef\xa0\x06\xf7\x47\xad\x2f\x2b\x5d\xf3\x98\x8c\x2b\x51\x02\xfe\x9f\xe3\xaa\xe7\xa7\xf4\x24\x74\x03\x95\x1e\x14\x99\x6e\xd0\x3a\x8d\xf7\x59\x44\xea\x8a\xc7\xa4\x75\x24\x41\xe9\xe5\x03\x97\xc1\xdd\x3c\x8d\x2c\x5e\x1a\xf8\x0b\x4b\xe2\x0d\x5b\xfe\x25\x1a\x76\x6a\x08\xb8\xb3\xf2\xac\x5d\xa8\x2f\x35\xbe\xa0\xe0\x50\xef\xc8\x53\xd1\x8e\x8d\xbb\x98\xc6\x71\x03\x33\xaf\x7f\xee\x59\x7a\xfd\xe3\x97\xc1\x0c\x6a\xcf\xd1\xc1\xac\x12\xbe\xfd\x3e\xec\x2a\xe1\x88\x5f\xc3\xb2\xb2\x72\xef\xbf\x38\xae\xbe\x8a\x92\x3b\xde\xde\x72\xb9\xbe\x54\x0e\xdf\x41\xfc\xac\xb2\x75\x74\x18\x3a\x9d\xa9\xe5\x70\x26\xdc\x99\x5a\xde\xb5\xa9\xc5\xb8\x68\xc7\x19\x16\x84\xa9\x1a\x91\xba\x7a\x9d\xc0\xeb\x21\xe6\x82\x93\x3a\xa0\x01\xa4\x25\x5a\x64\x2f\x64\x7f\x55\x7d\x59\xb6\x17\x2b\x18\x04\x99\x90\x27\xff\x2c\xfe\xf6\xc2\x7a\xa9\x02\xc4\x8a\xe8\x24\x83\xf5\x2f\xf5\x1d\x9d\xdb\x40\xa5\xdd\x73\x25\xb1\x2a\x89\x82\x10\x44\xbd\x36\x9e\xe9\xc6\xbc\xfd\xbe\x52\x24\x97\x06\xfd\xba\xb1\x4d\xcb\x1b\xdf\xee\x00\xb9\x9d\xa1\x26\xdd\x2f\xc8\x29\xd3\xd2\x28\x9f\x16\x17\x83\x44\xcf\x34\x49\x00\x51\x04\x32\x1e\x9b\x6e\x80\xdf\x5c\xc4\x43\xe3\xce\xbf\x69\xdc\x43\x1d\x77\xa8\x63\x09\x6d\xec\xa9\xfb\xca\x99\x76\xc4\x06\xe9\xac\xa0\x0d\xad\x31\xc0\x7e\x19\x9c\xe0\x13\x51\xaf\xc5\x06\xb6\x3d\xfb\x2b\xcf\xbd\x20\x53\x22\x08\x8b\xc8\x01\x7a\xdb\x37\x09\x03\xf9\xc9\x4c\xd2\xc6\x80\x78\x28\x81\x70\xaa\x8a\x5b\x3d\xad\x24\xea\x76\x99\xe4\x5d\x26\x79\x97\x49\x5e\x3d\xea\x5d\x26\x79\x97\x49\x5e\x9b\x03\x11\x93\x84\x28\xd2\x28\x55\x9c\xc3\xe3\xb7\x92\x2a\x4c\xef\x5f\x86\x60\x61\xe6\xd2\xc9\x16\xbf\x19\xcd\xc2\x6d\xf8\x41\x68\x16\xe6\xac\xad\x33\x3f\x94\x7e\xac\x09\xb1\x7e\x75\x93\xc4\x36\x4c\xa3\x64\x97\x38\x87\xd7\xdf\x25\xeb\xa8\x0e\xbd\xb3\x51\xa0\x60\xeb\x5e\x8e\x93\x2c\x1d\x81\x76\x13\xb7\x1e\xc3\xf7\x3b\xef\x43\xe1\xa0\x4d\x74\x7f\xa8\x7c\x74\xeb\xa4\x94\x43\xb1\xd8\x7c\x41\x3c\xb2\xb3\xde\xbc\x71\xae\xc4\x12\x33\x7c\xb7\xd3\xed\x8c\x55\x9d\xb1\xaa\x33\x56\x75\xc6\xaa\xce\x58\x85\x3a\x63\xd5\xc6\xc6\xaa\x2f\x48\xa6\xea\x0c\x57\x9d\x58\xb5\xbf\xe9\x1e\xaa\x96\x79\x48\xd6\xba\xd6\x28\xe9\x45\x0e\xd5\xda\xc8\x7b\x3b\xed\x5f\xd6\x84\xdc\xdf\xb8\x11\xbc\x1f\x7e\x25\x5f\x9a\x25\xed\x12\x58\xec\x76\xf4\x8b\x8d\x2b\xee\x4a\x87\xd6\xae\x55\x17\xf6\xbc\x62\x71\xba\xb0\xe7\x2e\xec\xf9\xe0\xc2\x9e\xf7\xae\xac\x64\x5c\xae\x02\x24\x32\xa5\xb3\x56\xe6\x3f\xbb\x3b\x1b\x12\x8d\x80\x14\x0c\xca\x71\x4c\xb2\x84\x2f\xc0\x92\xb2\xe2\x3a\x77\x5d\xdc\x2c\x49\xd4\x87\x7e\xa3\xbb\x91\xbf\x96\xce\x71\x28\x32\x69\x31\xef\x83\x90\x42\x4f\xfe\x59\x49\xe7\x6f\x85\x97\xc9\x10\xf9\x4c\x25\xdc\x4a\xeb\x09\x7b\xc4\xea\x9f\x04\xa5\x0b\xed\x3d\x38\xc9\x55\x90\xbb\x27\xb5\x60\x95\x11\xa1\x16\xc1\x9b\x24\xcd\xd4\xe2\xbf\x46\x8c\x2a\xef\x61\xa3\x33\xc6\x85\xe1\x6a\xfa\xe3\x39\x66\x71\x42\x84\xbe\x54\x5d\x3b\x11\x66\x8c\x2b\x10\x37\x60\x06\x31\x7a\xa2\xd8\x08\x27\xfd\x9b\x61\x6b\x3f\xf3\x3b\x3a\x5d\xaf\x5d\xac\x6e\xcd\x5d\xf7\x29\xe1\x13\xa8\x60\x99\x97\x75\x7a\xdd\x40\xe7\x19\x2d\xed\xdc\x5b\x31\x04\x85\xe5\x63\x15\x38\xa4\x9c\x85\x3e\x5e\x09\x25\xb2\xe6\xdd\x12\xc6\xfc\xea\x57\x2b\x70\x23\xe5\x67\x16\x80\x04\x1e\xc3\x90\xab\xe3\x70\x3f\x86\x1d\xba\xdf\x8a\x96\xdd\x2f\xae\x74\x37\xfc\x28\x88\x12\x8b\x31\x56\x4a\x33\x99\x7d\x62\x9c\xdc\x63\xf9\xd8\x1a\xe3\xa4\xf4\xf2\x81\xb3\x9c\x12\xc6\x49\x79\xe0\x2f\xce\x72\x5a\x52\xe7\x1a\xce\xf4\xfe\xf2\xe3\xdb\x9e\xb5\x0d\x26\xfe\x5b\xc9\x95\x6f\xc7\x7b\xd6\x99\x69\xdf\x63\xde\xfc\x2a\x66\x7a\x30\x23\xac\xf0\xf3\x2f\xf1\xe4\x96\x6f\xa7\xee\x88\xae\x5a\xa3\x2f\xae\x10\x6e\x45\xe8\x58\x33\xb7\x77\x52\x10\xb7\x2a\x37\xed\x7b\x54\x2f\x63\xe6\x0e\x76\x63\x93\x18\xa0\x61\x19\xad\xdc\x9f\x21\x17\x15\x54\x94\x9e\x9d\x43\xa2\x35\x95\x61\x42\x7c\xc4\x85\x91\xbc\x62\x7b\x66\x8d\xd9\xce\x80\xf9\x9e\xa2\x3e\x8a\x6d\x6d\x7e\x41\x32\x41\x24\x61\xca\xa8\xda\xa6\xde\x95\x2b\xef\x4f\x99\xb5\x10\x9d\x63\x85\xcf\xb0\xc2\x09\x9f\x1d\xa3\xbe\x2f\xec\x4f\x25\xc2\x89\xe4\x08\x3b\xc2\x21\xb1\x6b\x02\xb3\xd8\xb1\x07\x8c\x22\x9e\x66\x34\xf1\x48\xed\xde\x8a\x4f\x59\x4c\x9f\x68\x9c\xe3\xc4\x23\x63\x8f\xd8\xe0\x89\x30\x95\x83\x0a\x87\x93\x04\xd9\x6e\xdd\x0b\x81\x7e\xee\x46\x29\x69\x4a\x13\x2c\x90\xe2\x76\xb4\xd7\xb6\x2d\x74\x3f\x27\x7e\xac\x0e\x05\x1c\xa5\xf8\x91\x48\x44\x15\xca\xb8\x94\x74\x92\x14\xc7\xf8\x61\x88\x60\xdc\x67\x17\x43\x30\x8d\x46\x0a\x71\xc3\x07\x5d\xe7\xd6\x4f\xe0\x7a\x4c\x31\x63\x04\x3a\xe6\x6a\x4e\x84\xed\xde\xbe\xfc\xd6\x56\xce\xb7\xc6\x88\x6e\xb6\x98\x86\x23\x7b\x3b\xa5\xb3\xb5\xc6\xd9\x56\xdd\x6c\xa7\x6b\x36\x2b\x9a\x2f\xe0\xa5\x6d\xaf\x0d\x5e\x50\x59\x56\x07\xdf\x85\xcb\xb6\x34\xe2\xd7\xc0\x47\xfb\x8d\x2a\x82\x9d\x16\xf8\x22\xeb\xf6\xa5\xaa\x80\x07\xae\xff\x75\xc8\x6e\x1d\x8a\x7d\x17\x80\xb1\xc7\xc5\xe9\x02\x30\xba\x00\x8c\x2f\x36\x00\xa3\x59\x9b\xa0\xf1\xce\xe9\x7a\x1b\x56\x90\xf2\x46\x01\xf1\x0b\x88\x52\x58\x3e\xb6\xad\x29\xa5\x45\xe5\x61\xfc\x2e\xa4\xfa\xda\x09\xbf\x86\x74\xdf\xd5\x29\xda\x6b\x9d\xa2\x83\x9b\x76\x27\xf8\x75\x82\x5f\x27\xdb\xb4\x9c\x70\x27\xdb\x1c\xae\x6c\xf3\x56\x0a\xcb\x97\x04\xa1\xab\x85\xa7\x52\x66\xcc\xca\x00\x5b\x03\x47\x03\xee\x81\x3c\x4b\x38\x8e\xd7\x05\xe1\xfc\x82\x0a\xb9\x66\x85\x68\x66\xda\xd5\x1f\x1c\xb8\x64\xb6\x14\x7f\x63\x46\xfe\x5b\x88\xa9\x6d\x9c\xfa\x9b\x86\xd5\x02\xfd\x42\x30\x59\x29\x28\xed\xa5\xb4\x90\x2a\x4d\xb7\x52\x38\xe4\x1f\x0f\x9c\xaa\xfd\x96\xbe\x86\x7a\xf1\x05\x3b\x08\x3a\x27\xc0\x6f\xb3\xf0\xf9\xc1\x48\xad\x9d\x6a\xd7\x65\x55\x76\x46\xfd\x4e\xf1\xed\x14\xdf\xbd\x2f\xe3\x21\x29\xbe\x6f\x28\x51\x9b\x34\x91\x17\x29\x63\xb8\x9d\x6c\xdd\x89\xd6\x9d\x68\xdd\x89\xd6\x5f\xac\x68\x7d\x18\x2b\xdc\xc9\xd5\x9d\x5c\xdd\xc9\xd5\x9d\x5c\xdd\xc9\xd5\x7b\x5f\xc6\x4e\xae\xae\xc8\xd5\xf0\x97\x4b\x93\xde\x54\xc8\x6e\x2d\x5c\xb7\xc8\x89\x7e\x2f\x92\x75\x27\x55\x77\x52\xf5\x61\x4b\xd5\x07\x33\xa1\x2f\x2f\x11\xb2\x4b\x25\xec\x52\x09\xbb\x54\xc2\xb7\x48\x25\x74\xbc\x64\x95\x84\xb2\x2c\x58\xfc\xb8\xc4\x81\x0e\x56\xb6\x28\x46\xbb\x6d\x78\xc7\xbe\x96\xda\x01\xcd\x6f\x53\x69\xaa\xf4\x9b\x6b\xe8\x80\xea\x4f\xf5\x9c\xb4\xa0\x19\x85\x1b\xdf\x7a\x84\xb0\x9f\xec\x9b\xef\x0b\x0c\x7c\x79\xd4\x5d\xfd\x29\x14\xec\x5a\x57\x7f\xea\x05\xe7\xed\x0e\xd7\x9a\x99\x3b\x1a\x35\x36\xde\x77\x3a\xed\x37\x07\x97\x6b\x3e\xe9\x6f\x1a\x2e\x57\x7b\x93\x2c\x25\xef\x9c\xfc\xb3\xf6\xa2\x78\x83\xb2\x5b\x1b\xdf\x0e\x9f\x88\xfa\x52\xae\x86\xae\xec\x56\x57\x1f\x62\x4f\xd3\xdd\x8a\xf5\xbf\xdb\xd9\x76\x45\xc6\xba\x22\x63\x5d\x91\xb1\xae\xc8\x58\x57\x64\x0c\xfd\xc6\x8b\x8c\x6d\x2c\x3e\x9a\x71\x7c\x29\x12\x64\x57\x64\xac\x13\x22\xf7\x37\xdd\xdf\x96\x10\x79\x80\x16\x84\x83\xa8\xa6\xe6\x2d\x08\x6f\x8e\xfb\xe1\x46\xd2\x16\xfb\xc3\x2d\x68\x87\xff\x61\xff\xaf\xc3\xff\xe8\xf0\x3f\x1a\x66\xdd\x05\xb3\x76\xf8\x1f\xa8\x0b\xd7\xec\xc2\x35\x0f\x39\x5c\xb3\xc5\x36\x76\xf8\x1f\x2d\xc5\xb9\x17\xc2\x00\x71\x32\xd7\x46\x38\x20\x3f\x2d\x2b\x1a\x07\x2b\xa5\xb9\xb1\xfe\x76\x70\x40\x6a\xa7\x7d\x10\x2a\xc9\x2b\xe2\x80\xd4\xd1\x75\x6b\x05\xe4\x7d\xe0\x81\xb8\xd1\x76\x89\x8b\x5d\x88\xf5\xe1\x87\x58\x1f\x5c\xe2\xe2\xc1\x48\xb2\x9d\xba\xd7\xe5\x2e\x76\xb9\x8b\x9d\x32\xdc\x29\xc3\x7b\x5f\xc6\x43\x52\x86\xdf\x58\xc2\x7e\x41\x5c\x90\xdd\x64\xed\x4e\xd4\x36\xef\x75\xa2\x76\x27\x6a\x7f\xa1\xa2\xf6\x61\xac\x70\x27\x67\x77\x72\x76\x27\x67\x77\x72\x76\x27\x67\xef\x7d\x19\x3b\x39\xfb\xd5\x70\x42\xea\x84\xed\x96\xf9\x36\xef\x49\xd2\xee\xa4\xec\x4e\xca\x3e\x6c\x29\xfb\x60\x26\xd4\x61\x86\x74\x98\x21\x1d\x66\x48\x87\x19\xb2\x95\x7c\xf3\x6f\xf6\x58\x7e\x08\x6e\x62\x7f\x65\x7f\xf8\x2e\xe1\x93\xfb\x45\x46\xf4\x7f\xcf\x69\x4a\x98\x04\x69\x94\xaa\x45\x28\xcf\x34\xac\xfc\xf2\x9a\x7f\xb8\x1b\x5e\x7d\xba\x08\xb3\x69\x3e\x5c\x3e\x5c\xdc\x0f\x6f\xfa\xb7\x7e\x5d\xfc\xac\xc2\xb5\xb0\xdf\x95\x44\x32\x4b\xf2\xb7\x44\xeb\x9e\x70\x6a\xee\x14\x56\xb9\xdc\x6e\x64\xb7\x83\xbb\xc1\xed\x8f\x90\x0d\x34\x3e\x1f\xde\xf5\xbf\xbb\x28\x11\x44\xe9\x79\xff\xec\xaf\x0f\xc3\xdb\xe6\xe7\x83\xff\x1e\xde\xdd\xdf\x35\x3d\xbd\x1d\x5c\x0c\xfa\x77\xcd\x5f\x7f\xec\x0f\x2f\x1e\x6e\x07\x2b\xd7\x63\xe5\x68\x57\x2b\x21\x12\x16\x09\xe2\xfc\x51\x64\xb9\x86\x28\xd6\x10\x79\xf1\xd1\xb1\xc3\xba\xbe\x4e\xd1\x83\xd5\xe9\xa9\x6d\xdc\x30\xd8\xa0\x21\xa3\x8c\xc4\x54\xe2\x49\x42\xe2\xa5\x96\xdc\x1a\x36\xb5\x84\x4b\x83\x7a\xd6\xda\xb3\x17\x39\x35\xcf\x8b\x0c\x2f\x40\x90\xa3\xa8\x08\x8b\x6b\xfa\x30\xfb\xd0\xd8\x03\xd3\xbc\x8b\x3e\x91\x52\x4f\x51\x2e\x04\x61\x2a\x59\x20\xf2\x99\x4a\x25\x97\x1a\x75\xdb\xd7\xd4\xac\xbd\x53\x7d\x83\x73\x2c\xd1\x84\x10\x56\x1e\xbf\x20\x09\xc1\xb2\x66\xcc\x76\xf7\xdb\x2d\x8b\xdf\x2b\x6b\x8d\x31\x97\xd1\x14\xd3\x24\x17\xa4\x72\x5a\x78\x9a\x61\x41\x25\x67\x83\xcf\xfa\x2e\xd3\x07\xf9\x1a\x3e\xe7\x62\xbb\x13\x33\xf8\x6b\x48\xc1\x57\xe5\x7f\x7e\xba\x2f\xff\xab\x74\xe6\x2f\xee\xcb\xff\x5a\x4d\xeb\x41\xc3\x55\xca\x3e\x42\x9f\xee\x4f\xd1\x27\x08\x71\x12\xe8\x7e\x8e\x0d\xc5\x5e\xdc\x9f\xa2\x0b\x22\x25\xfc\x52\x7c\xac\xa8\x4a\x60\x6e\xdf\x51\x86\xc5\x02\xb9\xe9\x9b\x44\x57\x1c\xcd\x11\xf1\x4b\x53\x5d\x3c\xf6\xf7\x9c\x81\xea\x5e\xac\xde\x05\x9f\xd1\x08\x27\xbb\x2d\x62\xff\xaa\xc4\x07\xae\x6f\x57\x2e\x45\xf8\xf6\xf2\x5a\xf4\xaf\xce\x21\x89\xd4\x0d\xb5\x66\xe6\x57\x44\x6a\x22\x89\x38\x8b\xad\x97\x46\xdf\xfe\x8b\x40\xa8\xff\x3b\x87\x44\xdc\x5c\x52\x36\xd3\x2d\xa2\x13\x74\x7d\x3b\x62\xd7\x22\x36\x86\x50\xa2\xa5\x61\x43\x73\x54\x22\xc6\x15\xa2\x69\xc6\x85\xc2\x4c\x69\x45\x00\xc4\x00\xbb\x22\x86\x03\x9c\xf1\x34\xcd\x15\xd6\x07\x6d\x69\x51\x99\x31\x87\xdc\x11\x35\x8c\xc1\xb5\x52\xb3\x86\x46\x4e\x28\xe6\x92\x09\xdd\xbe\x96\x51\xca\x3a\x34\x8d\x97\x54\x59\xd7\x04\x16\x02\x97\xa5\x89\x0f\x54\x91\xb4\xfa\x7e\xcb\x20\xcf\x7f\xd5\x1a\x08\xce\x4c\x52\x05\x11\x7d\x11\xcd\xa9\x22\x91\xd2\x47\x70\x2b\x9a\x78\xb8\xfa\xe1\xea\xfa\xa7\x50\x82\xf8\xd0\xbf\x3c\xff\xf3\x7f\x94\x7e\xb8\xbd\x5c\xfa\x61\xfc\xe3\x9f\x97\x7e\xf9\xff\xad\xa4\xa7\x6a\x4f\x4b\x7a\x7e\x30\x97\x23\x10\xa9\xc1\x26\xec\xa6\x8a\x68\x8a\x67\x04\xc9\x3c\xd3\x14\x20\x8f\xcb\xfb\xab\x45\xca\x0b\x8e\x63\xca\x66\x26\x03\xf4\x82\x2a\x22\x70\x72\x89\xb3\x8f\xce\x7e\xbd\xc5\xea\xfc\xdf\xbb\x52\xbe\xee\x87\x9f\xfb\x97\x61\xc6\xef\x87\x9b\xdb\xeb\xfb\xeb\x95\xb3\x2e\xb5\xb0\x7c\x8c\xf4\xe3\x53\xf8\x5f\x74\x82\x74\xeb\x5e\xf2\x4d\x89\xc2\x5a\x23\x40\x5f\x9b\xa4\x39\x9f\x48\x43\x59\x02\xa7\x26\x13\x34\xa5\x70\xa5\x18\x0b\xde\x37\x46\xb8\xf6\xda\x83\x3f\x37\xe6\x03\xd0\x96\xdd\xa5\xcc\x62\x2c\x62\xf4\x77\x59\x4d\x1f\x07\xc3\xb1\xf9\x81\xc4\xe8\x08\xcd\x95\xca\xe4\xe9\xc9\xc9\xf3\xf3\xf3\xb1\x7e\xfb\x98\x8b\xd9\x89\xfe\xe3\x88\xb0\xe3\xb9\x4a\x13\x93\x2e\xaf\x57\xe1\x14\xdd\x08\xae\xaf\x10\x50\xd0\x89\xa0\x38\xa1\xbf\x92\x18\x4d\x0c\xff\xe3\x53\xf4\x4b\xc4\x05\x39\x2e\x36\xc6\x1a\x95\xec\x3d\x62\x0d\x4f\x27\xfa\xa5\x1a\x66\x52\xdd\x4f\x14\x93\x88\xc6\x56\xcc\x20\x2c\xe2\x60\x79\x34\xbe\x0a\xdd\x9e\xcb\x34\xd4\x1a\x4d\x96\xab\x62\x39\x03\x65\x05\xc7\x24\xc8\x76\x57\xbc\x4c\x70\x5a\xf1\x19\x1a\xb5\x35\xd7\x2a\xba\xbe\x5b\x31\xdc\xaa\xee\xd5\x4c\x4f\x38\xe2\x09\x9a\xe4\xd3\x29\x11\xa1\x43\xba\xa7\xb5\x19\x2a\x91\x20\x11\x4f\x53\x90\x18\xf4\x57\xb9\x34\x54\x0d\x2b\x66\x47\x7b\x3c\x62\xb0\xff\x5a\xcd\x01\x0a\x88\x39\xb0\x3a\x46\x48\x8c\x30\x5b\x98\x6e\x26\xf9\x34\x6c\xdf\xc0\x50\xe0\x18\x51\x35\x62\xfd\x24\x41\x82\xa4\x5c\x91\x20\x87\x12\x9c\x67\xe5\x05\x07\x16\x29\x48\x96\xe0\x88\xc4\x86\x1e\x12\x1e\xe1\x04\x4d\x69\x42\xe4\x42\x2a\x92\x86\x0d\x7c\x0d\xb6\x1a\xbd\x66\x54\xa2\x98\x3f\xb3\x84\x63\x3b\x8f\xea\x67\xdf\x94\x4f\xe3\xc0\x41\x04\x0c\x84\xe0\x02\xfe\xe7\x07\xca\xe2\xbd\x71\xa8\x87\xbb\xc1\x6d\xf8\xef\xbb\x9f\xef\xee\x07\x97\x9b\x71\x1f\x4f\x59\x30\x3c\xd0\xe1\x4f\xd1\x9d\x59\x04\x2e\xb4\x44\x24\x1a\x26\x75\x69\x49\xa9\xf8\x81\xc7\x5b\x72\xdf\xcb\xfe\xd5\x43\xbf\xc4\x51\xee\xce\xbe\x1f\x9c\x3f\x54\xf4\x01\x3b\xbf\x92\x0c\x6f\xd4\xbf\xf0\xb7\xb3\xef\x87\x17\xe7\xe3\x1a\x85\xf1\xc3\xed\xe0\xec\xfa\xc7\xc1\x6d\xa1\xdb\xd5\x2e\x51\x65\x30\x55\x66\x75\x6f\x98\xd2\x9c\xc7\x68\xb2\xa8\x07\x84\xd0\x92\x73\x02\xbe\xd8\x02\x12\xc5\xb4\x7a\x0a\xbc\xc9\x61\x73\x14\x5f\xa4\x3c\x26\x3d\xfb\x0e\x20\x69\x18\xe3\x8a\x91\x98\xeb\x1b\xd6\xbd\x63\x16\x18\x2a\x0c\xc8\x85\x5f\xb8\x53\xd4\x47\x52\xbf\x98\xeb\x43\x2d\xe8\x6c\x06\x86\xc3\xca\x50\x4d\x6b\xf6\x53\x58\x5e\xf8\xce\xec\x7f\x26\x38\x9c\x73\xdd\xad\xb5\x38\x7b\xab\x84\xf9\x10\x50\x57\xca\x2d\x0a\x0c\x06\x87\x9a\xa1\xb9\xcd\xd2\x8b\xd0\xb8\x5e\xe6\x3c\x1a\x7b\x91\x3e\x5c\xc0\xb6\xa4\xb1\x77\x66\x82\x3c\x51\x9e\x07\x9f\x5a\x60\x8f\xd2\x8e\xd7\x36\x5f\x2c\x00\x2c\x9b\x31\x8a\x54\x9a\xf1\xe4\x51\xdb\x82\x66\x61\x4f\xd0\xc2\x54\xf0\xb4\xa6\x8d\xf2\x31\x19\x5e\xdf\x29\x81\x15\x99\x2d\xce\x2d\xcb\xd8\xfe\x78\x9c\x5f\xff\x74\x75\x71\xdd\x3f\x1f\x0f\xfa\x9f\xca\x27\xde\x3f\xb9\xbb\xbf\x1d\xf4\x2f\xcb\x8f\xc6\x57\xd7\xf7\x63\xf7\xc6\x4a\x92\x6f\xe8\x60\xf9\x9e\x2e\xbf\x78\x8a\x34\xcb\x05\xd6\xe8\x00\xef\x02\xfe\x38\x21\x53\x2e\x0c\x9f\x4f\x5d\xe8\x82\x15\x61\xdc\xda\x5a\x5d\xac\x32\x8b\x53\xb0\x8c\xd5\x35\x69\xac\xde\x4a\x10\x9c\xc2\x3d\x81\x19\x1a\xb0\xf8\xe8\x7a\x7a\x74\x67\x7e\x4c\xb1\x78\x24\xc2\x7f\xfa\x2c\xa8\x52\x84\x95\x54\x3a\xec\x86\xec\x95\xc4\xa2\x83\x63\x74\xab\xf9\xbe\x7e\xdf\x5f\x6a\x9a\xd8\x63\xa2\x30\x4d\xa4\x1d\x6c\x69\x5d\x4f\xd1\x05\x16\xb3\xc2\x0e\xf7\x35\x9f\x4e\x4d\x63\xdf\x98\x61\xe8\x3b\xac\x34\x8b\x1a\xde\xab\x49\xc3\xdd\x8b\xd0\x9f\x7d\xd9\xcb\xc3\xcb\x54\xf5\x90\xed\x46\x53\x0f\x37\xb0\xe2\x46\x63\x2f\xe9\x86\xf6\x49\x0d\xad\xc1\xc4\xcd\xe3\xd5\x97\x4c\x7d\xdb\xcb\xe4\x54\x7e\xb1\x86\x9c\x4c\x2e\x95\xde\xf9\xa9\xd6\x36\x6b\x68\x89\x7c\xa6\xd6\x60\x10\x8e\xbb\x42\x42\x45\x33\x60\x5e\xc5\x59\x46\xb0\x90\x75\xbb\x5d\x16\x03\x1b\xf6\xde\xf4\x14\xf6\x61\x37\xd9\xf5\xd3\x43\x9c\x81\xc1\xc1\x0b\x11\x15\x8a\x6c\x41\x03\xa6\xad\x25\x0a\xb8\x01\xb4\xa5\x6b\x8b\x6c\x74\x49\xa5\x56\x1a\xcd\x8f\xdf\x59\xc8\xa5\xed\x08\xe2\x63\x7f\x78\x51\x11\x2e\xc6\xe7\x83\x8f\xfd\x87\x8b\xd5\x66\xc2\xd2\x77\xd5\x2d\x46\x47\x48\x3f\x2f\xfb\xcd\xe9\xd4\xdc\x19\x0e\x38\xca\xa8\xb4\x84\x81\xd1\xca\x42\xd5\x18\x7b\x75\x4c\xb2\x84\x2f\x52\xc2\xc0\xc4\x53\xba\x09\xf5\x7a\x4e\x31\xb5\x57\x4b\x30\x58\xb0\xe2\x58\xb3\x1b\x5c\x63\x47\x0e\xad\x8a\xc4\xfe\xe6\x2d\x83\x55\x55\x58\xf7\x8d\xf1\x9e\xd9\xff\xdc\x29\xac\xb6\x3c\x63\xfd\xb3\xfb\xe1\x8f\x83\xb2\x7e\x78\xf6\xfd\xf0\xc7\x3a\xa9\x66\xfc\x69\x70\x35\xb8\xed\xdf\xaf\x11\x4e\x2a\x4d\xd6\x09\x27\x52\x0f\xb8\xea\x3d\xa5\xd2\x47\x04\x45\x06\xf2\x0a\x51\x25\xd1\x13\x95\x74\x42\x01\x20\xcc\x7a\x22\x1f\x86\xc0\x59\x9f\x70\x42\x63\xaa\x16\x4e\x7c\x31\xfd\x96\xf7\x51\x73\x52\xdb\xbe\x31\x3b\x84\xfe\x49\xb0\xf2\x99\xcd\x71\x93\x3e\x45\xa0\xdb\x3e\x81\xd2\x16\x7c\xc6\xb4\x20\xcd\x66\x44\x98\xe1\x80\xf7\x25\x1c\x4b\xf0\x5c\x8f\x2a\x14\x56\x8a\x55\xf3\x42\xeb\x8c\x30\x22\x00\x04\xce\x77\x62\x04\x29\x41\xd8\x57\x5a\xe6\xca\x12\x1a\x51\x95\x2c\x50\x04\x36\x2c\x30\x67\xa6\x98\xe1\x99\x15\x0e\x40\xcd\xa9\x90\xc4\x5f\x0d\x8a\xda\xf5\xd4\x9a\xf6\xef\x29\xd9\xf2\x98\x3d\x5c\x9d\x0f\x3e\x0e\xaf\xca\x24\xf0\xfd\xf0\x53\x49\x84\xbd\x1c\x9c\x0f\x1f\x4a\xb7\xb9\x96\x64\x57\xcb\xf5\xd5\x66\x6b\x8e\xa2\x7f\xe9\x14\x9d\x9b\x4f\x4f\xf5\xe2\xd6\x40\xc4\x79\xe5\xb7\xb2\x0e\xb7\x2e\x24\xcf\xfd\x31\x60\x4a\xd4\xfa\x25\xda\x9a\x90\xac\x0f\xb2\x64\x43\xaa\x0f\x55\x58\xea\xfb\xaa\xea\x54\xae\x4e\xd9\xbd\x08\x41\x97\xc7\x85\x65\x29\x8c\x61\x00\xa3\x41\x93\x11\xab\xc6\xad\x55\x30\xec\x1f\xc1\x45\x9d\xe6\x52\x19\x57\x22\x10\x27\x7a\xfc\x8b\xd4\x0b\x0a\xae\xc6\x63\x74\x47\xc8\x88\x39\xeb\xc1\x8c\xaa\x79\x3e\x39\x8e\x78\x7a\x52\xe0\x13\x9e\xe0\x8c\xa6\x58\x4b\xd2\x44\x2c\x4e\x26\x09\x9f\x9c\xa4\x58\x2a\x22\x4e\xb2\xc7\x19\x44\xc0\x38\x77\xea\x89\x6f\x76\xc6\xff\xfd\xe2\x4f\xdf\x1e\x5d\xfc\xe5\xdb\x0f\xcb\x16\xb2\xa6\xfd\x1f\xb0\x08\x67\x32\x4f\x6c\xc4\x9c\x08\xd7\xc6\x1d\xf9\x9c\xac\xdb\xef\xab\xf2\x76\xed\xa6\xbf\x9e\xdd\x3c\x94\x2c\xd6\xe5\x7f\x5e\x0e\x2e\xaf\x6f\x7f\x2e\x71\xca\xfb\xeb\xdb\xfe\xa7\x12\x43\x1d\xdc\x7c\x3f\xb8\x1c\xdc\xf6\x2f\xc6\xee\xe1\x2e\xb6\xb7\x1f\x18\x7f\x66\xe5\xa5\x91\x8e\x03\x2e\xf5\x74\x8a\x3e\x72\x81\x7e\xf0\x3b\x79\x34\xc1\x12\xae\x18\x77\x67\xc9\x1e\xca\x78\x0c\x8c\x17\x91\x6c\x4e\x52\x22\x70\x62\x6d\x06\x52\x71\x81\x67\xe6\xa6\x97\x91\xc0\x2a\x9a\x23\x99\xe1\x88\xf4\x50\x04\xd4\x30\xeb\xc1\xa6\x80\xaa\xc5\x67\x55\x3b\xdf\x6d\xce\x14\x4d\x89\x53\xc1\xed\x3f\xef\xcd\x66\x6c\xb1\x39\xd7\xf7\xdf\x97\x85\xbd\x8f\x17\x3f\xdf\x0f\xc6\x77\xe7\x3f\xac\x5c\x4f\xf3\x59\x69\x64\x77\x10\x80\x74\xc6\x93\x3c\x65\xe1\xdf\xdb\x8f\x6d\x78\x75\x3f\xf8\x54\x1d\xdd\x75\xff\xbe\x4c\x19\xb7\xe5\x00\xb7\x0f\xdf\x5d\x5f\x5f\x0c\x4a\x2e\xe1\x0f\xe7\xfd\xfb\xc1\xfd\xf0\xb2\x44\x3f\xe7\x0f\xb7\x06\x8d\x70\xd5\x34\xdd\x08\x6a\x26\xaa\xa7\x15\x4e\x73\xdf\xac\xb0\x15\x27\xea\xdb\x80\x72\x73\x96\x8f\x02\xb8\x1d\x13\x0e\x06\x56\x9d\x23\x6f\x52\x8d\xcc\x48\x6b\xd9\xa1\x2a\x6f\x13\x6a\x66\xc7\x2b\x37\x7a\x15\x57\xbe\xf7\x43\x30\x50\xa0\x46\xd9\xc6\x49\xc2\x9f\x4d\x28\x6f\x4a\xf5\xad\x6c\x81\xd1\xf4\x2b\xb2\xf0\x10\x1e\xd7\x70\xbc\xf2\xb6\x90\x48\x10\x75\xc9\x73\xa6\xb6\x27\xb9\xfe\x55\x89\xef\x0c\xae\x7e\x1c\xff\xd8\x2f\x53\xe0\xf0\x62\x35\xab\x09\x9b\xa8\xb9\x8a\xfb\x57\x3f\xfb\x4b\x18\x02\xbe\x7b\x5e\x43\x35\xb2\x6b\x94\x50\x2d\xf6\x46\x58\x6b\xaf\x09\x48\x34\x88\x50\x30\x39\xa4\x7a\x72\x10\x60\x9a\x19\x7f\x92\xe1\x4f\x66\x90\xa7\xee\x8f\x4a\x7b\x12\xd6\x05\xac\xa9\x2e\x9e\x1e\xda\xb1\x5a\x35\x43\x84\x3d\x51\xc1\x01\xcf\x16\x3d\x61\x41\xb5\x34\x6e\x5a\xd6\x73\x3d\x85\xff\xdd\xac\x4d\x30\x8c\x56\x18\xd7\x1d\x17\xea\xdc\x07\xf2\x6e\x67\x0d\xa9\x0b\x68\x5d\x0e\x65\xad\x37\x74\x2c\x7f\x5b\xb3\x39\x3b\x06\xfc\x96\x27\xfc\x8f\xe4\x9c\xe2\x44\x33\x80\xfd\xc9\x8b\xfd\xab\xbb\x61\x59\x7e\x2c\xab\x19\x01\x5f\xde\x5a\x5e\x04\x43\xa5\x19\xb9\x53\x26\xee\xfe\x7a\x61\xb4\x0b\x00\x3d\x36\xe7\x36\x50\x2c\x40\x00\x72\x28\x28\x19\x16\xb2\xf2\x85\x44\x00\x84\x56\x04\x5c\xe9\x3b\x0b\xc2\x99\x9e\x38\x8d\x47\x8c\x7c\xce\x08\x93\x10\x1c\x60\xee\xb3\xc2\xd7\x2e\x8f\xd1\x70\x0a\x2c\x41\xbf\xce\x50\xce\xac\x03\x4c\x5f\xb8\x66\x90\x3d\x2d\xca\xda\x21\x78\x0d\x11\x0c\x2f\x8c\xb8\x60\xa9\x62\xf0\x23\xf6\x93\x77\xa2\xc1\xa3\x29\xd7\x0c\x48\xef\xa2\x6d\xef\x14\x61\x26\x69\x0f\x69\x85\xa5\xba\xa7\x90\x3a\xa0\x15\x4a\x1b\xc2\xa5\x39\x8d\xfd\xf3\xf5\xaf\x81\xa5\x38\xe1\xf0\x32\xa8\xbf\x0b\x2a\x57\x41\x83\x68\x9c\x18\x8f\xc9\xb8\xfd\x9d\x10\x71\x41\xac\x9f\x65\xe3\x6b\x60\x1d\x63\xbf\xc7\xf2\x71\xc9\xf7\x30\x64\x52\x61\x16\x91\xb3\x04\xcb\x2d\x83\x90\x9c\x8d\xa3\x57\x96\x38\x6e\x6f\x1f\x6e\xee\x87\xdf\xad\xe1\xf2\xd5\x8f\x97\xc3\x80\xa2\x24\x77\xee\xb9\x89\xe0\x38\x46\x9a\x7d\xce\xb8\x71\x05\x5a\xc1\xbf\x80\xfe\x36\x79\x3d\x3e\xa0\xb2\x04\x3b\x5e\xa4\x23\x58\x3b\x47\xe8\x4a\xa0\x76\x21\x50\xa4\x57\x02\x05\x26\x0f\xb7\xd5\xe0\x59\x34\x05\x49\xac\x75\x2b\x4b\xb0\x9a\x72\x91\x1a\x2e\x5f\x9a\xb4\x69\x7c\x75\xa3\x94\x29\x22\x44\x9e\x29\xea\xb0\xdc\xab\x52\x2a\x54\x78\xe7\xb3\x4b\x22\x25\x9e\x91\x5d\x1c\xd0\x75\xca\xc3\xdd\x8f\xe1\x3f\xc1\xc1\xdc\x46\xf6\x2f\x8d\xd0\x45\xbe\x3b\x7a\xba\x66\x1f\x4d\x20\xcf\x0d\x4f\x68\xb4\x65\xc0\xdd\xc7\xfe\xf0\x62\x3c\xbc\xd4\x4a\x7c\xff\x7e\x70\x51\x12\x25\xe0\x59\xff\xe3\xfd\xe0\xd6\x82\x58\xf7\xbf\xbb\x18\x8c\xaf\xae\xcf\x07\x77\xe3\xb3\xeb\xcb\x9b\x8b\xc1\x9a\xc8\x9c\xc6\xc6\x97\xad\xab\xd5\x57\x4f\x97\x7e\x81\x1d\xd6\xbc\x2c\xb4\x97\x41\xd6\x18\xa6\x09\x38\xc1\xb9\x71\x86\x63\xc4\x78\x4c\xe0\x67\xe9\xac\x33\x1e\x39\x1a\x0d\xd5\x57\x49\x82\x70\xae\x78\x8a\xc1\x6b\x93\x2c\x46\x0c\x4f\x34\x6b\xc5\x49\x12\x84\x77\x89\x9c\x31\xcd\x62\x75\x63\x06\xa2\x3d\x4a\x88\x66\xe7\x59\x90\xec\x67\xfd\x06\x53\xca\x20\xd2\x36\xc5\xe2\xd1\xb8\x99\x8a\x2e\x8b\x43\x21\x11\x96\x23\xa6\xc7\x45\xac\x61\xa8\xcd\x0a\x9f\xb6\x7a\xab\x71\x75\x52\xfc\x48\xf4\xaa\xa4\x79\x34\x47\x99\xe0\x33\x41\xa4\xb4\xb6\xe5\x08\x33\x13\x80\x60\x5f\xd7\xd7\xd0\x88\x31\xae\x97\xc2\x99\xb0\x63\x92\x11\x16\x13\x16\x51\x93\xd6\x07\xbe\x7b\x6f\xda\x9c\x09\x9c\xcd\x91\xe4\xe0\xf4\x86\x65\x07\xfb\x95\xf9\xc8\xdd\x64\x66\xc6\xe6\x71\x68\x81\x16\xb9\xe6\x13\xd7\x20\x27\x9a\x55\x86\x8f\xdd\x65\xe8\xdc\x2e\xc6\x0e\x98\x66\x09\x51\x06\xac\x1f\x96\x1c\x36\x43\xaf\x75\x69\x3f\xf4\x36\xd5\x6d\x82\xbe\xb0\xdd\x98\xb1\xb4\x23\x3a\xae\xb1\x6c\xdb\x23\x85\xbe\xc7\x2c\x4e\x74\x2b\xce\x87\x51\x3e\x8b\x90\x8a\xd2\xd7\x54\xe3\x4e\xe3\x2e\xb7\x68\x84\x73\xb9\xcb\x35\x5a\xc9\xc5\x34\x56\xc1\xa3\x22\x28\x04\xc8\xdb\x26\x62\xc2\xea\x66\x9a\x45\xe2\x84\xdb\x55\x32\xaf\xe7\xa6\xfe\x13\x82\xd1\x34\x5c\xb3\x99\xa0\x2c\xa2\x19\x4e\xb6\xd2\xfd\x2a\xc1\xf8\x36\xc6\xfd\x6b\x3a\xd5\xe4\xf3\xcd\x92\xdb\x56\x11\x91\x42\x82\xb2\x1d\xa6\xdf\xc2\x0d\x2c\x49\x36\xab\x81\xc8\x22\x9a\x04\x0b\x9e\x1b\x7f\x1c\xac\x0b\x89\x6b\x8e\xea\x71\xdd\x76\xeb\x93\x81\xcb\x01\xd0\x5b\x6c\xb6\x89\xfc\x69\x5a\xbf\x4a\x2b\xb6\x77\x13\x8c\x87\x93\x9b\xfa\x36\xeb\x76\x20\x78\xf8\xaf\x55\xb4\x73\x89\x33\x4d\x33\x16\xb6\x1f\x17\x73\xb4\x4a\x92\xad\x0a\xe6\xe2\x67\x02\xdf\xb9\xcf\x0b\x69\xbf\x1b\xc5\x12\xda\x00\xa8\xe5\x4e\x4a\x31\x04\x41\x8e\xb9\xa5\xf1\x69\xae\x65\x59\x84\x21\x0a\x01\x7d\x4d\x8e\x67\xc7\xc8\x15\x61\xe8\xa1\xfe\xcd\xcd\xe0\xea\xbc\x87\x88\x8a\xbe\x71\x31\x8b\x36\x60\x69\xc4\x14\xb7\xd2\xca\xc2\x15\xd0\x48\x89\x98\x91\xd2\x9c\x5d\x74\x13\x84\x2a\xcf\xa8\x54\x36\x7c\x56\xf3\x95\xa0\xd4\x09\x4d\xab\x62\xb6\xa1\x90\x5c\xcd\x77\x21\x0d\x2c\x65\x9e\x6a\x5d\x76\x4c\x71\x3a\x16\x3c\xd9\x85\x29\x9c\xc3\x54\x40\x5d\xf6\xe9\xf9\x14\xa7\x48\x37\x6b\x43\x41\xbc\xcb\xd1\x8b\x74\x5a\x30\xd2\x7c\x59\xdf\x9b\xc1\xbd\xe5\xbc\x0f\x36\x1e\x8d\xba\x10\x08\x48\xdf\x6f\x60\x15\x85\xd9\x78\x6c\x2d\xf5\x63\x1c\x45\x5a\xe5\xde\xf3\xa4\x82\xfa\x39\xce\x25\x60\x3b\x7a\xb1\x69\xae\xa3\x73\x37\xcc\x4c\x73\x30\x08\x06\xd6\x57\xae\xe4\x11\x2d\xda\xaf\xe9\x77\xb2\x58\xea\xd5\x55\xb8\x79\x90\xde\xa4\x62\x2e\x61\x49\x60\x27\xa5\xa9\x90\xa3\xe6\x64\x81\xe6\xf8\x89\x94\xba\x74\x09\x31\xba\xe1\x05\xcf\x45\x1d\xa3\x1b\xb1\x73\x92\x09\xa2\x25\xfd\xaa\x03\xc5\xd3\xf4\x6d\x99\x12\x3b\xba\xee\xe8\xfa\xdd\xd3\xf5\x99\x29\x94\xd4\xf7\x85\xb1\x76\x12\xe0\x4c\x63\xe3\x8c\xf3\x64\xdc\xc2\x26\xd2\x7e\xc5\x4b\x9e\xb0\x4a\xd9\x28\x80\x04\xe0\x39\xc8\x47\xa5\x6b\x93\xeb\xbb\x2e\x48\xb1\xb5\xc3\x5b\xb1\x0c\xce\x65\x16\xd4\xcb\xd9\xe5\xbc\xd7\xb5\xb2\xaa\x25\xf4\xe2\x62\xce\x99\x91\x6f\xbc\xbb\x2c\xac\x7f\x5a\x3a\x4c\x4e\x14\xa1\x6c\xa9\x1a\x9b\xa1\x67\xbd\xc0\x46\xee\xf8\x47\xce\x15\x96\xdf\x1c\x8f\x98\x16\xa2\x1e\xc9\xc2\x98\x5b\xb5\x98\xf2\x3b\x2d\x8b\x1f\x49\xc2\x24\x84\x7b\xff\xce\xb8\xe7\x34\x89\x3b\x73\xb5\x51\x4d\x4d\x11\x38\x08\xba\xf6\xbd\x40\x88\xae\x6d\xd4\x4a\x49\x45\x00\x34\xc8\xf9\x66\x2e\xf6\x99\x19\xfe\x8c\x28\x48\xb1\x56\x54\x81\xce\x14\x9b\x2a\x73\x4b\x43\x5f\x6b\xba\x32\x54\x21\x38\xf8\x49\xe2\x7c\x37\xc6\x2f\x97\xdb\x58\xcb\x19\xbd\xb6\x70\x67\x63\xde\x4f\x9c\xdd\x28\x12\x7c\xa9\x74\x1b\x96\xc8\xec\xf4\xc4\xb0\x03\xe7\xbf\x26\xec\xf8\x99\x3e\xd2\x8c\xc4\x14\x43\x04\xbc\xfe\xd7\x89\x9e\xd7\xbf\x9f\xdd\x5e\x5f\x8d\x8b\x4c\x9e\xff\x1a\xb1\x7e\x22\xb9\xcf\x52\x40\x8c\x33\x1f\x6e\x9f\x09\xe2\x44\x42\x3b\x17\xb0\xba\x16\x66\xc4\x11\x6b\x1a\x41\xcc\x23\x79\x8c\x9f\xe5\x31\x4e\xf1\xaf\x9c\x81\x2b\xbd\x0f\x7f\x9e\x25\x3c\x8f\x7f\xc2\x2a\x9a\x9f\xc0\xb9\x56\x27\xe4\x89\x30\x65\xdc\x54\x7a\xb9\x62\x48\xde\x95\x10\xad\xff\xef\x7a\xcc\x45\x52\x91\xd4\x9a\x6c\x44\x32\x85\xfe\x1f\x41\x26\x9c\xab\xfa\x4b\x8a\x4f\xa7\x92\x6c\x74\x21\x15\x4a\xda\xdd\x35\xfa\xcb\x9f\xbf\xfd\x83\x26\xa1\x6d\xd6\x78\x78\x77\x3d\xd6\xdf\xff\xfb\xb9\xfd\x5e\x6e\xc0\xee\xae\xb3\x82\xb5\x39\xe2\x31\x81\xf3\x39\x83\xdb\x4f\x80\xf3\x02\xd8\x1b\x90\x43\xb1\x8f\x75\xdc\xed\xbc\xd4\xfa\x6e\x2a\xdb\x56\x8b\x09\x2a\x76\x30\x47\x74\x84\x18\x47\xa9\x89\x35\xc5\x0c\xfd\xc7\x0f\xdf\xd5\x6f\x60\x2e\xe8\x56\x1d\x52\x0b\xd7\x10\x74\x29\xe9\xaf\x44\x22\x4d\x35\x9a\x8a\x79\xaa\xbb\x16\x44\xce\x79\x12\xa3\x67\x02\x6a\x92\x8d\x03\xf5\x5a\xb9\x20\x23\x16\x36\x01\x21\x87\x08\x27\x8a\xcf\x08\xdc\xd5\x4e\x51\x53\x44\x68\x51\xc5\x64\x69\x28\x2e\x48\xcf\x40\x7d\xdd\xfd\xc9\xc5\x56\xc3\x34\xe1\x91\x4b\x6a\xb1\x26\xb9\x78\x52\x3f\xf3\x69\xd5\xf4\x8a\x9a\x6d\xf8\xd5\x4d\xb6\x66\xdb\xfa\xa5\xb1\x49\x28\xd6\x86\x55\xdd\x99\xfa\xc1\xd0\x88\xb3\x71\x42\xd9\xe3\x56\x9b\x71\xed\x44\x39\xdd\x82\x5d\x33\xdd\xa2\xb7\x73\x1b\x0b\xc8\x06\xe7\xe3\x63\x9e\x24\x26\xb5\x25\xdc\x1e\x90\xbb\xcc\xba\x81\x30\x90\x99\x1c\x50\x12\x5b\xbf\x97\xd5\x84\x05\x61\x10\xf0\x36\x62\x93\x85\xf5\xd9\xca\x1e\x92\x79\x34\x77\x99\x79\x11\x67\x52\x8b\xd1\x5c\xa0\x88\xa7\xa9\x29\x6e\xca\x08\x52\x9c\x27\xd2\x46\xbb\xb3\x23\x85\x23\x35\x62\x45\x7f\x6b\x4e\x9e\xa9\x80\xb4\x5b\xea\x5e\x7b\x97\x4e\x51\x69\x69\xa5\xc0\x4d\xe3\x10\xb3\x01\x8c\x60\xc6\x13\x15\xa0\x3f\xf0\xe5\xb3\x64\x36\xac\x41\x33\x90\x73\x2e\xd4\x38\xae\xe5\x39\x6b\x89\xa6\xca\x08\x19\x39\x4a\x20\x68\x98\x3f\x69\xe1\x9f\x3c\x7b\xe3\xeb\xaa\x21\x68\xaa\x5e\x35\x82\x76\xc7\x68\xe5\xc8\x36\x25\xc1\x86\xb5\x32\x08\x1e\x51\x39\x26\x7c\xdd\x18\xef\xe0\xab\x33\xfd\xd1\xca\xc5\xab\x9e\x3b\x27\x04\xf1\xb8\x00\x9b\x33\xf7\xba\xcd\x08\x59\xb5\xa6\x16\x3a\xe1\xe5\x32\x47\x57\x4d\xe5\xa1\x6c\xc9\xd5\x63\x01\x93\xbd\x24\x20\x6b\x62\x31\xa1\x4a\x60\x51\x42\x0a\xf1\xfa\xa0\x24\x58\x40\x7c\xd6\x88\x19\xdc\x38\xa3\x29\xc4\x28\xa6\x12\x12\x44\xe0\x2e\x0d\x9c\x61\xa8\x9d\x12\x58\x39\xda\x45\x9e\xa3\x89\x3f\x87\xc0\xb2\x82\x34\x1c\xb3\xd3\x1d\x79\x7c\x2c\xad\x9f\xf1\x28\x2f\x04\xb9\x08\x24\x5c\x8b\xa9\x83\x28\x93\x74\x36\x57\x88\x32\x6b\x77\xc4\xc9\x8c\x0b\xaa\xe6\xa9\xec\xa1\x49\x2e\xb5\x16\x6a\x82\xd5\x4c\x3c\x0a\x51\x51\x2b\x2e\xb4\x6b\x12\x71\x5c\x69\x70\x59\x45\xd9\x82\x34\xda\x1d\xca\x41\xe5\xae\x58\x43\x38\x7d\x8f\x33\x58\x6d\x83\x42\xdd\x46\x03\x4f\x89\x4c\x1c\x20\x77\xc8\x4e\x50\x05\xa4\xe9\x1c\x00\x2a\xe4\xde\xbc\x14\xaf\x51\x88\x0b\x99\x64\x50\x41\x5c\xec\x36\x48\x5e\x65\x64\x4a\x83\xde\xe4\x9d\x4e\x69\xa6\x6a\x03\xb7\x96\x5d\x45\xb7\x01\xe6\x4f\xbb\xc5\x86\x64\x2c\xa0\x66\x40\x6a\x1b\xb1\x3b\x42\x9a\x81\xdc\x96\xf6\xde\x94\xc6\x85\x29\xd8\x44\x8f\xd5\x24\xbf\x8b\x13\xfb\x7c\x70\x77\x76\x3b\xbc\x31\x90\x13\xd7\xb7\x97\xfd\xfb\x71\x8d\x5f\xbb\xe6\xad\xcb\xfe\xed\x0f\xe7\xeb\x5f\xfb\xfe\xbe\x9c\x95\x5d\xf3\xca\xed\xdd\xea\x64\x8e\x16\x43\xac\x49\x0a\xab\xed\xe7\x14\x65\x0b\x35\xe7\xcc\x87\x28\xc4\x25\xde\x74\x84\x4c\x46\xb0\x82\x10\x22\x21\x55\x8d\xe3\xf0\x1e\xe2\x72\xd6\x4b\x98\xe5\xcd\x32\x30\x6c\x7b\x15\x8d\x36\x38\x91\x9f\x12\x3e\x01\xbf\x75\x5e\x2a\x71\xbb\x22\x02\x7d\xc7\x78\x9f\x73\x2a\xb3\x04\x2f\x96\x7a\x58\x77\xe5\x5c\xe1\x94\x40\xc4\x71\x81\x1f\xe7\x92\x45\xf4\xce\x40\x02\x93\xbf\xd7\xe9\x14\x32\x99\x14\xc5\x8a\xa0\x09\x51\xcf\x90\x37\xe7\x7e\xf5\xb6\x54\x17\x30\x22\x8f\x47\x0c\xcc\x39\x23\xbd\xc8\x71\x0e\xd1\x7e\xa3\x0f\x3d\x34\xfa\x10\x93\x27\x92\xf0\x4c\xef\xbc\xfe\xa1\xe1\x92\x19\xa4\x98\x26\x57\x5c\x79\xcb\xdc\x2e\xfb\x29\x48\x44\x33\x90\xcc\xc7\x44\xb7\xfb\x7a\x82\x47\x89\x92\x1d\x3b\x83\x31\x20\x1c\xc7\x5a\xc9\x06\x56\xe6\x86\x57\x84\x00\xb1\x60\xea\xa5\x5a\x99\x9b\x88\x14\xde\xfc\x6d\x7a\x0c\xdb\x2c\x9b\x3d\x6b\x77\x80\x3d\xbd\xa0\x4b\x76\xd7\x8b\x5c\x6b\x25\x3f\x90\x05\xa4\x60\xdc\x60\x2a\xb6\x74\xcd\xd6\xc5\xbc\xbe\x88\x93\x76\x50\xd3\xd1\x01\xb9\x6b\xeb\xd7\x61\x37\xc7\xad\x8f\xd5\x7b\x2d\x2d\xd5\xc5\x72\xf9\x8e\x5b\xaa\xad\x0f\x4d\x4a\x6a\x63\x08\x03\xaa\x2a\x5e\x19\x89\x36\xd0\xb8\xfc\x00\xef\xf4\x77\x6b\x35\x15\x2f\xae\x45\x61\x4d\x7f\xd8\x05\x9b\x1c\x5f\xcd\xc7\x27\x6b\x47\x1c\x25\x5c\x96\xb1\x72\x5a\x0f\xfa\xcc\x7e\xba\x6a\xdc\x83\x90\x7c\xb5\x5c\xb8\x51\x40\x43\xcd\xc2\x57\xc0\x20\xcd\x3d\xa3\xac\x87\xcc\xbe\xdd\x43\x14\xa2\x2d\x41\x21\x4b\x0a\xe4\x00\x16\xa3\xc2\x0d\x32\x62\x45\xcc\x8a\x44\xcf\x24\x81\x30\xb7\x88\xa7\x19\x98\xf8\xed\x70\x6d\x4b\x24\x36\x11\xc3\x3d\xc4\x73\xa5\x1b\x33\x39\x39\xce\x88\x6b\x13\x7e\x0a\xb7\x87\xf1\xbd\xd9\xe0\x77\x0f\x2c\x6d\x68\xdd\xdc\xa5\x94\xa1\x4f\x44\x41\x2b\x00\xdc\x1f\x4e\x10\xf4\x84\x6a\x08\x65\xfd\xda\xef\x70\xa2\xec\x4c\x36\xd8\xf9\x02\x38\xe5\xbb\x84\x4f\x56\x1b\x09\xa0\x71\xf4\x70\x3b\x74\x16\xc9\x22\x7e\x2a\x40\x2f\x2e\x79\x14\x07\x37\xb7\x83\xb3\xfe\xfd\xe0\xfc\x18\x3d\x48\xa2\x97\xc7\x4f\x17\xf2\xab\xbd\x4a\x62\x46\x6e\x91\x58\x98\x54\x04\x37\x19\x42\x88\x10\xa5\x2c\xe8\x35\x8c\xa3\x0c\xd3\xb2\x9a\xb0\x01\x24\x85\x5a\x43\x1d\x00\x0b\x55\xe7\x69\x23\xf3\xd6\x9d\x40\x88\x93\x1a\xbf\x9f\x28\x35\x33\xde\x74\x39\x32\x6f\x1d\xf9\x94\x23\xfa\x5e\x7a\x32\x70\xb4\xd4\x9c\x50\x81\x5a\x4d\xcb\x10\xd5\xb8\xfd\x9c\x82\x10\xf7\x4b\x9c\xad\x4e\x3f\xc5\xcf\x25\xa2\x35\xa2\x70\xe0\xbb\x7f\xe9\x73\xe0\xd8\xda\xd8\xb0\xc2\xdd\x27\x58\x38\xb4\x0c\x6f\xf5\x7c\xd3\x64\x7c\x48\x67\x24\x0b\x27\x56\x19\x84\x8d\x63\x95\x08\xce\x0e\xfc\x42\x19\x2a\x5d\x89\x3d\x34\xa5\x9f\x6d\xa3\x45\x7c\xbb\x7b\x35\x08\x78\x68\x88\xa7\x9c\xe3\xe5\x33\xb5\x81\xd8\x70\x03\xdf\xaf\x14\x22\xb9\xd4\x22\x51\xa4\xc5\x25\x41\x22\x2e\xf4\x4d\x01\xdd\x16\x5e\x88\x75\x22\x83\xc2\x42\x2f\xca\xb2\x57\x66\xd5\xe9\x2f\x6a\x90\xc4\x58\x91\x23\x2d\x7a\xad\x49\x80\xb6\x39\x32\x90\x4d\x83\x55\x00\x07\x56\xdc\x3c\x13\x32\xc3\xcc\x85\x66\x37\x0c\xd7\x5d\x79\x3b\xb0\x2a\xad\x02\x61\x48\x0f\x03\xf9\x0a\x52\x7f\x4a\xe3\x90\x19\xac\xe7\xca\x71\xd8\xe8\x97\x43\x58\xb6\x67\xec\x83\x71\x1a\x06\x9b\x67\xf1\x21\x0d\x36\xc1\x52\x21\x3b\xa6\x26\x53\x44\xa0\x22\xbe\xac\x11\xb6\xa4\xdb\xb7\x55\xde\x34\x09\x95\xb5\x58\x02\x9e\x11\xe9\x70\x53\x0c\x4a\x8c\xd6\x69\x9c\x20\x6c\x4a\x31\xfb\xb3\x6d\x6b\x32\xbb\x5b\x22\x64\x26\x10\xa4\xbf\xdc\xf4\x31\xea\xb3\x25\xbc\x2c\x17\x97\x55\x5a\x2f\x73\x27\xe1\xe4\x19\x2f\x24\xca\x84\x81\x96\x31\x91\xfb\x6e\xf2\xa0\x81\x95\x3f\xf2\xa1\x10\xca\xa5\x4e\x20\xb0\xc5\xac\x0f\x9a\x73\x72\xef\xf8\x05\x5c\x79\x95\xa8\x72\x2f\x90\x17\xcd\x15\xb6\x8a\x16\xac\x4e\x91\x71\x34\xc7\x6c\x46\xc6\xce\xc8\xba\x8d\xb6\xa4\xdb\x39\x83\x66\xce\x6d\x2b\xf5\x97\xd3\x8d\x51\x98\x6c\xfd\x17\xf3\xaa\x37\x20\xea\x43\x20\x15\x9e\x11\x64\x46\xd4\xca\x2c\x5d\x8a\x18\xb3\x60\xc3\xa0\x27\xd8\x56\x07\xe5\x28\xfa\x26\xe1\x1d\x42\x9f\x2e\xf0\x84\x24\x6f\x13\x39\x01\x5d\x5b\xe3\x3c\x78\xeb\x4c\x36\x00\x41\xcf\x60\xcf\xaf\xb0\x0c\x6b\xbd\x17\x79\x5d\x6e\xc0\xaa\x79\x96\xaa\x9f\xef\x30\x51\x57\x2b\x64\x9b\xa9\x36\x55\x10\x09\xaf\xbd\xa0\xd2\x46\x9d\x81\x2d\xbc\xfe\xaa\x36\xe5\xed\x06\x12\x14\xfc\x68\x18\xc7\xce\x15\x3f\xd6\x4e\x65\x6b\x90\x81\x96\x55\xf0\x86\x53\xc4\x38\x23\x88\xca\xe2\x65\x55\x4e\x87\xf2\x10\x3d\x5a\xc4\x37\xc6\x17\x5f\xa5\xcb\x17\x5f\x7a\x69\x4b\x4b\x01\x9e\xe0\x6d\x03\x2e\xbf\x9b\x11\xad\xa8\x62\xb1\x00\x88\x4f\xc3\x87\xcb\x32\xdd\xda\x71\xee\x5d\xe0\xbe\x77\x08\xae\x41\xa4\xae\xe2\x08\xc4\xc8\xca\xe0\x90\xc1\x41\xb5\x2f\xd9\x8f\x2c\x4c\xcd\x88\x79\xcb\x06\x10\x22\x95\x28\xc5\x19\xf8\xf4\x18\x57\xc5\x57\x06\x76\x49\xf9\x2d\xec\x39\x41\x5c\x9a\x1a\x5a\x0d\x2b\xb0\xce\xb4\xe3\xae\xdf\x62\x5d\xcb\xf0\x96\x0e\x9a\x77\x46\x9f\x08\x73\x34\xdd\x73\x67\x42\x0f\xca\x75\x9a\x2c\x8e\x30\x84\x19\x93\x38\xf4\x7c\xac\xe6\x48\xc6\x20\x73\x08\xf6\xc8\xf6\x4b\x76\x5f\x1b\x46\x63\x40\xd2\x4a\xe8\xf6\x2e\x30\x3c\xa4\x52\x8b\xdb\x6b\x32\xc1\xb1\x44\xbf\x63\x5c\xfd\x2e\x40\x36\x76\xc6\x0b\xf8\xd4\x99\xa0\x7a\x4b\x25\x5b\xe0\xd0\x5a\xc2\x41\x38\x40\xd8\x5a\xbb\xf2\xbb\xc6\x06\x14\x81\xef\x2f\x2a\x8d\x0e\x96\xb3\xe0\x9a\x6a\x5e\x75\x1e\x7b\x54\xbd\x16\xaa\x06\x4f\x53\x56\xaf\x38\xe9\x25\x43\xa7\x5c\xe7\xa2\xf7\x7b\xd1\xca\x35\xbf\x84\x08\xb0\x0b\xb5\xa5\xad\x23\xa7\xd6\x80\x20\xd7\xdb\x25\xb6\xc9\xf3\x6c\x92\xcb\x45\x39\x74\xcd\x96\xc1\x68\x40\xf9\x3d\x1e\xb1\x8f\x5c\xd8\x2b\x58\xda\x3a\x03\x13\x1c\x3d\x1e\x11\x16\x23\x9c\xab\xb9\x41\xdb\xb5\x7e\x85\x85\xa5\x06\x2d\x69\x00\xd9\x78\x28\x0d\x2a\x23\x2c\x62\x57\xf1\xe2\x89\xbb\x51\x8c\x58\xd0\x08\x54\x32\x80\x42\x4f\x50\xaa\xb6\x49\xd5\x24\x52\xeb\x57\x4d\x6b\x51\x57\x84\x75\xa9\x04\xeb\xea\x73\x56\x2a\x2a\x0b\x35\x18\x20\xc0\x89\x4f\x97\x57\x67\xe8\xac\x8d\x4e\xbf\xd3\xf4\xbc\xec\x85\xe8\x59\x8d\xc2\x98\xa4\xec\x0c\xb4\xa4\xf3\xad\xe3\xb5\x25\xd4\xe0\x69\x2e\x20\x5c\xb7\xae\xcd\xaf\xa3\x39\x4d\x0a\xdf\xc5\x37\x3d\x3f\x4c\xdd\x64\x42\x9e\x48\x62\x30\xeb\x23\x01\x91\xf9\xc6\x6a\xf8\x2d\xfa\x3f\xa6\x30\x29\xfa\xc3\x88\x7d\x02\x36\x9c\x24\x0b\x40\xd4\xf4\x2d\x63\x55\x69\xe6\xb1\x76\x00\xca\xa6\x02\xa1\xf2\x40\xcc\x5e\xcf\xf1\x13\x19\x31\xd7\xcc\xff\x41\x8f\xe8\xf7\xe8\x0f\x4d\xea\x9d\x0b\xb0\x7f\x61\x3b\xc7\xc7\x20\x7c\x3d\xb8\xe5\x2c\xa3\xb4\xfc\xc6\x99\x41\x4a\x46\xc8\x1a\x64\x0d\x0f\x8c\x4d\xd9\x13\x8f\x96\xb2\x38\xc2\x53\x8b\x05\x61\x6a\xcc\x78\x4c\xc6\xa4\xc6\xa5\xb9\x82\x49\x68\x21\xe0\x8a\xc7\x64\xad\x43\xd2\x33\xd3\x9f\xc0\x74\x23\xf3\x89\xdf\x0e\x48\xf0\xf7\xd9\xdc\xde\xfa\x50\xa6\xb4\xfa\x91\x7b\xf4\xd9\x6d\xc6\xbd\xad\x33\xd5\x85\x89\xf6\xe0\x42\xb0\x03\xa8\x77\xe8\x25\x58\x39\xf7\x7a\xf5\x38\x56\x1d\x01\xfa\x65\x3d\x73\x7b\x59\x05\xb8\xba\x50\xfb\x44\xd0\x19\xd5\xf2\x7b\x7b\x87\x2d\x70\xc2\x6d\xbc\x19\x06\x64\xb4\x95\x3b\xa3\x58\x0a\x07\xb4\x72\xe4\xe9\xaf\x70\x42\x4e\x78\x5e\x15\xe0\xed\x02\x50\x19\xba\xfb\xad\xac\xbe\xd0\x7c\x78\x66\x32\x00\xc9\x9c\x9a\x9c\xfb\xfe\xd9\x05\xd2\xa7\x83\xa7\x06\x98\x0a\x16\x2d\x57\x73\x2e\xe8\xaf\x8d\x19\x4a\xcd\x32\x7a\xe1\x69\x2d\x12\xba\xcc\x38\xcb\xd2\x3a\x10\xab\x11\x29\x54\x49\x2b\xa9\xd3\x99\xd0\x24\x07\x0c\x56\xcd\x66\xa7\x79\x62\x0a\x37\x44\x5c\xc4\xa6\x72\xba\x2c\xa5\x8f\x41\x18\xae\x13\xef\xb1\xf2\x0d\x52\x0b\x55\x69\x4b\x43\x18\x0b\xce\x4a\x01\xf4\xaf\x39\xc9\xf7\x94\x81\xf7\xa6\x31\xcb\xf7\x78\x26\x8b\x20\x64\xb3\x36\x9a\x37\x17\xeb\xfb\x0f\x3d\x53\x19\xe4\xac\x3a\xcb\xa2\x87\x80\x32\x2a\xb9\x29\x0c\xba\x91\x45\xe7\xd6\x40\xdf\xef\xc1\xa4\xf3\x1a\xf1\x1c\xcb\x32\x52\x0d\xfb\xb1\xe4\xf7\xe4\x33\x38\xab\x2c\xe2\x85\xec\x24\xae\x86\x40\x45\xfa\x78\x41\x93\xc9\x16\x4c\x6e\x59\xa8\x5e\x19\x15\x5d\x18\x50\x3c\x5b\xab\x49\xa6\x56\x1c\xd2\x2e\x9e\x05\x05\x84\xb8\x45\xf1\xb2\xaf\x81\xeb\xae\x8b\x90\xc7\x68\x29\xc5\x88\xb5\x10\xd7\xe1\x96\x70\x51\xcf\xe3\x37\x30\x40\xd8\x86\xca\x5d\x2f\xfb\xed\x9b\x4e\x84\x61\x49\x87\x7a\x24\x96\xe1\x61\xd6\x1e\x06\x5f\x09\xe4\x6d\x0c\x88\x5e\xb4\x79\xbd\x93\xe1\xc9\x71\x1c\xe1\x68\xde\x38\xa9\x09\xe7\x09\xc1\xac\x49\x7a\xad\x7d\x5c\x3d\x22\x06\xdc\x14\x58\x77\x92\x00\xc2\xaf\x5b\x02\x5b\x15\xb2\x10\xdf\x59\x0c\xc8\xec\x86\x87\x9b\xd8\x40\x37\x50\x45\x98\xb3\xfc\x50\x36\x4b\x48\x75\xad\x2c\x84\x7e\xcf\x76\x92\x44\x79\x12\x94\x85\xcc\x88\xd0\xa3\xd6\x4b\xfc\x44\x98\xd6\x19\xec\x38\x9c\x33\xe3\xd9\x25\x44\xfb\x62\x50\x3d\xdf\xb5\xf3\xa7\x41\xd6\x61\x3c\x62\x70\x70\x79\xf9\xb0\x6a\x5a\x95\x5a\xcd\x08\xed\x52\x5b\x9f\xce\x40\x88\xd8\xf8\x78\xde\x95\xcd\xc4\x1b\x9f\x49\xd3\xf7\x18\x62\x0c\x76\x76\xad\x05\xee\x97\x02\xaa\xc1\x6c\xac\x83\xe3\x7a\x25\x23\x32\x44\x6d\x94\xc3\x4e\x83\xa0\x8d\x26\x38\xa8\x17\xbd\x4b\x8a\xf2\x17\xee\x36\x68\x39\x94\x95\xae\xea\x96\x8e\x67\xb0\x4e\xae\x3a\xb7\x17\x36\x64\xbb\xec\xb2\xf5\xf9\x3d\x45\x98\xa3\x2d\xf0\xaa\x04\x06\x74\x02\xc8\x29\xff\xc9\x68\xd8\x54\x1a\x0b\x98\x2b\x73\x91\x66\x6a\x61\xab\xa2\xc1\xbd\x18\xe6\xf4\x1a\xc4\xb7\x3a\xf7\x70\xf5\x8e\x8c\x4b\x0e\xe2\xba\xce\xa0\x23\x6b\x56\xa8\x6d\xd2\x2d\x74\x88\x20\x52\x41\x6c\x68\x8a\x06\x31\x05\x66\xc7\x38\x69\xb4\x65\xed\x81\x69\x42\x9a\x6d\x81\xd2\x60\xc1\x5f\x95\xc8\x89\xe6\x5d\x38\x49\x2a\xf3\xc2\x90\x0e\xad\x7c\x91\xb9\x49\x51\x09\xb7\xbd\xb3\x3a\xc1\x13\xb2\x91\x7b\xfa\xc2\x7c\xb0\x92\x8a\xe0\x15\x88\xec\xce\xb2\x64\xd1\x2e\xa2\x3c\x0c\xbd\xab\x05\x49\x5b\x37\xb0\x10\x5a\x6d\xe5\xdd\x54\x86\x27\xdb\x6e\x88\x92\x44\xb9\xa0\x6a\x31\xb6\x46\xbf\xf6\x4c\xeb\xce\x7e\x79\x66\x3f\x6c\xa3\x51\x9f\x22\xd7\x9f\x33\x32\xc2\x3d\x25\xa8\xa9\xa0\x63\xa7\xd0\x66\xbb\xb5\x96\x5c\x0b\x9e\xb4\x6a\x61\x1d\x7a\x53\xbb\xa1\xea\x2e\xb6\x1d\x9e\xad\xcc\x31\xe6\x53\x87\x8b\xd4\x7e\x61\xab\x25\x4b\x36\xb0\x96\x3a\xf8\xe5\x4c\x50\x2e\x6c\x65\x90\x36\x41\x6d\x29\xfe\x3c\xce\xb0\xc0\x49\x42\x12\x2a\xd3\xed\x6d\xbb\x7f\xfa\xe3\xca\xd1\x9e\x99\x0a\x36\xd2\xd6\x83\xfa\x4c\xd3\x3c\x45\x2c\x4f\x27\x56\xca\xc5\xf2\x31\x04\xbf\x74\xa9\xfa\x06\xc3\xc9\x0d\xb0\x04\x18\x20\x02\x38\xd3\x11\x0b\x80\xad\xad\xa9\x02\x47\x73\x4a\x9e\x00\x76\x53\x30\x22\xe5\x31\xba\xe2\x8a\x9c\xa2\x4b\x9c\xdd\x83\xa0\x66\x4a\x4a\xce\x8c\x75\x1c\x4b\xa4\xa5\xd6\x9c\x51\xd5\x1b\x31\x8b\x86\xed\x56\xe5\x24\xe2\xcc\x20\xa2\x46\xb0\xb0\xbe\x09\x30\xf7\x3a\x68\x50\xe5\x12\x1b\xa9\x6c\x58\x6c\x81\x9f\xc7\x41\xf4\xea\xd8\x64\x07\x6c\x40\xc7\xb7\xf8\xd9\xc4\x6b\x9f\x63\x85\x4d\xb5\xd8\x55\x92\xbb\x0d\x88\xb2\x15\x84\x0c\x10\xb0\x0b\x1c\xe1\x16\x8d\xc2\xd7\x3e\x33\xd1\xa9\x5f\xd3\x63\x72\x8c\xbe\x4b\xf8\x44\xf6\x90\xf4\xa0\xd9\xf0\x50\x12\x25\x7b\xc6\x41\x05\xff\x36\xa9\x60\xdf\xb8\xd5\x2f\xf8\x3e\x94\xfd\x9b\xd2\xcf\x06\x04\x43\xfe\xe9\xf4\xe4\x24\x5d\x1c\x4d\xf2\xe8\x91\x28\xfd\x17\xc8\x14\xb5\x2b\xe4\x10\xa4\x70\x1d\x1e\xd5\xba\xd5\x59\xc6\xb2\x6a\x45\x91\x36\xad\x46\x12\xc0\x4d\xd7\x57\xba\x2f\xac\xea\xa0\x8f\x38\xab\xaf\x1a\x69\xa7\x2c\xf2\xa6\xe3\x55\x02\x5c\x7e\x1d\x6d\xc5\x14\x8e\x0d\x71\x9e\xa7\x09\x9e\x55\x54\x96\x0d\x94\x94\xeb\x94\x5a\x2a\xd2\x73\x87\x78\x0b\x7d\xca\xca\x51\x66\x5f\x39\x77\x24\xb8\x15\xad\xbb\xe5\x78\xc4\xfa\x12\x3d\x13\x53\x0f\x16\x72\x12\xc1\x3b\x91\x53\x39\xf7\x19\x89\x60\x2f\x85\x46\x0d\x1c\xae\x41\x4d\xb0\x8a\xa3\xd3\xac\x9c\xff\xc6\x6a\xa0\x38\x91\xa4\xa7\x1b\x06\x48\x34\x17\x48\x88\x9e\x05\xce\x32\x22\x46\xcc\x42\x9b\x02\x80\x37\xe7\x36\x48\xa4\x29\x9a\xbc\xd3\x28\x5f\x57\xa3\x0c\x93\x3e\xca\x09\x8b\xeb\xce\x37\xe4\x37\xae\x4c\xf5\x58\x91\x1b\xa8\x65\xd1\xb6\x91\xde\x6f\x6f\x36\x6e\x39\xe6\x75\xda\x79\xbf\x12\xa6\x0f\xe5\xa6\x53\x50\x20\x65\x51\x55\xd3\xd9\xfa\xbc\xfa\x5e\x12\x73\x00\x19\x1b\x3e\x8e\x39\x91\x81\x11\x1f\x79\x5b\x5c\x42\xa7\x44\x4b\x1f\x23\xa6\xc9\x38\x74\x38\x18\x80\x6d\x87\xb7\xad\x3b\x8d\x04\x97\xd2\x46\xde\x9b\x76\x56\xe7\x4f\xed\x50\xcb\xcf\xa0\x84\x0f\xaf\xaf\xc6\xcb\x55\xfd\x82\x67\xae\xbe\x9f\x7d\x58\x9b\x64\xdf\xd8\xd4\xda\x6a\x7e\xc5\x5a\x6c\x50\xcf\xef\xe4\xec\x62\xe8\x8b\x58\x55\xba\x5e\x2e\xe8\x17\x22\xab\x37\x97\xf4\x5b\x9e\x71\x50\xdc\xaf\xd2\xc4\x8a\xf2\x7e\xeb\x37\xab\x1c\xef\xbb\x0b\x6c\x5e\x65\xeb\xd7\xf2\x87\x32\xcd\xac\x0b\x4b\xdf\xd3\x36\x35\x5c\x2b\x11\x08\x8c\x2f\xed\x61\x07\xc1\x4b\xbf\x25\x15\x4e\xb3\x30\xe5\xd2\xe1\x86\xda\x69\x9a\xa3\xd6\x74\x09\xbe\x2a\x9e\x79\x84\x4d\x34\x4b\x75\x70\x4b\x5b\xb1\x99\xc7\xeb\xde\xc2\xa4\xef\x23\x8c\xf9\xf5\x72\x98\x93\x45\x11\xb5\x27\xad\xec\xe6\x4a\x70\x37\xd8\xfd\x27\xc4\x43\xc2\x37\x6e\xe8\xae\x49\x8a\x1e\x3a\x4a\x10\x2c\x6d\x38\x06\xe4\xf2\x55\xf2\x7c\x36\x30\x0f\xfb\x31\x9b\x6c\xe0\x23\x5f\x84\x21\xb8\x6a\x6c\x5d\xb1\xc8\x1d\x44\x2a\x04\x79\x22\x02\x68\xc7\xc6\xfc\xb0\xf2\x51\xc5\x89\x20\x38\x5e\x04\x2b\xe2\x03\x0e\x4c\xcf\x60\x1e\x93\x34\xd5\x0a\x3c\xa8\x26\x8c\x1f\xf1\xcc\xe9\x2c\xa5\xb7\xa0\x82\x06\x9d\xea\x1b\x2b\x08\x57\xd0\x5f\xb0\x23\xf2\x99\x4a\xa5\xe5\x8a\x9a\x58\x4d\xd7\x08\x48\x3c\x50\x57\x6b\x4e\xec\x0d\x37\xfa\xd0\xff\xee\xfa\xf6\x7e\x70\x3e\xfa\x50\x44\xe7\xbb\xf4\x33\x8f\x08\xe5\x00\xfe\x39\x1b\x31\x1f\x50\xeb\x01\x90\x61\x2f\x11\x8e\xe3\x02\xd9\xc0\x2a\x91\x46\x66\x5b\xc9\x91\x83\x53\xb1\x36\x94\x76\x45\x33\x0f\x90\x83\x74\xa8\x27\x6b\x85\xeb\xac\x74\x72\x4c\x26\xd5\x8a\x94\x97\x3d\x5d\x36\x21\x76\xab\x32\xba\x36\x51\x0e\x5c\x90\x91\x67\xa7\x2b\xc1\xed\x7c\x82\xcd\x25\xbc\x19\xb7\x73\x1b\xb2\xc5\xa6\x7e\xa4\x9f\x49\x7c\xdb\x20\x55\xed\x25\xa3\xa5\x55\x24\x60\xed\x2e\xe4\x8c\x6e\xa2\xf1\xfb\xa9\x3c\xe8\xef\xda\xb3\xa5\xeb\x02\x92\xad\x80\x57\x05\x6c\x55\x85\x30\x8a\x88\x50\x98\x32\x34\x85\x83\xcd\xa2\x05\x02\xc0\x0e\x02\x3e\xec\x3f\xa2\x94\x32\x40\x0e\x58\xb5\xb4\x0f\xe5\x79\x6c\x20\xb4\x5e\x0e\xaf\x1e\xee\x4b\xa2\xea\xf7\xd7\x0f\xe5\xa2\xee\xfd\x9f\x57\xca\xaa\x95\x16\x56\x05\x0b\x05\x53\x2c\xb2\x10\x2d\xca\xac\x5f\x99\xda\x89\x26\x0b\x45\x1e\x6e\x2f\x76\x92\xef\xea\x9d\x65\x8d\x18\xe1\xa1\x74\x55\x8f\x88\xd0\xe6\xd3\x98\x44\xeb\x50\x4c\xdb\xd3\x91\x89\x82\xd2\xeb\x60\xad\x89\x16\xe1\x0c\x4b\x94\x61\x61\xfd\x50\xb1\x09\x80\x2a\x57\x06\x33\x9a\xd7\x2a\x04\x89\x4f\x44\xfd\xa8\xaf\x3e\xce\xf6\x91\x05\x61\x45\x59\xf0\x8f\x92\xf1\x93\x69\x78\x83\x93\x66\x87\xb2\x22\xd5\xc5\x09\xcb\xd0\x03\xb2\x3d\x84\xb8\x0b\xc7\xa6\x42\x7c\x5f\x37\x07\x2b\xe2\xe2\x09\xb5\x4a\xca\x99\xa6\x48\x03\xa7\xea\x30\x58\x83\xe6\xf8\xd4\x7c\xdc\x12\x91\x2e\x88\x6a\xd7\x6d\x15\x4b\x89\xfa\x37\xc3\x9a\xb5\xbe\xa8\xba\x90\xbe\xac\x72\x36\x89\xf7\x66\xed\x1b\x24\x29\x48\x4f\x3c\x08\x54\x24\x3b\xd3\xdd\x60\x90\x8c\xd3\xff\xa6\x1c\x49\x70\x08\x68\xbd\x75\x2a\x43\x29\xed\x78\x0d\x30\xef\x66\x99\x78\xc5\x32\x6c\x08\x7a\x14\x0e\xc8\xa6\x81\x84\x40\x3f\xcb\x31\xc6\xbd\x10\xf8\x87\x9b\x82\xb9\x36\xb6\x60\x6f\x60\x48\xc5\x6c\xda\xa0\x21\xfd\x68\x28\xda\x83\x65\x00\xfc\x87\x2b\xc8\xe8\x62\x83\x6d\xee\x7a\x38\xdd\x90\xda\x36\x03\x50\x2a\xc6\xe7\xcc\xdf\x16\x8b\x1a\x67\xd8\xda\x1d\x40\x89\x72\x95\x12\xea\x0a\xeb\x1d\x8f\x58\x10\xb0\x22\x8d\xda\xa3\xcf\x88\x2b\x4e\x02\x15\x6f\x19\x00\x5b\x43\x92\x8e\x17\x7e\x4a\x3b\x50\x4d\x91\x57\xf3\x72\x79\x91\xa5\x7e\xec\xe9\x94\x73\xec\x12\x11\x9d\x05\xc5\xc6\x01\x86\xf6\x25\x68\x2f\x28\x28\x60\x3b\x06\x73\x34\x18\x2d\x70\x50\xae\x2e\x48\x5e\x8f\x39\x91\xec\x2b\xe5\x53\x3d\x69\x62\x4b\xa2\xe0\xaa\x7b\x40\x4b\x75\x98\xda\x96\x57\x1f\xf0\x3d\xa0\x33\x6d\xaa\x38\x04\xc7\x6a\xad\x99\xca\xf9\x78\x81\x12\xc2\x58\x24\xe8\xb4\xc9\xaa\xfe\x39\x23\xd1\x36\x10\x32\x37\x58\xe0\x94\x28\x22\x56\x85\x23\x95\x8b\x49\x83\x88\xe3\x76\xd0\xf6\x6b\x76\xd1\x54\xda\xa8\x96\x64\xf1\xda\xed\xc5\x3a\x48\x18\x3f\x8b\x8d\xd0\xaf\xf4\x34\x7e\xb4\x96\xff\x0d\x67\x61\xfb\x29\xa6\x61\xa3\xad\x02\x04\xa0\x5d\xe7\xf4\x3a\x50\x28\xf7\x4b\xa0\x22\xa5\x70\xa1\x03\xc1\x40\x59\x3f\xca\x26\xf0\x93\x75\xbc\x74\x2f\xbc\xdb\x65\x38\xb8\x14\xda\xca\xa1\x2a\xe5\x4e\x00\x95\x80\x4a\x65\x70\x40\xea\x01\x4c\x40\x68\xa9\x8b\x90\x0c\xdc\x7e\x16\xde\xae\x30\xe8\x5a\xc9\xaa\x5a\x5c\xaa\xb2\x5c\x6b\x78\xdc\xbe\xc0\x1d\x3a\x89\x66\xdf\x12\xcd\x3a\x52\x2e\x45\xd7\x6a\xea\x24\xa2\x82\x33\x63\x8b\x3e\x5b\x80\x80\xf2\x04\x21\xf7\xc8\x5e\x91\xb6\x72\x2c\x5c\xfd\x94\xf9\x7f\x95\x39\xb8\x23\xea\x90\x54\xeb\x92\x2a\x8f\x03\x17\x14\x78\xa0\x92\x50\x1a\xb0\x71\x35\x30\x5a\x13\x06\x69\xac\xfc\xc3\x2b\xe3\xc0\x82\xe4\xe6\x05\xcf\xd1\x33\x95\x73\xa4\xf8\x88\x41\x9c\xa0\xf7\x06\x28\x8e\xcc\x8b\x3d\x78\x0b\x60\x10\x64\x3e\x49\xa9\x42\x38\x98\x61\xc9\x24\xd9\xb3\xe7\x59\x7f\x00\x33\xae\xcd\xb3\xaf\x83\xe8\x59\x73\x68\xb6\xb0\xaf\x15\x8d\xec\x9a\x4a\x1f\xc4\x34\xbf\x6c\x32\x7d\xa0\xf1\x84\x1a\x66\xed\x99\xeb\xb2\xe9\x51\xbd\xb5\xc1\x82\x86\x02\xb2\x2b\x95\xaa\x72\xb7\x58\x43\xcf\x9a\x4c\xfa\x62\x23\x5a\xa5\xd2\x17\xaf\xef\x23\x97\xbe\xa9\x4c\xd9\xaa\xdc\x4a\xf7\x49\x83\xfd\xdb\xe5\xec\x2a\xee\x02\xe7\x43\x49\xe9\xa6\x51\x52\x3a\x34\x54\xb3\x22\x21\x60\xfb\xf0\xf2\x4d\xd4\xc1\x22\x3f\x2b\xa4\xa2\x20\xdd\xb2\x8c\x09\x43\xaa\x9c\x9f\x71\x05\x39\x35\x11\x94\x70\x5f\xca\xf3\x1c\xb1\x7a\x09\x64\x35\x4f\xdc\x35\x45\x63\xaf\xe8\x67\xc1\xf9\x73\xb3\xb0\x16\xad\x9f\x7c\x90\x9b\x51\x96\x6d\x31\xf6\xaa\x88\x59\xb8\xf8\x9a\x82\x93\xb4\xe0\xb1\x4d\xc2\x71\xcd\xa9\xac\x1f\xfa\x52\x02\xc5\xda\x73\x61\x2f\xdd\x3d\xaa\x76\x4b\xdc\xb9\x75\xbe\x89\x97\x91\x2d\x37\xb6\x01\xd3\xb1\xd7\xe3\x2b\xae\xda\x6d\xaa\xcc\x02\xaa\xe8\xde\xb0\x50\xab\xf0\x04\xba\xf1\x1e\xb8\x77\xed\xd8\xb1\x09\x75\xf1\x08\xdd\x95\x3d\x29\xcd\xd8\x56\xef\x7f\x89\x59\x6f\x5a\x1a\x38\xf0\x37\x0a\x1b\xef\x4b\x43\xcb\x01\xd4\x04\xb6\x61\x92\x15\x4e\xec\x85\xbb\x9c\xc5\x44\x30\x82\xd5\xfc\xf5\xb2\x2c\xce\x76\x35\x4f\x07\xe3\x7b\xd9\x8c\x8b\xb3\xbd\xd4\x85\x0f\x87\x5b\xae\x91\xbe\x76\x9c\xfa\xf5\x36\x96\x22\x1b\x7c\xe0\xab\x04\x2f\xa9\x8e\x35\x66\xc3\x00\x84\x66\x13\x2a\xdd\x29\x11\xa3\x5e\x9d\x7b\x99\x94\x94\x1a\xbb\xcf\x52\x32\x8a\x3e\xed\x61\x6d\xe5\x35\x4b\xf2\x45\xe4\x7e\xbc\x7c\x3a\xc2\xaa\x2a\xce\x79\x90\xa1\x00\xa5\xb4\x15\xa6\xcc\x72\xaf\x55\x49\x09\x5a\xa6\x4c\x71\x5d\x1e\xc2\xc1\x67\xb8\x7c\xf1\x09\x2e\x5d\xba\x43\x97\xee\x50\xb3\x47\x5d\xba\x03\x42\x87\x96\xee\xb0\x4e\x05\x5d\x65\x00\xf5\x3e\x39\xa8\xb6\x59\x2a\x71\x63\xf6\x77\x8d\x1e\xb9\x7d\x48\xbf\xb3\x21\x86\xf1\x50\xf6\x17\xfb\x43\x6d\x48\xd4\xd2\x67\xd5\xd9\x86\xf6\x4c\xb6\xa8\xba\x05\xb0\x88\x13\x8b\x43\x67\x03\x96\xcb\xf6\xa7\x55\xa6\xd2\x11\xfb\x9e\x3f\x93\x27\x22\x7a\x08\x2b\x94\x72\xa9\x80\x0f\xbb\xf8\x18\x38\x08\x25\x48\x73\x13\x07\x81\xd1\x15\x4e\x49\x6c\x2a\x1e\x06\x61\x8d\xd6\x60\x6b\x5d\xad\x75\x70\xab\x80\x1c\x6a\xb6\xc1\xc5\x4d\x8c\x98\x09\x35\x34\xe1\x6d\x20\x2b\x50\x37\x31\x20\x98\xdf\x79\x47\xf0\xef\x8e\xd1\xbd\xbe\x9f\xa8\x2c\x8f\x37\x40\x5f\x6b\x1a\xdb\x88\xcd\x04\xcf\x33\x6f\x43\xe3\x13\x53\xfa\xd6\x44\x3f\x2d\x3b\x82\x61\x30\xce\x0b\x1c\xe1\x58\xeb\xe2\xab\x09\xe7\x4d\xa2\x50\xb7\x82\x30\x0a\x09\x48\x1f\x43\x1f\x5a\x67\x43\xdd\x8d\xff\x36\x00\x6e\x59\x05\xc4\xfe\x42\xce\xe5\x73\x22\xc1\x2a\xe4\xad\xee\xa5\x3c\xf2\x32\x56\x41\xed\x38\x57\xd9\x44\xbd\xdf\xc2\xd9\xf6\xeb\x61\x10\x8a\xce\x6d\xcc\x97\x49\x52\xb5\xf7\xc4\x8b\x59\x4b\x5b\x47\xcf\x36\xf1\x8b\x9b\x5c\x64\x1c\x24\xb1\x64\xe1\x60\x1b\x2c\xd2\x5b\xc6\xb3\xdc\xc4\xb5\xd1\x30\xcc\xa9\x96\xb2\xa9\x54\x97\x58\x45\x73\xcd\xb9\x0b\xc4\xb3\x3d\xc5\xfb\x15\x5c\xf9\x65\x2d\xa8\x35\x33\x38\x0b\x7b\x6f\x70\x29\xb4\xb0\xa8\x9b\x7b\xdf\x85\xd7\x3b\x49\x22\xd5\xfd\x19\xb7\x9b\x2d\x68\x1d\xd8\x45\xdd\x27\xf6\x89\x9e\xe8\x3a\x2a\x5a\x37\xfe\x76\xb4\x55\xae\xb8\xb5\xf7\x48\xc2\x1d\x20\x64\xce\x2d\x60\x57\xf1\xa2\xad\xd0\xda\xe0\xfe\x17\x74\xbb\x2c\x20\x8b\x92\xff\xa4\xc5\x11\x6f\x71\x4d\x71\xa6\x95\x08\xc5\xf5\x2d\x29\x66\x46\x8e\x35\x71\xb2\x08\xa3\x5c\x50\x77\xf6\x2b\x39\xe1\xcd\xd4\x01\x16\xca\x93\xb0\xa2\x52\x84\x83\x62\x73\xc6\xe1\x8f\x23\x95\x63\x1f\x98\x08\x34\xe1\x8a\xa0\x9b\xfc\x77\xe7\x58\x17\x4e\xbc\xab\xd9\xd3\xb5\x84\xbd\xc3\x2e\xe3\x3a\x7c\xc3\x56\x27\x8d\xb2\x59\x00\x8e\x58\x6f\x25\x6e\x53\xfb\xa0\xf6\xcb\x76\xf5\x1b\x6a\x3f\x75\xb2\xcf\x36\xdf\xae\x00\x6f\xda\x3a\x36\xbb\x14\xe7\x6e\x03\x61\xad\xf4\x14\xc2\x56\x5a\xfb\x1d\xa0\xcf\x52\x70\xd4\x63\x2b\x4d\xfd\x97\xff\xcb\xd4\xca\x32\x4b\xf3\x5f\x88\x8b\x11\x33\xbf\xf7\x7c\x9d\x0a\xfd\x42\x01\x00\x8b\x53\x52\x40\x64\x8a\x32\x98\x1e\x40\x8a\x58\x30\x34\x03\xf6\xeb\x61\xfa\xf5\x18\x1e\xf3\x09\x11\x8c\xe8\xa1\x39\xf0\x01\xcf\xcc\x52\xcc\xf0\x0c\xa0\x85\x7b\x10\x19\x07\xe2\x6a\xa1\x8a\x18\x92\x36\xf5\x0e\x81\x5b\x69\x66\x69\xf3\x6d\x8b\xba\xbf\xd0\xa7\x11\x65\x2d\xb2\x69\x11\x5e\x51\x4f\xfd\xb7\xb6\xff\xed\x24\xf6\xfb\xfe\xdd\x0f\xe3\xdb\xc1\xdd\xf5\xc3\xed\x59\x49\x6c\x3f\xbb\x78\xb8\xbb\x1f\xdc\xd6\x3e\x2b\x72\x55\xff\xfa\x30\x78\x68\x78\xe4\x1a\xb8\xe8\x7f\x37\x28\x15\xd1\xfe\xeb\x43\xff\x62\x78\xff\xf3\xf8\xfa\xe3\xf8\x6e\x70\xfb\xe3\xf0\x6c\x30\xbe\xbb\x19\x9c\x0d\x3f\x0e\xcf\xfa\xfa\xcb\xf0\xdd\x9b\x8b\x87\x4f\xc3\xab\xb1\x0b\x3b\x0e\x1f\xfd\x74\x7d\xfb\xc3\xc7\x8b\xeb\x9f\xc6\x41\x97\xd7\x57\x1f\x87\x9f\xea\x66\xd1\xbf\xbb\x1b\x7e\xba\xba\x1c\x5c\xad\x2e\xd6\x5d\xbf\x1a\x8d\x75\x80\x83\x8b\x2c\x30\x1a\x05\x62\xd2\x64\x61\x49\x9b\xfe\x0a\xbe\x8b\x1b\x43\x8f\x47\x3d\xf7\x97\x29\xad\x7d\xa4\x59\xa0\xf3\x8b\x15\xdc\x63\xc4\xbc\xe3\xd2\x5f\xaa\x0a\xcf\xa4\x4b\x3d\x2e\x8d\xf6\x14\xf5\xe1\xac\x80\xc2\x50\xea\x14\x32\x1b\xfc\x48\x9d\xab\x1b\xe8\x30\xa1\x29\x05\xaf\x37\x3a\x42\xd5\x0d\x2f\x37\x68\xe7\x04\x43\xb0\x7e\xbb\x78\xd5\x69\x90\xd5\xac\x66\xa0\x94\x53\xe4\x38\x34\x31\xe6\x04\x83\x3d\xbb\x60\x38\xa5\x51\x35\x05\x03\xe0\x57\x51\x01\x35\x52\x6d\xb1\x44\x60\xe5\x96\xe7\x04\xfd\xf0\x97\x62\x50\xe0\xc1\xb0\x9a\x77\xbe\x54\x51\xcf\x3e\x10\xb9\x59\xd5\x75\xe4\x59\xea\xc9\x1d\x73\x6b\x5a\x86\x73\x6b\x2b\x77\x83\xbb\x29\x67\x01\xdc\x58\xc9\xf7\xa4\x8f\xb7\x99\x51\x85\xc6\x4f\xd1\x1d\x40\x9d\xc8\x42\x75\xd7\xbb\x98\x25\xf9\x8c\x32\x44\xd3\x2c\x21\x45\xcd\xf7\x09\x99\xe3\x27\xca\x5d\xf9\x0a\x53\xe5\x03\xd6\xd1\x8a\x56\xe8\x08\x35\x1e\x94\x53\xd4\x8f\x63\x59\x66\x70\x25\xca\x71\x2c\xf3\xa8\x3c\xec\x10\x21\x8c\xc5\x9e\x6d\x56\xe8\xa8\x38\x72\xb0\x62\xfb\x07\x73\x59\x66\x87\xe5\xbb\x77\x87\xeb\x5f\xaf\xe0\xd8\x91\xf2\x78\x2b\x61\xe0\x1e\xcb\x47\xc7\x9a\xd7\x09\x04\x0e\x56\x67\xb7\x1e\x2d\xbe\x4e\xdb\x4e\xfd\xca\x8e\xe1\xa0\x6d\xd7\x67\x23\x2a\xf4\x9a\x2e\xdd\x8c\x93\x4a\xe9\xae\xd6\xfd\x95\x4a\x7f\xd5\x76\xb6\x57\x6f\x4f\xbd\x34\x06\x47\x72\xec\xe9\x7f\x83\x79\xdc\xc0\xa7\xd7\xfe\xcb\x95\x22\xdb\x38\x58\xb7\x4d\x7d\x40\x4b\x49\xba\xd6\x0f\xb4\x92\x0e\xf7\x04\xef\xd4\x5e\x18\x84\xc2\x0b\x34\x02\x77\x1f\xa6\xcc\x96\xe3\x21\xde\x1f\xe5\x0a\x50\xeb\x73\xec\x4b\xc4\xe1\x09\x7f\x2a\x29\x97\x29\x91\x12\x37\x00\x96\x04\x26\xb1\x5d\x18\x83\x3f\xa1\xf6\xc3\x96\xf4\xe4\xce\xe4\xbd\xfe\x6a\x95\xd1\xe7\x36\xd4\x8c\xdd\x44\xb5\xc0\x1a\xbb\x48\x5b\x74\x6d\xf2\xed\x34\x7f\xe9\x15\xc1\x34\x5c\x04\x31\x46\x4d\xee\x9f\x96\x66\xb5\xea\x82\xd5\x56\x59\x0a\x5d\x78\x9b\xc7\xe0\x04\xad\x6f\x8d\x88\x6d\xfd\x2a\xb8\xbc\x3e\x1b\x50\x5d\xc9\xdf\x19\x56\xa0\x8e\x78\x9a\x1a\xb9\xa0\x64\x4b\xed\x21\x6c\xd2\x1c\x0b\x69\x4a\xe6\xd1\xdc\x78\x99\xf4\x95\xd1\x1b\xb1\xe7\x60\x43\x4a\x81\xc0\xfd\xb0\x25\x40\x13\xfd\xac\x8f\x1b\x7d\x2a\x85\x57\x83\xc8\x48\x21\xd6\x37\x20\x04\xe3\x10\x2c\xca\x47\xad\x21\xf0\x60\xbf\x76\x20\xf5\x2d\x6a\x05\x56\xd6\xb7\xa9\x62\xa0\x9f\x5b\x50\xa8\x6f\x07\x4d\xb9\xed\x10\x82\x5a\x81\x75\x23\xd8\x43\xa9\xc0\x57\x85\xf7\xf6\xe9\x9a\x26\xbb\x37\x9d\x58\x8c\x0a\x3d\x5d\xb7\xda\xbf\x77\x33\xfa\xbd\xf1\x3b\xe4\x0d\xa0\x26\x41\x6b\x1e\xe1\x1b\x1d\x69\x99\xd5\x25\xdb\xdb\x40\x0c\x89\x8e\x0c\x6a\xe0\x57\x10\x69\xd9\xbf\x19\x7e\xd5\x43\x5f\x85\xd9\x66\x5f\x6d\x75\x00\xed\xb8\x6d\xb9\x40\xd0\xa6\x4a\x29\x07\xe5\x63\x07\x7b\x55\x39\x89\x76\xcf\xec\x41\x44\x4d\xe7\x50\x7f\x59\xfa\x06\x9c\xd3\x50\xfe\xce\xf8\x6f\x7d\xc0\xb3\x75\x01\x19\x19\x97\xca\x9a\xb5\x8b\x47\x6c\xb2\xa8\x3a\x79\x7a\xde\xcb\xd3\xfa\x94\xee\x5c\xd2\x4d\xb7\xb7\x9c\x9e\xbc\xe7\x40\xdc\xd5\xf7\xc1\x9a\x84\xe7\xbe\x89\xb9\xe6\xd3\x80\x8b\x35\x45\x29\x74\x11\xec\x75\xb3\x2a\xd9\xcb\xdc\x62\xd6\x6e\xca\x3a\xf9\xe7\xbd\x91\x5b\x8b\xb0\xef\x7e\xdd\x8a\xd8\x88\xff\x06\xe1\xba\xa3\xb2\x97\xa5\xb2\x7d\x64\x3c\x94\x07\xb7\xf9\x05\x7a\x66\xe4\xb8\xa0\x19\x67\x70\xd5\xca\x84\x67\xf0\xa5\xba\x7f\xeb\x0b\xe6\x6e\xe8\xf3\x0d\xd6\x64\xbd\xd3\xf7\xce\x04\x0e\x18\xb7\xeb\xf2\x58\xab\x43\xed\x2b\x5b\x84\x88\x53\x93\xdd\xa8\x68\x4a\x7a\x88\xb3\x64\x11\x04\x3b\xd8\xf3\x0a\xe4\x66\x62\x94\xe6\x84\x0a\xd7\x89\xc5\x18\xdc\x28\x1d\x7e\x43\x69\xbc\x89\x46\x76\x88\x34\xb9\xea\x5f\x0e\xce\xc7\x83\xab\xfb\xe1\xfd\xcf\x35\xf8\x91\xe5\xc7\x0e\x42\x32\x78\xe1\xee\xe7\xbb\xfb\xc1\xe5\xf8\xd3\xe0\x6a\x70\xdb\xbf\x5f\x03\x2f\xb9\xaa\xb3\x26\xe8\xc2\x5c\xd6\xa9\x6f\x9b\xc0\x17\x3a\x33\x6f\x4d\xef\xcb\x20\x93\x41\x27\x94\x34\x00\x4d\x9a\xd4\x7f\x16\x13\x81\x62\xf2\x44\x12\x9e\x15\x66\xd5\xda\x05\x0b\x10\x28\x6b\xda\x5f\x85\x42\x09\x6d\x56\xd7\xf8\x14\x99\x5a\x6f\x41\xb9\x5b\xdf\x20\x88\x7c\x58\x10\xf6\x95\x42\xe4\x73\x96\xd0\x88\xaa\x20\x35\x90\x0b\xeb\x5e\x31\xee\x43\x88\x4e\x5d\x43\x5c\x7b\x8b\x46\xd9\xbb\xce\x1f\x7a\xd2\x97\xb5\x7d\x7f\xa2\x3c\x22\xda\xda\x02\x42\x7b\x50\xec\x1b\x9c\xc6\x4b\x80\x6d\x5b\x8c\xee\x25\xcc\x03\xcb\x39\x3a\x36\xbd\xaf\x01\xcc\xad\x7e\x90\xeb\x6f\xc3\x55\x71\x32\xa5\x73\xbd\x3a\x50\xa6\x1d\xa5\xbe\x71\xb8\x4b\xa9\xb0\xe6\x1e\x90\x37\x6c\xec\xfa\x86\x01\x0b\x4b\xf5\x62\x98\x89\x39\xc5\x48\x90\x94\x2b\xad\x80\x99\x88\x80\x9e\x16\xaa\x28\x4e\xe8\xaf\x80\x51\x25\xc8\x71\x10\x41\xe1\x90\xbd\x0a\xf7\x81\xc5\x8f\x38\x1e\xb1\xf3\xc1\xcd\xed\xe0\x4c\x33\xa4\x63\xf4\x20\x01\x7e\xaa\x34\xf5\x73\x4b\xde\x46\x1c\x0b\x23\x19\x28\x93\x8a\xe0\xa6\x60\x30\x22\x04\x17\xed\xf9\x83\xef\x6f\x00\xdf\xd5\x93\x37\x3c\x2b\xd9\xa6\x9c\x01\xe0\xaa\xb1\x2a\x72\x90\x33\xb0\xf7\x9c\xac\x5b\xfc\x5c\x5a\x91\x10\x7e\x03\x24\x91\xf2\xaa\xbf\xe0\x6a\x03\x80\x67\xfb\xf9\x95\xfa\xbc\x81\x6f\x57\xcd\xf3\x1e\x42\xec\xa4\x2a\xd0\x40\x0d\x60\xa8\xaf\x7a\x53\x99\x67\xa3\xa8\x28\xde\x02\xaa\xa3\x42\xfa\x13\x32\xc3\x0c\x89\x9c\xb1\x0a\x3c\x6c\x68\x69\x5b\x0e\x9a\xd9\xf4\xa8\xea\x35\xc3\x29\xcf\x19\x28\x0d\x10\xc6\x5a\x33\x18\x99\x11\xa6\xd6\x0c\xe6\xad\x80\x58\x2a\x43\x3d\x5c\x2c\x96\x9a\x81\x36\xc1\xb1\xd4\xf9\x93\xa0\xf4\xf2\x66\xd7\xb2\x0b\xca\x2b\x39\x95\xf4\xa1\xf2\xf7\x73\xbd\x96\x8d\xe5\xe3\xce\xdd\xdd\x63\xf9\xb8\xbe\xab\x98\x44\x8f\x9b\x5e\x36\xd5\xcc\xcc\xc4\x56\xae\x5e\x32\xf6\x2d\xf4\x53\x5b\x9a\x05\x0a\x96\x47\x8f\xe8\xfb\xfb\xcb\x0b\x34\xa5\x5a\xee\xd5\xd7\xca\x15\xd6\x32\xf6\x83\x48\x9c\x5d\xd8\xda\x56\x73\x91\xf8\xbb\x17\x36\xde\x89\x52\x81\x94\xa0\x6f\x34\x3c\x23\xce\xd8\x2b\x2c\xda\x5e\xa5\x34\x8b\xc0\x2c\xe6\xa9\x99\xc7\x89\xcc\xa7\x53\xfa\xf9\x58\x61\xf1\x4d\xc3\x7a\x98\xa8\x8a\xf1\xdf\xf9\x64\xac\x47\xb4\xe3\x45\x5c\xd7\x1c\xb2\x05\x95\xfd\xb2\xd9\x99\x9d\x9b\x77\xff\x2f\x9f\x40\xb6\x7b\x26\x38\x20\x00\x82\x77\xce\x46\x2a\xd8\x57\x1c\x25\x1d\x23\x97\x40\x55\x02\x39\x89\xb8\x10\xc4\x26\xc9\x9b\xda\xa2\x19\x16\x8a\x82\xb5\xd6\x81\xa4\x94\xd0\xf1\x8b\x2d\x0a\x4b\x7e\xcf\x71\x81\x44\x3d\x21\x04\x1c\x3c\x19\x4d\x36\x53\x7a\xcf\x4a\xbe\xc9\xca\x09\xb4\x91\xa7\x16\x37\x13\x0c\x32\x6b\x45\xac\xc1\x13\x61\x6a\x2f\xfa\x09\x34\x51\x93\xb6\xdf\xce\xcb\x60\x4a\x7c\x0e\xcf\x8b\xcb\xcd\x85\xf4\x86\x51\x4d\x4a\x60\xb8\xe7\x6d\xa2\x94\x75\xa9\x37\x39\xfa\x9f\x5a\xfb\x8e\xe1\xd5\xe5\x75\x59\x13\x1a\x6f\x57\xbb\x28\xf5\x5d\x84\xb5\x3a\x68\xff\x2d\x81\x7c\x24\x31\x56\x8c\x00\x40\xc2\x2a\xa7\xd5\x3d\x37\x7d\x6a\xda\xaa\x74\xb9\x76\xcb\xb7\x40\xad\x29\x35\xf3\x89\x40\x4a\xe7\x3e\x02\xd1\x37\xc9\xdd\x87\x81\x3c\x88\x04\x42\xa8\x57\x5a\xb1\x4c\x99\x71\xcd\xf9\xbc\x64\x87\x5b\xc8\xe8\x66\x30\x5a\x68\x24\x99\x20\x91\xbe\xca\x4e\xd1\x4d\x42\xb4\xe4\x95\x6b\xe9\x2b\x4f\x12\x87\xf0\xb5\x5a\x3a\xdc\x08\x95\xee\xc5\xe7\x15\xe8\x1e\x2b\x26\xe6\x10\xee\x56\xcf\x2c\x58\x83\xfd\x23\x2e\x04\xeb\x0b\x26\x64\x30\x24\x96\xb5\x48\xe0\xf0\x0b\x13\x37\x0b\xa6\x24\x5c\xba\xc8\xe8\xaf\x9a\xfd\x0a\x22\xe7\xbc\x31\xc9\x31\x9c\xed\xcb\xcc\xc1\x2d\xe5\x0b\x4e\xc2\xdd\x87\x4d\x71\xd5\x2d\xe4\x9a\xca\x1d\x58\x12\x71\x56\x4d\xd1\x57\x7f\xf0\xd1\x1f\x16\x6f\xd5\xde\xad\x76\x68\x70\x4b\x16\xa6\xb6\x10\xfb\xac\x70\x5d\x14\xca\xcc\xc2\xf8\x5e\xfd\xe7\x85\x01\xb9\x48\x09\xa0\x4a\x16\x55\xe7\x90\xbe\x6b\x9b\xb6\x58\xcf\x73\x9c\x8b\x8d\x20\x29\x0a\xd4\xf2\x4d\x38\xb7\x4d\x46\x29\x86\xa5\x17\xa1\x9e\x5d\xda\x62\x12\x20\x46\xdb\x50\x23\x59\x42\x82\xb3\x64\x63\x96\xb1\x56\xc5\x6b\x66\xca\xbb\xba\xd5\x40\x4a\x2e\x44\x99\x97\xf2\xae\x95\x28\xb0\x34\x81\x0e\x5b\x6c\x73\x6c\x31\x5b\x59\xc4\xd3\x1e\x20\x01\x2a\x01\x89\xff\x85\x03\xad\x2a\x38\x58\xa3\xf7\xba\xcc\xa7\xd2\xee\xb4\x4a\x73\x2a\x7d\xa1\x79\xc9\xf9\x8e\x1e\x38\x3d\x99\xc5\x18\x12\x47\x77\x89\xc2\x29\xcd\xdf\x78\x0f\xa0\x4d\x12\x23\x83\x5e\x60\x90\x8f\xed\xda\x79\xcf\x49\x86\x05\x61\x6a\xc4\x6e\xf5\x28\xcc\x17\x45\x24\x86\x8b\xc3\x71\x68\xf4\x50\xb3\x76\x8a\xb0\xfd\x0a\x16\xbd\x29\x10\x4e\x8e\xcd\x4b\xa0\x9a\xbe\x60\x92\xfd\x77\xe6\x1d\x83\x79\x60\x31\x7f\xf4\x54\xe9\xb4\x50\xe3\xb5\x00\x19\xcd\x29\x40\x0e\xc4\x44\xda\x0b\x89\x2a\x8b\x29\xe1\xc5\xef\x9c\x38\xfc\x65\xf8\xcc\xf3\xaf\x3a\x86\xed\x0c\x05\xcc\x19\xe8\xe4\x88\x05\x7d\xac\x80\xeb\x34\xca\xfa\x96\xaa\x04\xec\x33\x8d\xbd\xe3\x0b\xfe\x69\x76\x88\x0b\x3a\xa3\x2c\x28\x9a\x64\xa7\x97\xe2\x0c\xcc\xbb\xe6\x0c\xf2\xa9\xbf\xd3\xee\x6d\x96\xc1\x31\x8c\xf8\x7f\xfe\xfb\x6f\xc7\xb4\xc9\xfb\x21\xc7\x76\x05\x0e\x61\x27\x37\xdb\x96\x70\xe7\x03\x14\x91\x06\x74\x8a\x40\xa7\x95\xa5\xcc\x89\xe2\x57\x7b\xb9\x69\xa2\xe1\x6a\x6e\xdc\xbd\x65\x72\x07\xdf\x88\xc8\x57\x9c\x0d\x73\xc5\xbc\xed\x5a\x52\x09\xd9\x01\x7a\x24\xe6\x24\x7b\x03\x41\x58\x90\x7c\xc9\x4c\x33\x62\xc5\x27\xd2\xe0\xa1\x18\x78\x57\xf3\x43\xb1\x3a\x2d\x17\x66\x15\xef\x2f\x22\x25\x0a\x77\x78\x10\x8d\xec\xca\x67\x98\x28\x52\xdd\x7e\xe5\xa6\xad\x70\xee\x00\xe7\x70\x97\xa8\xcd\x39\x96\x2f\x17\x9a\x53\x5b\xf6\xc9\x58\xd3\x43\xe1\x61\x5d\x90\x8e\x19\xa4\xc9\xf6\xd4\x1b\x92\x4b\x22\x0c\xa7\xf3\x70\x58\x96\x12\x42\x14\x47\x88\xd1\x5c\xe3\x6b\x24\x29\xa6\x1b\xe5\x13\xe8\xf7\xeb\x31\x26\x4b\xce\x06\x3c\x23\x62\x1c\xe7\x6a\xe9\x58\xac\x8a\xf1\xd7\x1f\x9d\xe7\x6a\xb1\xbe\x7d\x99\xe0\xe5\xb2\x37\xab\x70\x3d\xf5\xfb\x0d\xcd\xae\x97\x98\x83\x10\x9f\xb2\xd4\xdc\x80\x9a\x49\x2a\xa8\x99\x36\xe6\xb4\x64\x22\x81\x1b\x98\x29\x80\xd4\x2b\x34\x29\x7b\x45\x1b\x6c\x6f\x18\x39\x9a\xe4\x85\x49\xc9\x57\x4b\x88\x8f\x47\xec\xa3\x29\x37\x02\x5a\x9e\x19\x40\x04\x09\x3f\xe4\x73\xc6\x25\x29\x65\xa0\xd5\x54\x40\xb0\x19\xa4\x76\x18\xf5\xc2\x7a\xf1\xd1\xee\xb2\xfa\x9b\xe3\x9f\x2e\x6f\xf8\xf2\x94\xeb\x29\x70\x27\x71\x30\xa2\x19\xd5\xb4\x33\xae\x3d\x69\x2f\x57\x85\xb7\x88\xe9\x02\x1c\x2c\x95\x2c\x7a\xc8\x4f\xaf\x42\x10\x09\x79\x22\x60\x4e\x87\x31\x86\x75\x2e\xca\x76\xbd\x06\x76\xb2\xee\x00\x15\xe9\x9f\xc0\x16\x50\x5c\x1d\x41\x39\x49\xae\x8e\x16\xcb\xe9\x3f\x3b\x67\xaa\xd5\x05\xa6\x6c\x20\x9e\xf7\xc3\x7a\x1f\x0b\xa2\x10\xf9\xac\x88\xad\x08\x7a\xef\x72\x09\x97\xd3\x0f\x50\x7d\x3a\x54\xb3\xec\xf8\xe2\xb5\x99\xfb\x2e\x83\xdc\x25\x4b\xc6\xee\xca\xb7\xc9\x83\x73\xcc\x62\x9b\x11\x6b\x95\x0c\x2d\x6c\xc1\xec\x8c\xd1\xcd\xe7\x0a\xd8\xbc\xce\x00\x28\xdd\xb4\x69\x10\xdd\xe1\x22\x73\x0a\xa3\x56\x59\x20\xbc\x82\x0b\x2d\xb9\xe7\x4c\xd1\x44\x13\x87\x1d\x83\x44\x53\x88\x8c\xb3\x40\x85\x10\xd9\xde\x84\x85\x47\xa5\xa4\x6c\x36\xb6\x2b\xe9\x92\x3b\xdb\x5d\x0c\x65\x9a\xba\x34\x4d\x99\x1f\xbf\x73\x0d\xad\x36\xaa\x1b\xb2\x06\x9c\x32\x97\x56\x0a\x1a\x07\xe3\x6e\x32\x16\x60\xce\x65\xa3\x8e\x69\x6c\x96\x82\x9a\xc2\xd3\x30\xd1\x4d\xec\xee\x20\xd3\x2d\xe3\x38\x14\x57\x88\xb4\xa9\xa2\x26\x01\x0c\x22\xf5\x55\x43\x2e\xac\x6c\xcc\x81\x1d\x32\x2f\xa2\xd9\xb2\x57\x3e\xd3\xbf\x92\x4e\x8b\x5d\x77\x36\x1d\x01\x27\xc9\x04\x47\x8f\x5e\x0b\xf3\xb6\x08\x2e\x5c\xd9\x00\x2d\x57\x42\x5d\x34\x43\x5c\x7a\xa0\x11\x48\x37\xa1\xb7\xd0\x20\xf9\xd8\x61\x17\x9d\x9b\x55\xb3\x10\x69\x06\xba\xc9\x8c\xde\xe4\x36\xc4\x24\x4b\xf8\x22\x6d\xb8\xcf\xaa\x29\x84\xbb\x44\xea\x34\x65\x30\xee\xf5\x2a\xab\x30\xbd\x8d\x2f\xb3\xa5\x7c\xa4\x3d\xe0\x4a\x6d\xc0\x25\x3f\x25\x7c\x02\x26\x55\x6b\x7e\x70\x39\x36\x41\xaa\x47\xf5\x3c\x6f\x9a\xf9\x53\x3d\x91\x54\x66\x89\x56\x66\x9a\x7b\x30\x39\x27\x2f\xbb\x6f\x06\xa3\x60\xbd\x75\xb0\x7d\xb4\x76\xed\xe7\x2f\x81\x60\x7c\xe1\x24\x01\xf3\xae\xe1\x5f\x15\x2b\x9b\x49\xf6\x3b\x36\x4e\x6a\xc5\x47\x4c\xe1\x99\xdb\x5c\x2b\x5c\xf2\x67\x46\x84\x9c\xd3\xac\x54\x2f\x71\xe7\xf0\x70\x4b\xd1\xf6\x3f\x26\x18\x7a\x03\xde\xc9\xb3\x23\x83\x50\xa2\xe9\x43\x66\x38\x2a\xac\xa2\x51\x82\xa5\xa4\xd3\x45\x00\x2c\xe2\x23\x6d\x21\x7d\xab\x6c\x46\x08\x4a\x94\xd5\x31\x1a\x33\xbe\xfd\x64\xd6\xef\x9e\x55\xf8\x50\x3e\x7e\x34\x0e\x11\xdc\xf4\x7d\xb2\x8c\x22\xe3\x6e\x6a\x8b\x26\xd3\x88\x44\x6b\x20\x04\xb6\xcb\x84\x5f\x09\xfe\xd3\x6c\x46\x28\x84\x49\x3b\x6c\xab\xc8\x78\xc0\x8f\x10\x0c\x47\x95\x52\x29\x61\xf3\xb5\xe2\xe4\xec\xc2\x9a\x38\x3d\x58\x08\x60\x2a\x14\x1f\xf7\x90\xdc\x09\x64\xab\x0d\x5d\x9c\x93\x84\xec\x25\xe2\x7a\x0b\x22\xa9\x86\x33\x04\xe4\xb1\x92\x34\x8a\x32\x03\xeb\x8d\x0b\x5b\x04\x82\x37\x40\xf5\xd4\x0f\xfd\x27\x33\x50\x1b\x0b\x5e\xb7\x8b\x60\x18\x84\x55\x6e\xab\xbb\xd4\x61\xfe\x99\x16\x2c\xc9\x15\xdd\x94\xe8\xaa\xe8\xd4\xcb\x2b\x87\x48\x6a\x6f\x1c\x32\xbd\x34\xae\x4f\xa4\x4d\x78\xc7\xda\x03\xb0\x15\x07\x5a\xe6\xd3\xed\xe8\xc2\x3a\x50\x15\x47\x33\xa2\x4c\xf5\xff\x98\x3e\xd1\x38\xc7\xc9\xbb\xa2\x89\xbd\x25\x7c\xec\x69\xf5\xeb\x0f\xf9\x66\xa7\xf6\x8e\x28\xe9\xae\x84\x25\x18\x45\xbb\x39\x07\xb8\x05\x87\x71\x2c\x8d\xe0\xfa\xc5\xcb\x2d\x3b\x83\x24\xd8\x91\x59\xa8\x80\xdf\x90\x40\x65\xe6\x18\xdb\x2f\x3c\x2c\x40\x09\x10\x0b\x97\x40\x04\xcd\x1a\xbd\x3d\xd7\xab\x92\xf6\x97\x2e\x7a\x6d\x4e\xe3\xd5\x51\x15\xd4\xdd\xc9\x83\x9b\xc9\x83\x0e\x64\xf3\x00\x2f\xff\xa6\x63\x70\x98\xf7\xcf\x01\x08\x87\x4b\x57\xe2\xfe\x44\xc4\x77\x44\x26\x07\x21\x29\x2e\x6d\xc5\x6b\xc9\x8b\x47\x0e\xe5\xa8\xc0\x0c\x3a\xdc\x2d\x3a\x8c\x93\x7c\x6b\xdd\x40\x2f\x77\xc1\xae\xa7\x97\xbd\xd0\x07\x00\x7e\x62\xc8\x8b\xce\x6d\x05\x11\x38\xbc\x41\x2c\xdd\x92\xef\x61\x4d\x94\xa2\x1d\x5e\xab\xf8\xc4\xa5\xe5\x7c\x89\xed\xb5\x49\x70\xad\x37\xf7\x25\x49\x6d\xd3\xb1\xec\x43\x47\x79\x61\x2f\x8e\xa5\xc6\xe0\x83\x2e\x58\xb8\xdd\x2d\x5a\x03\xad\xe3\xb6\x6c\x9f\x87\xac\xae\xec\xdb\xee\x69\xfc\x2e\xc7\x6f\x9c\x09\x32\xa5\x9f\xb7\x12\xc5\x6f\xe0\x53\xab\x5e\xea\x65\xae\x14\x92\x03\xf7\x0c\x14\x9e\x0b\x02\x1a\xed\x4a\xdb\x62\x53\x23\x56\x64\x46\xda\xb4\xc8\x47\xb2\x40\x5c\x94\x7e\xda\x16\x04\x72\xff\x45\xef\xcc\xbe\xce\x95\xca\xe4\xe9\xc9\xc9\x8c\xaa\x79\x3e\x39\x8e\x78\x6a\xe2\xf0\xb9\x98\x99\x3f\x4e\xa8\x94\x39\x91\x27\x7f\xfc\xc3\x1f\x8a\x2d\x9e\xe0\xe8\x71\x66\x60\x75\x96\xfd\x4e\xe5\x2d\x27\x58\xee\x16\xd9\xe3\x52\xd8\x5e\x38\x95\x39\xe8\xc6\x25\x8f\xea\x6f\xa4\xc2\x69\x16\x86\x82\x9a\xb2\x71\x52\xe1\xa2\x58\x05\xe4\x25\xea\x69\xa2\x39\xce\x32\xc2\x9a\xcd\x0e\x26\xd1\x74\x07\xd6\xe3\x52\x55\xed\x08\xc9\xe7\x2c\xc1\xac\x0c\xbf\x00\x95\x97\x04\x89\x08\x53\x16\x1a\xa0\x28\x25\x0d\xd4\x68\x20\x80\x0c\xff\xdf\x2c\x15\x11\xe6\x48\x65\x51\x52\xcd\x0d\xc7\x96\x37\x75\x45\x2f\x71\xb0\x74\xd5\x92\xb2\xc5\xda\x11\xb7\x6a\xab\x92\x14\xef\x96\x4b\x8b\x6f\x5e\xd2\x46\x70\x36\x26\x9f\x35\x93\x93\xdb\x02\x76\x3d\x48\x22\x51\xff\xa7\x3b\x24\x17\x4c\xe1\xcf\xa7\xe8\x92\x32\x10\x60\xbf\xe7\xb9\x90\xe8\x1c\x2f\x8e\xf8\xf4\x28\xe5\x4c\xcd\xd1\x25\xfc\xaf\xfd\xe9\x99\x90\x47\xf4\x33\xc1\xc2\xf2\x07\x5b\x92\xce\xd7\x37\xd7\x24\x24\x72\x26\x11\x79\xd2\x27\xf4\x0f\xff\x89\x52\xd3\xf2\x29\xfa\xf6\xe4\x0f\xff\x89\x7e\x07\xff\xff\xff\x47\xbf\x6b\xd0\xf4\x37\x83\xfc\x82\xca\xc5\xb7\x65\x77\x6e\xaf\xb2\x52\x5b\x54\x73\x3f\x13\xbc\xd8\xa9\xda\x96\x1f\x69\xf4\xc8\xa7\xd3\xb1\x26\x0c\x93\xc8\x37\xc6\x62\x09\x2e\x7a\x4b\xfc\x54\x6a\x6b\x4f\x9b\x4a\x76\x45\x0d\x19\xdb\xa9\x41\x7c\x70\xec\x5a\xe6\x45\xe5\x5d\x08\x22\x2a\x55\x33\xa6\x12\xbe\x22\xb1\xe6\xaa\x9b\x9c\x0e\x67\xdd\x73\xc9\xdf\xce\x82\x13\x22\xa4\x84\xf5\xd4\x7d\xe0\x5f\x18\xc5\x6a\x02\x7d\xec\x42\xd6\x1e\x87\xa5\xf0\xda\x2f\x26\x66\x12\xa6\xf6\x56\xf1\x92\x72\xa9\xf3\xf5\xa1\x92\x77\x5c\xec\xa4\x6f\x3d\x92\xc6\x54\x86\x35\xf5\x92\x5c\x0d\xdf\xb0\xb2\x3f\x64\x88\x73\xe1\x71\x8c\x8d\x5d\xc4\x56\x55\x5c\x6f\xc5\xa4\xc2\x04\x97\xb5\x3b\xf4\x7a\xea\xe7\xfe\x93\x75\xc3\x84\x48\x33\xf7\x76\x51\x2f\x0e\x46\xab\x45\x24\xcd\x12\x6b\x46\x5c\x03\x76\xb8\x6e\x43\xef\x3c\xbe\x05\x34\x0e\x61\x8f\x90\xbf\xc1\x9c\x64\x6b\x01\x04\xea\xf7\x33\x17\x11\x39\xe3\xbb\x85\xbd\x26\x94\x2d\xc5\xcb\xb7\x2f\x45\xe4\x57\xef\xc2\x16\x9d\x72\x78\xc0\x3c\x2e\x94\x05\xe3\x16\xb0\x55\x28\x02\x20\xd2\xf2\x6c\x00\xd0\x6e\x1f\x58\x97\x4b\xb5\x11\x76\xe0\xda\xc6\x70\x5c\x30\x3c\x57\x5a\xa3\x52\x51\x43\x60\xcd\x0b\x57\xc4\xae\x41\x50\xd1\xce\xe3\x08\xaa\xc4\x14\x91\x4a\x95\x6a\xec\xd8\x94\x4a\x11\x5b\x62\x95\x9a\x82\x4d\x3d\x24\x30\x04\x65\xaa\xb9\x6e\x4f\x12\x71\x34\xc5\x11\x65\xb3\x5e\x00\x53\x09\x90\x11\xe1\x75\x50\x47\xa4\xf7\x58\x3e\xee\x37\xd0\x70\xe7\x02\x96\x34\x2e\x8a\xa8\x59\x60\x19\xe3\xd8\xa0\x4b\x18\x7d\x0a\xcb\xc7\x26\x64\xa5\x25\x58\xb7\x15\xa3\xf3\x4b\xe1\xc0\xe0\x56\x8d\xcf\xa5\xa0\x93\x50\x9f\x82\x9a\x0d\xae\xa4\xb2\x05\x79\x74\x19\x7f\xd8\xa3\xb0\x54\xd1\x4d\x57\x8c\x5f\xce\xb9\x50\xe3\x2d\x71\x61\xab\x69\xf4\x8c\x1c\x25\x00\xe8\xc2\x9f\x88\x78\xa2\xe4\xb9\x0c\xaf\xba\x09\x2d\x1a\xa3\x59\x10\x55\x07\xf8\x9b\x69\xc6\x21\x85\x66\x8a\x52\xcc\x16\x86\x51\x6a\xe6\x82\xe5\xa3\xf4\x85\x5c\x91\x4c\x71\x92\xf4\x90\x20\xb9\x34\x05\x8e\x25\x49\xa6\x47\xae\x14\x46\x8c\x12\x3e\xa3\x11\x4e\xd0\x24\xe1\xd1\xa3\x34\x19\x6e\x6c\x66\x98\x54\x26\x78\x44\xa4\x0c\x24\xab\x22\x9b\xdd\xe6\x18\x42\x15\x57\x45\x44\x4a\x19\x95\x8a\x46\x4e\x64\x2a\x40\x29\x4c\x2d\xf1\x08\x83\x49\x18\x32\x36\x61\xb8\x5a\xd2\x23\x06\x9c\x33\x67\xb6\x68\x12\x5c\xd7\x16\x73\xcf\x05\x89\x37\x1d\xa0\x3d\x40\x08\x3a\x0a\x19\xab\xf2\x81\x5c\x73\xa4\xce\xec\x67\x70\x8c\x57\x91\xc0\x6d\xf9\x44\x79\x82\xf4\x27\xad\x04\x6b\x04\x31\xe5\x3e\x04\xbe\x24\xb9\xf8\xc8\xf0\x03\x43\x34\x83\x21\x37\xe0\x98\xad\xa3\x69\xbd\x8a\x20\xf2\x40\x9d\xae\xaa\xd7\x9c\xb2\x28\xc9\x63\x5f\xa9\x51\x8b\x00\x4f\x9a\x48\xdc\xf2\xe8\xb5\xd7\x82\x42\x0f\x61\x89\x9e\x49\x92\xe8\xff\x9a\x08\xf8\x23\x5f\x38\x41\xb3\x64\x53\xdc\x02\x3a\x71\x5c\xba\x89\xa2\x0e\x0e\x9d\xf2\x06\xab\xb9\xc9\xf9\x4f\xb9\x32\x45\x32\x0d\x3a\xa5\xb3\x6f\x19\x38\xc3\x49\xc2\x27\x70\xd2\x01\xb8\xd2\xe5\xb9\x06\x69\x75\x79\x14\x11\x12\x93\x18\x7d\x1d\x1c\x5c\x8f\x47\xf1\x4d\x3d\x8c\x62\x69\x45\x0e\x00\xb4\xb2\x6a\x58\x6b\x84\xae\x2c\xd7\x79\x3b\x46\x37\x15\x60\x96\xb0\x7e\x3b\xae\xc2\x74\xf5\x96\xb6\xf0\x6d\x80\x2e\x2b\x93\x78\xb9\x1d\xda\x10\xe8\xb2\xd4\xe7\x1e\x80\x2e\x2b\xf3\x6c\x88\xdd\xe7\xb3\x17\xcd\x39\xd6\x93\xba\xe0\xed\x13\xc1\x0c\x40\x98\xb9\x3b\x4b\x24\xe8\x0e\xe4\xa2\x8e\x10\x0f\x0b\xc4\xb3\x52\x0d\xf1\x6d\x41\x3c\x2b\x83\x39\x64\x10\xcf\xca\x50\x0f\x17\xc4\xb3\x66\xa0\x2d\x40\x3c\x8d\x73\x7f\xac\x89\xba\x1d\x53\x80\xc4\x96\x49\x3e\xbd\x83\x54\xef\x95\x63\x3c\x33\x81\x03\xe6\x1a\x73\x77\xb4\xc5\xb4\x86\xd1\xda\x1c\xc8\xa6\x70\xa8\x8a\x13\x62\x53\xda\xf3\xde\x37\x03\xfe\xb0\xa9\xd9\xbd\x17\x5a\xbb\xc1\x0e\x19\xe1\xcc\xe6\x94\x37\x95\x9a\x39\x9c\xec\xd9\xed\xf0\x51\x01\x83\xb0\xc4\xf2\x5b\x21\x88\x5d\x56\xaa\x36\xcc\xf9\xb3\xad\x9c\x04\x64\x68\x88\xb2\x91\x04\xa1\xd3\xb1\x55\xda\x9a\x56\x8e\x32\x45\x66\x55\x9d\xb6\x38\x34\x94\xa9\x3f\xfd\x71\x2d\x27\x32\x10\x8b\x4e\x3d\x0c\x6a\x27\x78\x67\x87\x7d\x46\x62\x14\xcd\xb5\x56\x24\xb5\xfa\xa2\xa7\x63\x6e\x56\x89\x52\x4c\x9d\x22\x95\x4b\xe3\x5a\xa2\x72\xc4\x4a\x98\xa4\xc7\xe8\x23\x14\x84\xc5\x69\xa6\xf5\x2f\x3f\x3f\xaa\x29\x69\x94\x7f\xfb\xed\x9f\x08\xfa\x16\xa5\x04\xb3\x92\x0e\x0b\x6a\x93\xbe\xfa\x00\xc3\x4f\xcd\xc9\x88\xd5\x6e\x05\x1a\x7c\x36\x35\xa6\x5c\xbc\xdf\x90\x4d\xb9\xd3\x89\xa1\xd0\x21\x8e\xe6\x48\xe6\x13\x53\xa9\x37\xb0\x61\x38\x41\xfa\x82\xcf\xc0\x51\x0d\x37\xb2\x1b\xf4\xaa\x53\xf8\xb2\x31\x00\xd6\xdd\xd8\xf6\x36\xee\xc3\x3d\x72\x24\x49\x09\xdb\xa9\xc6\x69\x66\x38\x5f\x78\xf0\xa5\xc1\x7d\xe9\x19\x1f\x82\xd6\xcf\xb0\xb5\xec\x6b\x59\x1a\xc2\x79\xc1\x4b\x96\x27\x58\xd8\xa3\x3f\x62\x5a\xd1\x10\xe4\x89\xf2\x5c\x26\x0b\x14\x73\x46\x7a\x40\x09\x79\x34\x37\x8e\x55\xad\xb3\x60\x5b\xb0\xe2\x89\xca\x5c\x2b\xb4\xd0\x96\xab\x8f\x21\x15\x36\x98\x54\x73\x0a\xfd\x68\xf5\x9b\xc0\x57\x61\x96\x1c\x6a\xa7\x45\x85\xb0\xb1\x15\x9e\xdf\x12\x36\xb6\x44\x55\x1d\x6c\xac\x87\x8d\x5d\x5e\x97\x43\x84\x8d\xad\xec\x79\x3b\xd8\xd8\xba\x2d\xdf\x02\x36\xb6\xd4\xcc\x17\x03\x1b\x5b\x59\xd1\x2f\x06\x36\xb6\x32\xaf\x0e\x36\xf6\xcb\x83\x8d\xdd\x11\x18\xb5\x9e\x17\x1b\x7c\x25\x45\xd9\x62\x63\x22\xfb\x4a\xa2\xe1\xb5\x26\xb0\xe8\xb1\x1c\xd4\xe6\xaf\xab\xdd\xc1\x58\xeb\x99\xd0\x66\x60\xac\xb5\xaa\x7a\x33\xab\xdb\x15\xe0\x09\x14\x83\x57\x06\x63\x2d\x4d\xa0\x8b\xaf\xdc\x3c\xbe\xb2\x96\xf8\x6c\xdf\x7a\x78\x2e\xe8\xb2\x7a\x21\xb7\x84\x63\x2d\xed\x4f\xab\x48\x4c\x10\xdd\xf7\x40\x89\x2f\x2b\xcd\xdf\x97\x0e\xf9\x5a\x59\x3e\x5c\x45\x69\x81\xa1\xb5\x84\xe7\xd0\xe2\x8c\x12\x1e\xfa\xff\x3b\xca\xdd\x22\x32\xb8\xb2\xbc\xde\xaf\x62\x68\xb1\x05\xa9\xb6\xa6\x50\xa7\x95\xee\x27\x51\xd6\x25\x4f\x6e\xe8\x62\x76\x83\xb8\xcb\x48\xd4\x60\x63\xa6\x29\xdd\x57\xb3\xeb\x2e\x32\x8f\x85\x05\x0a\xf9\x52\x5e\xa8\xbe\x9e\xcc\x70\x8c\x8c\x5f\x49\x87\x05\xa0\x0e\xf3\xe5\x8c\x4a\x25\x1a\x63\x9b\x96\x46\xb8\x8b\xab\x34\xcb\x5b\x07\xc4\x04\xab\x3a\xdb\xee\xb3\x94\xa4\x5c\xac\x0b\xac\xaa\xfd\xd2\x96\xba\xd9\xe6\x53\x92\xcd\x49\xaa\x25\x99\xf1\xa6\x8d\xb4\xdd\x6f\x9f\x34\x6c\x73\xd7\x4c\xa0\x63\x89\x08\x02\x47\xa8\x7e\x37\x36\x88\x94\xad\xb7\x7b\xd7\x6d\xb6\x98\x99\x1b\x3a\x84\x1c\x98\xf2\x6a\x83\x9b\x7d\xa9\xe4\xee\x06\xfa\xae\x8d\xe9\xf0\x21\x35\xeb\xa3\x36\x56\xc4\x6b\xac\xc2\x9d\x2a\xbe\xb2\x85\xa0\x37\x70\xe5\x97\xbd\xf3\x9a\x13\x86\x55\x80\x37\x0f\xf0\x68\x40\x4d\x5d\x5e\x1e\x88\xcc\x91\x44\x1c\x85\x9a\x41\x69\x30\xcb\xeb\x55\xa2\x12\xa7\x51\xee\x40\x24\xb9\x68\x8c\x32\x6d\x63\xd0\x8e\x54\x8e\x13\xd0\x24\xc2\xea\x95\xd5\x4d\x9d\x2c\x6a\xd2\x1e\xdb\x79\x4c\x28\x53\x7f\xfe\x8f\x8d\x76\x53\xab\x56\x76\xdd\xa0\xe2\x16\x8e\x22\x22\x8d\x8d\xdd\x46\x21\xe3\x09\x7f\x82\x62\x5b\xbb\xec\xaa\x3e\xca\x7a\xde\x9a\xc1\x7b\x28\xe2\xb8\x20\x75\x23\x2e\xcc\x05\xcf\x67\x73\x67\x43\xd2\x67\x46\x4f\xad\x6e\x2f\x7f\x5c\xb2\x91\x6f\xbc\x97\xdf\xe5\x34\xd9\xce\x42\x77\x57\x2a\x43\xf6\x69\x78\x8f\xe4\xdc\x9f\xd6\x09\x34\x5b\xbb\xb1\xcb\x83\x6e\xdf\xa7\xfd\xd6\xfb\x6b\xa0\x9b\x9e\x83\xdf\x9c\xf2\x24\x01\x4f\x83\x24\xe9\x13\x11\xf5\xdd\xc3\x84\xef\xe9\x66\xc8\x79\x7e\x00\xf0\x75\x91\x18\xd1\x4a\xfe\xba\x31\xa2\xa1\x44\x6e\xf4\xd5\xa0\x05\x13\xaa\xc6\x19\x61\x75\x36\xb6\x9f\x96\x2b\xc0\xbc\xb3\x80\x41\x17\x3d\xb6\xb7\xa0\x41\xb7\x24\xaf\x1c\x38\xb8\x66\x1e\x87\x1a\x3c\x58\x61\x76\x3e\x96\xaf\xb8\x66\x5c\xe0\x90\x51\x7c\xfa\x7a\x89\x47\xac\x5f\xca\xa7\x70\x95\xb2\x27\x8b\x22\x20\xdb\xe8\x10\x21\x33\x83\x3a\x1b\xd6\xb0\x02\x6e\x34\xfd\x17\x68\x3a\x06\xbc\xd6\x84\x14\xba\xb0\x41\x88\x26\x27\xf1\x11\x8e\x16\x51\x42\xa3\x40\x67\x9e\x09\x9c\xcd\xeb\x38\x9e\xdb\xf9\x0e\x75\xe7\xad\x50\x77\x9a\x0a\x52\x6d\x12\xb7\xed\xe8\x8a\xe1\x94\x74\x68\x40\x75\x68\x40\x3d\x8f\x77\xc1\x8a\xd2\x5a\x6f\x08\xa3\xb0\x7c\xee\x3a\x48\xa0\x37\x80\x04\xda\xe6\xf0\x15\x78\x3f\xa5\x63\xd7\xc1\x14\x7d\x68\x05\x53\xe4\x2f\xc1\x83\x42\x9e\x69\x3e\x8f\x6f\x8c\x68\xb2\x3c\xb0\xb7\x84\x25\xaa\x11\x17\x36\x91\x9b\x56\xe1\x12\xad\xa2\x8b\x56\xeb\xf2\xb6\x28\x41\x9b\xad\xcc\x46\x00\x40\xb5\x77\xd7\x81\xc0\x01\x35\x6f\xc3\x81\x9c\x9b\x7d\x66\xb5\x6c\x56\x3b\x34\xcc\x6c\xd9\x44\xc1\xda\x2c\xc9\xc5\xd3\xc3\xfb\x4a\x74\x29\x8a\xac\x6d\x97\xec\xd2\x77\x3e\x68\x22\xd0\x9c\x27\xb1\x03\xa1\xf0\xab\xe5\x3b\xf0\x99\x00\x7e\x81\xdc\x66\x40\xb1\x73\xd0\xb6\x8a\x82\x60\xab\x52\x5a\xfc\x26\xc2\x70\xf7\xc0\x68\xf6\x61\x45\xf0\x9c\x64\x1b\xfb\xc1\x5a\x59\x44\x96\xcd\xdf\x2b\xc6\x58\x5a\x21\xb0\x9a\xd7\x0f\x73\xad\xdd\x77\xcd\xe0\x56\x89\x1e\x81\x71\x50\xd4\x95\xfa\x34\x74\x06\x4f\x9f\xa8\x33\x44\xe0\xb0\xc7\x95\x5e\x3a\x37\xbb\x56\x9e\xba\x2a\xb1\x6c\x11\x0c\xb6\x54\xb9\x6d\x77\x70\xa0\x14\x7f\x1e\x67\x58\xe0\x24\x21\x09\x95\xe9\x8b\x05\x03\x9f\x95\xdd\xb5\xfa\xac\x0a\x6e\x4c\x44\x2c\x4f\x27\x86\x14\xdd\x40\x6c\xb1\x3f\xc5\x91\xc8\x59\x08\x6d\xe6\x37\xc6\x17\x13\xcc\xe1\x5e\x00\xab\x52\x34\x87\xaa\xad\x53\x4c\x05\x23\xb2\xb1\x46\x26\x89\x72\x41\xd5\x62\x6c\x4b\x8e\xb6\x3f\x70\x77\xf6\xcb\x33\xfb\xe1\x6a\x0f\xb7\xcb\xea\x77\xfd\xf9\x12\xa7\x19\x11\x50\x26\xc8\x15\xbc\x09\xca\xaa\x5a\xd4\x06\xe2\x6b\x0d\x41\xf8\xf3\xd2\xb5\xdd\x14\x38\x8c\x9f\xc7\x41\x46\xd5\x38\xaa\x12\xc7\xba\xc3\x5a\x87\x3b\xb5\x6a\x92\x2f\x8c\xbc\xd4\xe0\x45\x7e\x81\x2a\x23\x36\x6d\xc2\x34\xad\x07\x1c\xb8\x82\xc1\x5e\x59\x6c\x4c\x90\xf2\x6e\x95\xaa\x86\x71\x5a\xac\x9f\xba\xe0\xa3\x15\x83\xed\x07\x5f\xb5\x18\x71\xd0\xc9\x9e\x86\xad\x0f\xba\x10\x79\xa6\xe8\x64\x19\xda\x46\xed\xaf\x84\x68\x3f\x81\x34\x6b\xe7\x66\x28\x75\x6b\xea\x8a\x96\x38\xb1\x9d\x9d\x96\xff\x2d\x8e\x98\x43\x08\x32\x08\x4b\x61\x1e\xdf\x75\x4a\x95\x72\x89\x02\xc6\x00\xad\xa9\xb3\x6c\x9b\xfd\xca\x85\x7b\x60\xa8\xf4\x6a\x4c\x44\xc7\x23\xd6\x97\xe8\x99\x20\x46\x2c\x84\x44\x4d\x0d\x57\x6f\xd5\x86\xda\x4f\x13\xa2\x7b\xf2\xb1\x29\x5a\x78\xa0\x4a\xfa\xf2\x63\xa6\x8f\x29\x4e\x24\xe9\xe9\x86\xa1\x6a\xa9\xe2\x10\xfc\x89\xd1\xb3\xc0\x59\x46\xc4\x88\xd9\x2c\x0e\x70\xb8\x70\x9e\x98\xf6\x9b\x42\x5c\xed\x1a\x90\x71\x84\xa3\xf9\x2b\xed\x11\x86\x64\x9c\x68\x4e\x62\x97\x2f\x5c\xde\x1e\x37\x6f\x63\xb0\xde\x60\xb3\x86\x53\x57\x3e\xab\x67\x3b\x49\x22\xcd\x51\x7c\x99\xe9\x8c\x08\x3d\x6a\x4d\xc3\x4f\x84\x21\x3a\x75\xe3\xb0\xb1\x3b\xe8\x19\x3c\x53\x9a\xf4\x9f\x30\x4d\x4c\x02\xbe\xeb\xda\x09\x81\xc6\xfc\x3e\x62\xc6\xdd\xcd\xa2\x52\x86\x2a\x65\x54\xce\x35\xa7\xce\xc1\x27\x09\x6a\x46\x53\xe2\x0c\x7b\xda\xe4\x34\x0f\xf4\xeb\xab\x39\xe8\x13\x15\x9c\xa5\x90\x24\x63\x71\x99\xdc\xf2\x49\xa2\xfc\xf1\xa8\x4d\x71\x5c\x2b\x11\xc7\xb1\x2c\x1b\x3f\x8d\x5a\x49\x7f\x2d\x99\x5d\x8e\x4a\x59\x81\x51\x00\x2b\x04\x41\x9c\xae\xb2\xd8\x2a\xf9\xb7\x4b\x6d\x58\x4e\x6d\xa8\x5f\x9b\x43\x4c\x6f\xf0\x87\x78\xd3\x14\x87\xa6\xed\xdf\x87\x64\xbb\xc7\x54\x87\x37\xce\x09\x78\x99\x74\x80\xb7\xcd\xdf\x78\x89\xd4\x8d\x2e\xc1\xe1\x0d\x13\x1c\x5a\x5b\x6a\xcb\xb1\xd9\xcd\xc7\x76\xa3\xe4\x80\x35\x60\x4e\x75\xbd\x5c\x12\x25\x68\x24\xf7\xc1\x1f\x64\x86\x5b\x46\xb5\x81\x16\x98\xad\x91\x9a\xf4\x0b\xde\x09\x09\x71\x62\xbe\xce\xdf\x44\x10\xfc\x18\xf3\xe7\x25\x5b\x9d\x0c\xd1\x34\x2e\xb9\x16\x7b\x04\x89\xa8\x24\xa5\x48\x16\x2a\x11\x23\xd2\x1a\x3b\xf1\x88\xcd\x29\x11\x58\x44\x73\xc8\x6e\x2c\x36\xc6\x64\xc9\x1a\x40\x23\x13\xcb\x10\x7a\x9b\x36\xd8\xf4\x16\xeb\x5e\xb5\x30\x79\x7c\x3a\xbb\xe7\x7a\x24\xa9\xf9\xc4\x0b\x33\x56\xca\x08\x4d\x72\xad\xb6\x7f\xd7\x40\x7c\xbf\xd8\x2f\x1a\x8c\xef\x83\x89\x82\x2f\x5a\x06\xe4\x17\xd4\xd0\x05\xe5\xbf\x50\x50\x7e\xcd\x12\x6f\x16\x98\xbf\x95\xc9\xef\xf5\x63\x86\x5d\xcf\xaf\x11\x37\xbc\x2e\x68\x2b\x9f\x8c\x5f\xfc\xe8\xd5\xce\xb9\xed\x09\xfc\xc9\x13\x85\x91\x88\x85\xa6\xb3\x09\x89\x63\xe0\xb4\x8a\xdb\x4a\xd1\x05\xed\x38\xf3\x80\xbe\x7b\xb1\xd4\xc4\x8e\x13\xce\x66\x92\xc6\x06\x6c\x25\xc3\x50\xb1\x35\x34\x5e\x00\xb8\x00\xec\x6f\x92\x10\xe1\xbc\x12\x02\x7d\x2d\x29\xb3\x68\x8a\xfe\xb7\x98\x13\xc9\xbe\x52\xc6\x58\x80\xd9\x02\x3d\x32\xfe\x9c\x90\x78\x06\x3b\x54\x1d\xcc\x11\xa2\xa4\x87\xa8\xf2\x9f\x09\x40\x23\xe0\xb9\x1a\xe9\xb1\x43\xac\x99\xd1\x00\x88\xfd\x36\xa8\x89\xee\x9b\xf9\xe6\x18\xa1\x21\x43\x53\x1c\xa9\x1e\x92\xf9\xa4\x68\x3f\xe6\xa6\xc8\xb5\xd6\xbe\x83\x89\x17\x8d\x74\x31\xe3\x35\x9d\xd7\x9f\x0d\xc7\x1d\x34\xb9\xf6\x13\x8a\x77\x8a\xad\x7b\xc2\xbb\x40\x8c\x5e\xe6\xd2\x06\x61\x20\xce\xfc\xd1\xb7\xf0\x4a\x1e\x23\x1a\xf0\x3e\x0d\xde\x32\xe3\x71\xa3\xad\xb3\x32\x95\x4d\xc7\x52\x04\x42\x5a\x41\xc9\x3a\xaa\xa0\x5d\xb3\xdc\x5a\x6a\x92\x4a\x10\x9c\x5a\xe7\x80\xbe\x6a\x40\xac\x31\x61\x90\x7a\xf4\x54\x18\x09\x73\x93\x2d\xbe\xa0\xec\x51\xef\x6e\x81\x8a\x0d\xf5\xe5\xa1\xe7\xba\x4d\xcb\xf4\x8d\x47\xce\x38\x33\x0e\xc2\x9d\xe4\x4e\x3a\x63\x38\xd9\xd0\xc6\xb1\xb4\x72\xcb\x3e\x3d\x27\x67\x59\x71\x41\x4b\x11\xc6\xd8\x87\x4c\x8f\x1b\xd9\x90\x2a\xf3\x0d\xe5\x3d\x8c\x62\x92\x11\x16\x13\x16\x2d\x80\x44\x18\x20\xe7\x08\x86\x13\x84\xe1\x3b\x9c\x1c\xa3\x73\x93\x5f\xe3\x25\x3c\x7b\xad\xc3\x85\x9e\x62\x46\xa7\x5a\x4f\x00\x23\xac\x1d\xe5\x88\x99\x61\x3a\x1f\x48\x50\xb4\xdf\xaf\x58\xdd\xce\xe8\x1b\xe4\x6a\x47\x54\x62\x56\xfe\x1e\xad\xbe\x70\xa0\xb7\x55\xbb\xa3\x9b\x73\x35\x08\x64\x3e\x39\x82\x7f\x97\x12\xce\x1c\x50\x4f\x81\x22\x43\x12\x02\xe6\x40\xeb\xf1\x82\x8b\xb1\x09\x58\x6e\x1f\x7e\xbb\x35\x79\x1c\x41\x1f\x25\xa5\x26\xa5\x8c\xa6\x79\x1a\x38\xef\x4c\xc5\x82\xc8\xda\x2f\x4d\x26\x46\xa6\xf5\x80\xc8\x81\x97\x23\x7d\xb9\xb2\x05\x9a\xd1\x27\xc2\x46\x2c\xe3\x94\xa9\x63\x74\xc5\x15\x09\x4a\x44\x18\xe8\x28\x9e\x29\x9a\x1a\xb4\x53\x41\xf4\x39\x30\xa0\xd8\x00\x34\x39\xc7\xaa\x87\xe2\x1c\x8e\x2a\x23\x4a\xb3\x0e\x7d\xe3\x2a\xd8\x19\x88\x8f\x16\x23\x66\x6e\xba\x29\xa6\x49\x2e\x88\x95\x59\xb1\xc9\x8b\x29\x86\x5c\x8c\xcc\x22\xa1\x05\x93\x48\xe9\x6c\xae\xf4\x16\x69\x19\xcf\xfa\x1b\xe7\x9a\x1b\xf1\x11\x9b\x10\x84\x51\xc6\x25\x55\xf4\xc9\xfb\x2f\xe9\x14\x61\x29\xc1\x82\x72\x8c\xce\x4b\xf6\x7f\x2a\x41\xf5\x6e\x8a\xab\xa5\x6c\x6c\x6d\xcf\xcd\xf9\x38\x3b\x6f\x64\xa9\x17\xbb\xca\x78\x22\x79\x92\xab\xd0\x05\x5b\xbf\xb7\x85\x69\xdc\x01\xf7\x83\x81\x98\x4f\x47\xcc\xd1\xb5\x3c\x46\x7d\x89\x24\xd7\xbb\x24\xcd\x56\x46\x82\x2a\x22\xa8\x41\x71\x22\xca\x6c\x82\x3f\xa7\xfe\x0c\xa4\x58\x3c\x6a\x11\x2a\xb4\xc0\x1b\x4c\xd1\x92\xb5\x63\x62\x24\x24\x80\xb5\x0a\xb7\x03\x4c\xff\x88\x71\x76\xc4\xc8\x0c\xaf\xdb\x91\x11\x2b\x6d\x09\xfa\x9a\x4e\x0b\x85\xb4\xc9\xe7\x18\xac\xdd\x18\x22\x9f\x9a\x76\xc9\x74\xdc\xb4\x49\xd3\x84\xe3\x35\x6e\xe3\x69\x71\xe8\xd1\xdf\xf9\xc4\x8c\x51\xeb\xfd\x5c\x81\x14\xa8\xd5\xab\x29\x17\x64\x8e\x59\xdc\x73\x9b\x55\x1e\x1b\xdc\x8c\xd6\xd4\xe6\x94\x31\x90\x04\x1d\x88\x30\x31\x58\x4c\x98\x05\x7b\x61\x15\x37\xbb\x15\xc5\x3e\x6c\x74\x57\xf8\xd6\xa0\xf6\x89\x31\x40\x18\x96\xb7\xc8\xec\x11\x97\x34\xcd\x92\x22\xa7\x29\xb0\x8d\x4e\xb5\x88\xe5\x78\x24\x7f\x02\xd3\x95\xd3\xda\xe0\x56\xb7\x3b\xa7\xe9\xac\x66\xe4\x9e\x91\xc2\xad\xe1\x6c\x5e\xa6\x0c\x66\xc0\xc2\xbe\x96\x44\xff\x53\x91\x42\xed\x33\xc2\xfa\x88\x39\x11\xe4\x1b\xe0\x32\xb6\xd9\xc0\x78\xa6\x45\x68\x03\xf3\x6a\xd7\x0f\x45\xc6\xc9\x5d\x3a\x27\xf6\x30\xb8\x57\x6b\x2e\xaa\xef\x28\xc3\xa5\xcc\xdb\x2d\x04\xbf\x24\xdf\x28\xb9\x2a\x70\xfb\x2d\x9a\x6a\x9a\x28\xbc\xae\xcc\xc8\x06\x94\x60\xf6\x99\x20\xdd\x9d\xa5\x66\x57\xf1\x06\x43\x44\xc0\x9c\x24\x19\x8a\xe9\x14\xcc\x52\x0a\xd8\xb7\x07\x16\x33\x58\xf0\xfa\xb0\xa7\x39\x33\x20\x71\xc6\x23\xf2\x6c\xf1\xb6\xed\xd5\x58\x34\x7e\x3c\x62\x43\xf5\x95\xd4\x22\x3a\x67\x33\x7d\xd1\xc4\x4f\x54\x16\x45\x4e\x22\xce\x64\x9e\x12\x61\xbb\xd0\x37\xb2\xa6\x48\x5b\x20\x00\x3b\x19\x4a\x8f\x4d\xef\xfd\x13\x4e\x68\xec\x0a\xf1\xe8\x1f\xcd\x99\xd3\xa3\x94\xce\xa3\x58\x13\x12\x66\x37\x37\xd6\x6b\xf5\x66\x62\xfd\x8f\xa1\xe4\x8e\xd2\x42\xc8\xc7\xd6\x56\x7f\x52\x15\xf1\xed\xaa\xaf\x10\xef\x27\x4b\x93\x42\xab\x05\x23\xbb\x0a\xe7\xeb\x50\x0c\x1d\xa2\x6e\x6e\x42\x80\x75\x3f\xce\xe8\x63\x06\xb7\x11\xfb\xa9\x4c\xd0\x8e\xda\x70\x96\x50\xbc\x27\x14\x64\x03\xa9\xb0\x16\x2f\xcc\x75\xc0\x85\xd5\x70\xec\x9d\xd3\xbc\xb5\xe7\x3b\x96\x89\x90\x11\x4e\x96\x77\x78\x85\xbd\xd9\xbc\xbf\x5a\x09\xb0\xc7\xcd\xb4\xbd\x32\xe9\x37\xe2\x49\xb2\x49\x09\x93\xca\xcc\xcf\x8a\xcf\x57\x8f\xa8\xe8\x47\x6f\x80\xdb\x0b\x38\x35\xe6\xf2\xc6\x89\x35\xa5\x48\x65\x77\x29\x7c\xc9\xa8\x61\x0b\xcb\x5a\x47\x8c\x4f\xa1\xc8\x4d\xd2\x14\xd5\x95\x09\x9e\xd2\x4d\x50\x96\x4d\xa0\xd3\xad\xb3\x8b\xaf\xb1\x32\x38\xeb\x39\x88\xa6\x86\xbc\x6c\x8f\x90\xaf\x87\xad\xb8\xb9\xe2\x0c\xa5\x38\xdb\x6a\xc1\xd7\x79\x85\xfa\x28\x35\x2e\x39\xbb\x7a\x80\xb7\x48\xa0\x5e\x0c\x2c\xf2\x33\x5e\x14\xa9\xd1\x4d\xf8\xb9\x6c\x23\x72\x78\xd0\xaf\x0f\xd9\x94\x6f\x70\x38\x8b\x54\x66\x7b\xfa\xb0\xa3\xd9\xe0\xfc\x79\x2f\x85\xd9\x7d\xb3\xa6\x6d\xce\xe3\x59\x1d\x51\x6f\x7c\x32\xdd\x0a\xbe\xa4\x8d\x32\x64\x22\xa1\x79\x72\x93\xbb\xb5\x7c\xb4\x82\x16\x11\x0c\x67\xf5\x52\x5d\x96\xe8\x70\xef\x6b\x54\x69\x07\x19\x53\xb8\x0b\xa6\xbe\xa9\x6f\xf5\x15\xd6\xcc\x1e\x92\x56\x8b\xb5\x23\x76\xc3\x66\x38\xc0\xae\x47\x8f\xfa\x5b\x7f\x42\xd7\x16\x39\x68\xbf\x18\xc0\xcd\xa4\xb5\x73\x15\x91\x99\x36\x45\x6d\x4a\x13\x2d\x62\x0f\x6b\x0c\x9c\x2e\x41\xcc\x07\x54\x99\x50\x79\x27\x3d\xe5\x82\x06\x85\x41\x9d\x8c\x84\x28\x14\x28\x09\x9d\x3c\x81\x42\x0f\xa6\xc5\x39\x7f\x36\xd1\xe9\x82\x6a\x9e\x65\x84\x55\x05\xe6\x1e\xcd\x0b\xa8\xb5\x96\x18\x63\x93\xff\x80\x9b\x98\x41\x6c\x6b\x1f\x17\xa3\x6a\xd8\xd2\x7d\x94\x78\x6a\x9f\x7f\xe7\x7a\xbd\xd7\x5f\x2c\xef\x4d\xed\x08\xef\xcb\xad\x6f\x3c\x3a\x2f\xe5\x6f\x1e\x30\xf5\x11\x3e\x75\x4a\x0f\x46\x53\x41\xc0\xc1\x9f\x7a\x4c\x0d\x03\xaa\xcb\x39\xdc\x77\x77\xe7\x3f\x9c\x3c\x0c\x11\x51\x11\x4a\xe8\x23\x19\xb1\x48\x3e\xf5\xb4\x78\xfc\x8f\x9c\x28\xfd\x73\x83\x47\x80\xa6\x84\x49\xe0\x04\x54\x2d\x61\x0f\xd5\x2f\xa4\x5b\x18\xfd\xdf\xf3\xf2\xf7\x2b\x48\x7e\x29\x7d\x18\x68\xd7\xd5\xbb\x01\x32\x85\x92\x1e\x66\x69\x65\x0d\xc5\x18\x5b\xe4\xa0\xae\x1a\xe6\x16\xe9\x42\xec\xef\x39\xdb\x50\xe8\x3a\x2b\x3e\x0a\x46\xd1\x20\xd3\xa5\x19\x06\xac\xeb\xcd\xf2\x90\xcc\x37\xb5\xad\xaf\x63\x22\x45\x5a\xb6\xb3\x2d\x17\x85\x43\x91\x12\x84\x00\x0b\xf1\xf4\x64\xef\x7a\x8b\xc4\xe1\x27\x16\x7c\x74\x3c\x62\x97\xce\xe3\x5c\xfc\x2a\x0b\x3d\x3c\x9d\x04\x10\xe0\xe5\x56\xa0\xd9\x98\x4a\xff\x03\x14\x74\x91\x79\xa2\x4c\x45\xbb\x29\x65\x38\xf1\x03\x35\x4f\xea\xb8\x84\xc0\x2c\x9a\xef\x6a\x42\xa6\xd3\x31\x49\x36\x91\x44\x87\xd3\x41\x22\x35\x7d\x47\x8f\x0d\xa7\x73\x9b\x9a\x8d\xc5\x64\x6c\x25\x5a\x53\xf7\x09\x15\x26\x68\x9c\x98\x8a\x72\x04\x81\x8f\xb2\x9a\x3d\x66\x00\x22\xf4\x2e\x5a\x49\xdd\xb8\x28\x4d\xda\x86\x0f\xc9\x86\x5e\x10\x56\x23\x26\x72\x06\xc5\x26\x7c\xc4\x02\x46\x05\x5e\x78\xe4\xfc\x07\xd6\x9b\x33\xd3\x6c\xc2\xc0\x71\x9b\x97\xb5\x7e\xc6\x73\x09\xb6\x9a\x94\x28\x7d\x41\x7d\x0d\x75\x60\x4d\xc8\x50\x0f\x65\x82\xa6\x60\x6e\x95\xdf\xd4\x6c\xdd\x19\x56\x38\xe1\xb3\xbe\x50\x74\x8a\x23\x75\x8f\x77\xd2\xc0\xb1\x6d\x66\xdb\xf0\x53\x37\x0c\x34\x3c\xd7\x8b\x3f\x23\x8c\x08\x98\xa8\xd6\xc9\xeb\x8f\x30\x3c\xd9\x8a\x73\x83\x95\xcd\x1a\x46\xa5\xb7\x58\xe0\x5c\xf1\x54\xeb\xb7\x38\x49\x16\x3d\x63\x91\x25\x68\x8e\xe5\xdc\x6d\xb4\x31\xa6\xb5\xb9\x9b\xec\xe2\x9e\xe1\x68\x4e\xee\xa0\x2a\x72\xdd\xe2\x56\x46\xf9\x81\xb0\x3c\xfd\x70\x8a\xfe\xa7\x98\xe3\x59\xff\xec\xfb\xc1\xf8\x7c\x78\xd7\xff\xee\x62\x70\x1e\xcc\xc7\x3e\xb9\x1c\xde\xdd\x2d\xff\xfa\xfd\xf0\x7e\xf9\xc7\x9b\xeb\x9b\x87\x8b\xfe\x7d\x5d\x2b\x17\xd7\xd7\x3f\x3c\xdc\x8c\x3f\xf6\x87\x17\x0f\xb7\x83\x9a\x4f\x1f\xee\x9b\x1f\xde\xfd\x30\xbc\xb9\x19\x9c\xbb\x55\xf9\x5b\x70\xba\xc0\x7a\x0c\xa9\x17\xf5\xd3\xa8\x1e\xc0\x23\x54\x7e\xf1\x14\x3d\x54\x4b\x1f\xd8\x58\x64\x83\x63\xf1\x8c\xa5\xe6\x61\x10\x0a\x3f\x62\xc8\x7d\xae\x17\xa5\xe9\x53\x13\xae\x13\xcd\x09\x4a\x38\x7f\xcc\x33\xcb\xda\x4c\x7c\x18\xe3\xc6\xf0\x43\x64\xd0\xda\xf7\xc3\xfb\xd3\xe5\x12\x0c\xbe\xb1\x00\x31\xcb\x9d\x01\x18\x17\x76\xec\x14\x6c\x29\x0e\x9a\xbf\xb0\xde\x06\x3d\xf8\x9d\x59\xd5\x8f\x69\x0d\x33\x55\xe9\x26\x8e\x6d\xd1\x5f\x37\xb1\xa0\xe1\xf2\xbe\xae\x5a\x4d\xbf\x1c\xa6\xf6\x14\x9a\x90\x08\xe7\x26\xa8\x49\xdf\x53\x42\x70\x11\x0e\xb8\xa0\x87\xfd\x35\x6a\xe9\xa8\xb6\xc1\xca\x9e\xe9\x89\xcb\x47\x9a\x65\x24\xfe\xb0\x2c\xbf\x94\xab\xc3\xda\x9a\xe4\x7c\x8a\x82\x33\xa9\xf5\x7a\xd0\xf9\x5d\xe1\x94\xf9\xc2\x7b\xd2\x20\x70\xa3\x08\x65\x01\x20\x67\x7d\x27\xf8\xc2\x16\x14\x5c\x63\x58\xa1\x67\x02\x29\xd5\xb9\xad\x1c\x65\x74\x6f\x7d\xb6\xa1\x3b\x63\xd3\x76\x75\xe0\x4a\xa9\xd6\x8d\xcc\x78\x1f\x02\xb7\xfe\x5e\x92\x3a\x46\xbc\x43\x5e\xec\xb9\x69\x14\xb8\xb3\x8b\x79\x83\x11\x37\x04\x37\xb8\xdb\xa0\xc6\x42\xbe\x42\xbe\x5a\xbe\x91\xd6\x5c\x16\x9a\x6d\xb7\x19\x8f\xc3\x02\x29\x01\x5c\xb7\x1f\x58\x09\x04\x79\xed\x5a\xdd\xf3\x18\x2f\x34\x71\x40\xac\xb1\xcc\xb3\x8c\x0b\x85\x1a\xda\x30\x6e\x7c\x33\x3e\xb8\x73\xec\x3c\x3c\x8f\x83\x46\xb4\x84\x21\x6b\x6a\x69\xb4\x83\x47\xb0\xeb\x5a\x30\x8e\x30\x40\x16\x14\x41\x5f\xf7\x28\x2d\xa9\xd4\x25\x0a\xad\x13\x7e\x77\xc9\x30\xc8\xf4\x05\xdf\xb6\x0c\x5f\x5d\xef\xd7\xae\x85\xda\x2d\x4f\xc8\x54\x8d\x6b\xbd\x3e\x2b\x0c\x9c\xba\x45\xd6\x84\x28\x43\x67\xf3\x3d\xb4\xd8\x5e\x4b\xf8\xa3\x0d\xec\xd1\xaa\x41\x60\x21\x10\x9c\x2b\x23\x9f\x16\x3a\x0c\x72\xab\x09\xe6\x05\xdb\xa9\xcd\x05\xf3\x42\xa0\x96\xf9\x8d\x3f\xd4\xa7\x4d\x1d\x8f\xd8\x00\x02\x28\x0a\x45\xc4\xa5\x88\x81\x16\xb0\x56\xfe\x2f\x15\x1d\x7d\xd5\x68\xcd\x66\x84\xd7\x82\xee\x6d\xbd\xfc\x64\x81\x8a\xc2\xb2\xa5\xef\xda\x9c\x1e\x63\xf5\x76\x22\xa0\x99\xb0\x2d\xe3\xae\x48\x66\x2d\xf3\x66\x9e\x45\xa4\x0f\xc4\x87\xe9\xae\x8e\xd1\x4f\xce\xf2\x03\x81\xaf\x45\x4d\x66\x1b\xbb\x91\xe0\x85\x03\x85\xac\x5b\xd8\x7d\xe0\x2c\xee\x3b\x14\x76\xf5\x02\x7b\x40\xa7\x9a\x55\x2e\x29\xe0\x8c\x19\x8b\xec\x06\x69\x1f\x67\xfe\xa3\x3b\xb2\x3a\x2a\xe0\x23\x94\xe1\xb4\x91\x55\x20\x74\xb0\x64\xf1\xbf\xcc\x66\x99\x4c\x54\x57\x58\xcb\x96\x45\xb4\x1e\x54\x7d\x7e\xc0\x03\x68\x12\x55\xd1\x94\x26\x09\xc8\x01\xc7\xa8\x0f\xe5\x81\x21\x91\x53\x5f\x85\x2e\xc0\x82\xce\x18\x5f\x97\x63\xd6\x40\x4c\x51\x40\x4c\x77\xcd\xc4\x24\x81\x9a\x8a\x3c\xfe\xfd\x50\xd4\x1e\x30\x5d\x34\x6f\xc1\xcb\x88\xd8\xed\x91\x5c\x36\x50\xde\xdf\x22\x3a\x7a\x69\xb8\xc1\x87\xff\xaa\x1f\xfa\xa7\x1c\x0b\xcc\x14\xc4\xfc\x5a\xd1\x5d\x90\x20\xf5\x88\x7c\x86\xf8\x0c\x66\x0c\xc1\xf0\x53\xb8\xb9\xce\xe5\x0f\xe1\x5e\x88\xc6\x3d\x44\x8f\xc9\x31\x54\x67\x13\x5a\x96\x98\x14\x6f\xce\xb5\xe4\x30\x62\x4b\xb1\x8c\xc7\xa8\x9f\x48\x6e\xbf\x20\x2c\x4a\xa0\x1c\x77\x10\x9e\xec\x29\xdf\xba\x95\x26\x0b\x50\x50\x60\x2b\x8b\xe6\xb9\x7d\x10\x7c\x08\x45\xc6\xc0\x27\x9e\xc0\x49\x2f\x7e\xff\x3d\xcf\x8c\xb7\xa2\x29\x4e\xe2\x05\xcb\x39\x2c\x5d\x43\x2f\xb6\x49\xa6\x54\xe0\xaa\x0d\x82\x37\x60\x63\x8a\x18\xd3\x00\x81\x05\x7d\x8d\x15\x4a\x08\x96\x0a\xfd\xe1\x9b\x8d\x62\x43\xdc\x04\x0b\xee\x6a\x8f\x6f\x91\x28\xe6\x52\x0d\x42\xe1\xce\x77\x0c\xb5\xe3\xb0\x50\x08\x23\x46\x9e\xc3\xc8\x52\x0e\xc1\xc0\xae\x20\x1c\x09\x72\x5b\x4d\x3c\x99\xc9\xcc\x87\x6c\x0d\xa3\x32\x35\xf0\x11\x07\x77\x6c\xdd\xa7\x76\x58\x35\x94\x65\x95\x27\x1b\xe2\x09\x90\x5c\x45\xd0\xff\x1c\xab\x11\xb3\x9c\xd5\x85\x8d\x04\x69\x5e\xfd\x24\x29\x07\xda\x63\xc8\x25\x61\x7a\xc2\x50\x9f\xfd\xd8\x2f\xd0\x15\xa8\x5f\x3e\xda\xb9\x64\xa7\x2b\x0e\x8b\x89\xc7\xf3\x78\x47\x61\xdb\xb5\xd2\x4e\x9d\x7d\xf9\x15\x85\xe0\x9a\xee\x2f\x4c\xa1\xfc\x16\xc2\x30\xa9\x1b\xf2\x9a\x83\xb5\x6c\xd3\x5f\x21\x1b\xef\xbb\x83\xf6\xa2\x72\xbd\x7d\x1c\xae\xd9\x67\x5e\x63\x6e\x6f\xd8\xdc\x40\xb6\xd8\x45\x01\xf7\x61\xf7\xaf\xe5\xf1\x2d\x0d\x7d\x18\x43\xd2\xdf\x7a\x2e\x58\x24\xd1\x39\xd6\x61\x62\xaf\xe3\x20\xa7\x27\x48\x21\x80\xe0\x3f\xc7\xf8\xec\x9b\x0d\x9e\xd7\xec\x7d\x4f\xbf\x57\xcc\xdf\x4d\xc5\x07\xc1\x2d\x4f\xbc\x59\xd8\xeb\xc7\x7f\xc7\x11\x44\xfa\x43\x4f\x2e\xc7\x60\x19\x90\xc9\xc1\x58\x63\x30\xe6\xd7\x8a\x87\x99\xe0\x11\x91\xf2\x18\x0d\xe0\xa2\xb1\xff\x44\x78\xea\x1c\x12\xc1\xcb\x23\xa6\x35\x13\x87\xdf\x12\xb4\x5f\x26\xf1\xba\x13\x60\xc0\xe0\x76\xf2\xe5\xa4\xeb\x6b\x94\x34\x69\x13\x0e\x8b\x0e\xda\x80\xb2\x06\x68\x30\x3b\x45\x31\x8f\x1e\x89\x38\x11\x24\xa6\xf2\x14\x7c\xeb\xaa\xd1\xa9\x97\x6a\x6d\x7b\x67\x49\xa3\x29\x50\x60\x4d\x52\xdc\x99\xe9\xdf\x06\x58\xbb\xf0\xda\x1e\xa2\x53\x50\x27\x5c\x4e\x86\x09\x42\x76\x70\x37\x84\x29\xb1\x80\xb8\x7e\x6f\xca\xaa\x2c\x84\xd3\x34\xb4\xd0\xd6\x94\x4d\x24\xf6\x11\x83\xb3\xe5\xb4\xef\xe7\x44\x12\x17\x70\x60\x26\xa5\xb8\x8d\x65\x36\xec\x22\xc3\x6a\x2e\x21\x75\xb5\xbc\x06\x56\xe9\x82\x4f\xf5\x0a\xe1\x0c\xe2\x15\x8c\x95\xa2\xf8\xc8\x27\x58\x4a\x45\x93\x64\xc4\x18\x21\xb1\x44\x90\x65\xfa\x55\x6d\x86\xbc\xfe\xb4\x87\x70\x1c\xa3\xff\xfd\xf5\xc7\x8b\x9f\xef\x07\xe3\xe1\x15\x18\xad\x87\x17\x83\x6f\x7a\xfe\xc7\xeb\x87\x7b\xff\xab\xb1\xb0\x3c\x11\x81\x52\xfc\x08\x2a\x1e\x93\xc4\x26\x4f\x90\x11\x0b\x47\xea\xb0\x03\xf4\x13\x49\x5c\xa4\xab\x15\x53\x3c\x84\xa2\xdd\xc3\xc6\x8a\xc5\xc6\xe6\xb7\x81\xf2\x7b\xeb\x3f\x59\x4d\x83\x8e\x78\x7c\x17\x4e\x0c\x84\x1c\x19\x2c\x83\x64\x72\xab\xfb\x16\x04\x47\xd8\x8c\xb2\xa6\x78\x3c\xc2\x9e\x5e\x52\x88\xff\x81\x2c\x7e\xd4\xea\xf5\x0d\xa6\xa2\x35\xed\xd5\xa3\x01\xb9\x13\xa3\xf5\x74\x2c\xab\x87\x4a\x1a\x59\xd8\x64\xdb\x34\xc6\x7c\xd6\x01\xc1\xbd\xf9\x74\x2d\xbc\x14\xf9\xac\x84\x43\xa9\xf0\xf9\x1c\x0e\xca\xc9\x5f\x34\x05\x0d\x8e\xd8\xfd\xf5\xf9\xf5\x29\x22\x09\x9e\x70\x08\xe5\xb7\x21\x41\xae\x09\xbb\x60\x11\x4f\x83\x86\x4a\x08\x25\x3d\x94\x15\x08\x25\xa1\x11\xed\xd8\xb4\xb1\x06\xa9\x24\xe3\x62\x19\xdf\x63\xbf\x2a\xa0\x9d\xec\x0d\x17\x6d\xae\x7f\xfd\x1a\x2c\x1d\xcf\xb4\x22\x57\xe1\xbc\xf6\x6e\x9e\x12\x6c\x6a\xe9\x1b\xb7\x90\xb5\xe5\xdb\x00\xd6\x24\x29\xd5\x53\xd4\x07\x47\x1e\x5b\x17\x7c\xf1\x26\x67\xe8\x87\xbf\x48\x34\xc9\xd5\x88\x95\xdb\xe0\x0c\xf5\x7f\xba\x43\xdf\x61\x15\xcd\xbf\x19\xb1\x6b\xad\x66\xfe\xf0\x97\x06\x28\xa5\x8d\xd1\x09\xf5\x9a\x9c\x63\x85\x2f\x38\x8e\x29\x9b\xd5\x41\x13\x16\xf5\x63\x06\xf7\xfd\x53\x74\x6d\x75\xf8\x22\x13\xc4\xa7\x04\x07\x0d\x01\x43\x86\x89\x38\x2e\x02\xac\x9c\x95\xe1\xdb\x8c\x66\x06\x17\xd6\x88\xdd\x1b\x4c\x46\xcd\x55\xa9\x42\x19\xb7\x35\x8c\xb4\x56\x66\xd0\x2a\xb1\xcb\x90\x22\xc9\x02\xe9\xd5\x01\x32\xf6\x9b\x61\xe5\x31\x90\x67\x96\x99\xfd\x88\x81\x82\xee\x73\x53\x12\x1e\xe1\x04\x62\xf2\x8e\x02\x9b\x9e\x56\xdb\x79\x0e\xf9\xe1\xa6\xe8\xf9\xa2\x1c\x3a\xeb\x21\x0b\xbc\x50\x16\x6e\x14\x18\x00\x60\x1f\xad\x37\x36\xe5\x9a\xe3\x18\x2c\x36\x30\xbe\x25\x66\x75\xf4\x87\x1e\x9b\xcd\x2c\x8b\x7e\xea\xd3\xb6\x78\xce\x1c\x16\x49\x04\xe6\x7b\xb6\x80\xf0\x6d\x28\x3a\xc2\x21\xf4\xa3\xe0\xce\x96\x28\x97\x76\xd1\xdf\x89\xc1\x67\x23\x66\x22\x05\x4b\xfb\x12\xa2\xf7\x04\xbd\x73\x06\x81\x8c\xcb\xb9\x62\x79\x66\x03\x1b\xad\xac\x9f\x09\x72\xe4\x33\xa0\xe2\xd2\x9a\xea\x1b\xf6\x18\xdd\x86\xea\x75\xcc\xa3\x3c\x75\xc8\xca\x90\x3d\x55\x94\x95\x2f\x49\x3c\xe6\x62\x5f\x47\xf1\x80\xd2\xa2\x08\xa4\x8f\xb7\xd6\x8f\x0d\xc1\xf4\xc3\x4f\x97\x25\xf5\x66\xc1\x17\x78\xc7\x6e\x51\x6b\xa6\xa1\x71\x56\x6e\xa9\xd4\xda\xce\x79\x89\x57\x05\xfa\x2b\x17\x20\x6c\x91\xcf\x19\x07\x23\xb7\x49\xcf\xe2\xf1\x57\x12\x0d\x6f\xb4\x04\xa4\x35\x5e\x7f\x06\x73\xa9\x4c\x70\x19\xa4\xeb\x98\xaf\x4d\xba\x40\x0f\x7d\x8b\x46\xf9\xb7\xdf\xfe\x29\x42\x9f\xdd\x1f\x7f\xfe\xcf\xff\xfc\xd3\x9f\x37\x49\x27\x71\x0a\x39\xb4\x5b\xac\x91\x2f\x27\x55\x16\x89\xc2\x1d\x58\xe6\x54\x3b\xec\x82\x3d\x80\x4d\xcb\xbf\x0d\xca\x63\x10\x3b\x84\x67\xf6\x84\xcb\xf0\x64\xa2\xd2\xd1\x2c\x22\x09\x24\x51\xbd\x32\x87\xf0\xc2\xae\x95\xe8\xff\xd7\x0a\xb0\xb2\xb1\x3e\x2a\xdb\xc5\x38\xd1\xc4\x8b\xd7\xba\x11\xf4\xb5\xb5\xff\x29\x70\x20\x7e\xe3\x2e\x38\x9e\xc4\x44\x98\x31\x79\x93\x9d\x37\x24\x02\x73\x20\x9f\xb3\x84\xc7\x0e\x1e\xb5\xc8\x05\xa4\x20\x20\x0c\x3e\x63\xcd\xb9\x7b\x16\x46\xcb\x7c\x64\x3c\x2f\x53\x1c\x19\x54\x50\x89\xbe\xfe\x7c\xaa\x7f\xeb\xa1\xc5\x29\x04\x91\xf6\xd0\xaf\xa7\x16\x2d\x07\x0b\x35\xd6\x3f\x7d\xe3\x64\x6d\xdb\x04\x0c\x9a\x4a\xf4\xd5\xc9\x13\x16\xa6\x66\xf4\x89\x19\xd1\x57\x96\xb3\xfa\xba\x78\xa1\x6c\x9e\x70\xfe\x68\x03\x6c\x97\x3e\x3c\x71\xc0\x6b\x40\xde\xde\x6f\x62\xb6\xde\x27\xe6\x2b\x74\x04\x2f\x10\x74\x9c\x4d\xd0\xf1\xdf\x25\x67\xe8\x78\x81\xd3\xc4\xfe\xea\x9e\xda\xf8\x5f\x2c\x6d\x4e\x5c\xec\x83\x7c\x92\x85\xb1\x94\x7e\x97\xf0\x09\xcc\xea\xd2\xcd\xd4\x44\xd0\xc2\x40\x8b\xdb\xa7\xb8\xb0\xec\x44\x5c\x22\x2a\xe0\x07\xa5\x5c\x99\x57\x80\xc7\xd5\xcd\xea\xb3\x1f\xd2\x7f\x1b\xbf\x30\x2c\x8a\x4b\xe2\x33\xc6\x61\x1f\xbd\xa6\x1b\xfd\x8c\xbe\xb6\x2c\xe8\x1b\x7d\xc7\xd8\x70\x65\xb3\x0c\x75\x1d\x2c\x7c\x07\x3f\x07\x1d\x50\x86\x4c\x5a\xe6\x8a\x2f\x7f\x3d\x39\x3e\x3e\xf6\x5f\x43\xd6\xfa\xff\x8b\xa8\x92\x24\x99\x9a\x96\xdc\x0d\xb6\x18\xb1\x4b\x57\x78\xc1\x19\xaf\x0b\x48\xc7\x4c\x70\xc5\x23\x9e\xa0\xa3\xc2\xa0\x1b\xf3\x48\xa2\x7f\xd7\x62\x6d\xb0\x94\xf0\xa3\xd6\xe3\x1a\x60\x60\x0d\xd2\xf3\x2b\x1d\x2a\x6b\x10\xaf\x1e\xab\x10\xc5\xcd\x2b\xb6\x58\x86\x55\x3c\x80\x16\x34\xe5\x9c\x58\xa4\x37\x21\xf4\xcb\xe4\xb3\x82\x47\x0d\x40\x7a\xb5\xa1\xec\xf5\x37\xe5\x12\xbb\x2d\xf0\xf4\x0c\x59\x37\x2c\x80\xc5\xbb\xb2\x9c\xc1\xcc\xb3\x17\xba\x4f\xf4\xe5\xc2\xc2\x52\x00\x32\x4f\x53\x2c\x16\x27\xc5\x69\x5b\x26\xce\x02\x69\x0d\x78\x4c\xe2\x16\x00\x5c\xb8\x89\x3d\x5a\x36\x8a\xc1\x8a\x97\xee\x46\xf3\x67\x37\x82\x5a\x86\x01\x62\x01\x61\x11\x8f\x2d\x5d\x17\xd9\xa7\x65\x89\xc5\xbf\xb3\x2c\xab\xb8\x88\x18\x59\x18\xe3\x98\x32\x10\x1e\xf6\x0d\xf7\x71\x03\xfb\xe6\x63\xa8\x8a\x4b\x66\x1b\xb8\x47\x87\xd7\x77\xee\x9b\xf6\x97\x2e\xac\x43\x59\x64\xc7\x49\x88\x8f\xc7\x66\x48\xe0\xe7\xe2\xfa\x85\xd8\x0e\x63\x9d\xc9\x7d\x6e\xae\xf9\xf7\x19\xbf\xa1\x89\xbe\xb5\x80\xc6\x8f\x47\xac\xf4\x73\x0f\x91\x84\xa6\x94\xf9\xd8\x3a\xc3\xdc\xf9\xd4\x48\xcf\x8f\x54\xe9\x2d\x93\xf1\xa3\xe6\x60\x0e\xd7\x29\x50\xa9\xfa\x6c\xe1\x48\xc7\x3b\xa6\xac\x05\x22\x97\x7a\x5c\x85\x8e\xae\x85\x59\xdd\xc4\x91\x15\x48\x69\x40\x78\x70\x7e\x47\x4c\xb7\xe6\xce\x52\x11\x2e\x1c\xb4\x17\x34\x77\xe4\x00\xf1\x03\x0e\x00\x7d\x94\x62\x7e\xbd\xfc\x5b\x23\xa0\x0c\x58\x9e\xee\x9a\x6c\x62\xc3\x87\xdf\xca\x4c\x77\x23\x88\xbb\xa9\x6c\xe2\x12\x61\x79\xea\x0e\xd4\x06\x14\x37\xb0\xe2\x4f\x4c\xa2\x04\x1b\xa4\x1a\xdd\x10\x44\x3e\xf6\x8c\x83\x34\x0b\xfa\x32\xd7\x8b\xe9\xc6\xd4\xd8\x49\x08\xfb\xda\xfc\xfb\x1b\x64\xef\x86\x6f\x7b\xf6\x3e\x17\xd2\x23\x80\x98\x3d\x87\x1a\x8d\x24\x36\x36\x74\x40\x25\x9e\x61\x11\x1b\x6b\x79\xa8\x55\x98\x0c\x5e\x2d\x7f\x2d\x78\x8e\x9e\xa9\x9c\x8f\xd8\x3d\x77\x06\x47\xc4\xb8\xc7\x75\xee\x81\x32\xba\xd4\x1f\x96\xc0\x04\x60\xd4\x75\x14\xa0\x99\xf0\x4e\xb9\x46\x10\x05\x3b\x66\x3c\x26\xbb\x01\x18\xdd\x17\xbe\x0a\xe7\xbf\x16\xc4\xe4\x83\xc1\x4d\xd1\x94\x4e\x4b\xa4\xdc\xd0\x36\x5f\xdd\x78\xb8\x87\x6c\x3b\x50\x12\xf8\x79\x23\x74\xed\x10\x1b\xcc\xdf\x6a\xd0\x8a\xd3\x38\x83\x6c\xe0\xd2\xda\x7b\xb4\xe4\x5d\x37\x21\xaa\x41\x2b\x6a\x75\xf7\x9b\xb9\x47\xb0\xec\x3e\xc0\x18\xa3\x99\xe0\x79\xe6\x53\xe6\x5d\xba\x9f\xd9\x06\x2b\xd3\x0c\xd9\x94\x9f\x5a\x9d\xea\x82\xb2\x47\x43\xf1\x2f\xb5\x47\x06\x10\x9b\xc4\x25\x18\x37\x57\xa5\x15\xe6\x70\x84\x28\x8b\x92\x1c\x2e\x3e\xa9\x70\xf4\x68\x40\xbd\x9b\x8c\xbe\xfa\x9b\xf1\xfa\x64\xca\x06\x89\x29\x4f\x12\xdb\x6d\x71\x81\x16\x65\xac\x9f\x28\x46\x18\x3d\xdc\x0e\xeb\xfb\x7e\xa4\xcb\xce\x9c\xfa\xdb\xb3\x4c\x20\xf0\x3f\x3f\xd0\x8d\xe2\x2e\x2b\xb0\x78\xa4\x44\xea\xde\xb8\xd4\x04\xba\x6a\x88\x54\x69\x05\x22\xbe\xad\x31\xed\x6f\x4c\xa7\xb3\x2c\x1f\xeb\x85\x4a\x36\x09\x10\xd0\xa3\xf8\x74\xf3\xd0\x0f\xbe\x5b\x45\x2a\x9f\x6e\x1e\x50\xd0\x87\x01\x3c\x4c\x48\xa4\x7c\xa4\xf1\x31\x3a\x2b\x70\x88\xab\x92\x79\x4c\x9e\x68\x64\x52\x5c\x7b\x5a\x2a\x1a\x31\x80\xf7\xd4\xba\xce\x91\xc3\x84\x42\x9f\x6e\x1e\x2c\x92\x94\x87\x9b\xb2\x90\xca\x00\x61\xb1\xd9\xb5\x53\x01\xd6\x64\x9c\x1d\x81\xc4\x86\x45\x5c\x78\x3b\x7a\xa0\x5c\xff\x7f\xec\xbd\x5b\x77\xdb\x48\x76\x36\x7c\x3f\xbf\xa2\xd6\xca\x85\xdb\xdf\x47\x51\xed\x9e\x37\x59\x1d\xaf\x95\x0b\x5a\x96\xa7\x35\x6d\x4b\x1e\x1d\xba\x27\x6f\x98\x45\x17\x81\x22\x89\x08\xac\x82\x51\x80\xd4\x4c\x26\xff\xfd\x5d\xb5\xf7\xae\x03\x8e\x04\x48\xc9\x76\x26\x7d\x91\x4c\x5b\x04\x0a\x75\xae\x5d\x7b\x3f\xfb\x79\x22\x9e\x15\x25\x19\x18\x0f\xaf\xa6\x76\x4c\xae\x7d\x24\xc4\x54\x4b\xcd\xa5\xb1\x95\x30\xcb\x00\xd4\x3f\x9a\xfa\xf4\x2d\x9d\x7a\x0c\x38\x00\x3a\xed\xa8\xcd\x3f\x71\x19\x7e\x5c\xee\x18\xcf\x97\x49\x91\x9b\x6b\x18\xbe\x0c\x43\x61\xef\x1e\x4b\x7b\xa3\xf2\x96\x11\x09\xb6\xc0\x00\x27\xb2\xd0\x73\x19\x64\xb0\xb8\xac\x60\x4c\x5e\x48\x24\x03\x3a\x3c\xc0\xde\x58\x7a\xae\x28\x55\x65\x6c\x8f\xd5\xdc\x09\xc0\xec\x32\x34\xa2\xe6\x12\x98\x49\xcc\xd9\x0a\xc2\xe7\xfe\xec\x7f\xcd\x3e\xc9\x87\x24\x4e\xf8\x49\x21\x74\xca\x4f\x8a\xff\xf3\x69\x52\xfb\x13\x7f\xf5\xfd\xf7\x9f\x50\xcb\xa6\x8b\x76\xc1\x4d\xa3\xa3\x1d\x3c\xed\x71\x0a\x5b\xfc\xc2\xcc\xd2\x23\xc6\xe9\x7d\x72\x2f\xd8\x27\x1c\xee\x4f\x44\xe0\x77\xd8\xb0\xcd\x65\xdb\xb8\xb1\x43\x86\x0d\xe8\x54\xdb\xc7\x8d\xf5\x0c\xdb\xab\xf5\xf4\x1f\xd7\x4b\x33\x5a\x3f\xac\xa7\xaf\xbe\x87\xff\xac\x8d\xd1\xbe\xc5\xeb\xb2\x67\xda\xaa\xdd\xb2\x11\xb5\x2c\x4b\xb7\x17\xcd\xe5\xfe\xcd\x88\x8d\xdb\x8b\x60\xd6\xb6\x2d\x7c\x5e\x88\x63\xb3\x5b\x91\xd7\x71\x04\xfa\xba\x41\x98\xd9\x1b\x11\x3c\x92\x6d\xd2\x33\x45\x02\xdc\xb3\x9b\xf6\x32\x04\xe0\xc2\x8f\x23\xf8\x78\xe0\xf9\x61\xed\xa9\x3d\xbb\xa7\x39\xfd\xd5\x4c\x85\x18\xc1\x20\x73\x63\x1e\x1f\x58\xc9\xca\xa3\x7d\x75\x7c\xe4\xa8\xa8\xd3\x24\x72\x8f\xe9\xb6\x3e\x66\x15\xd9\xe9\x88\x2e\x13\xed\xf2\xfe\x5c\x4d\x2c\xb4\xd2\xdd\xaf\xed\x77\xd7\xb4\x96\x42\xc1\x1e\x17\x75\x6b\x99\xf8\x81\x2b\xe2\x48\x28\x9c\xb9\x52\x2f\xb6\x83\x49\x42\xfd\x87\xdf\xd2\xcb\x1f\x1a\x94\xa1\xce\xbc\xfc\x00\x99\xd9\x8e\x0c\x6b\xcb\xa5\xb1\xd6\xec\x57\x3b\x02\x4b\x78\xcb\x3f\xa8\x4a\x77\xd9\x41\x15\xc2\x2f\x0e\xd4\xb2\xa5\x4f\xd9\x52\x1e\x31\xb6\xca\x53\x8c\x1d\x14\x1b\x70\x2b\x7b\x0d\x38\xbb\xcd\x79\xf7\x32\xea\xc5\xa5\x3c\x5f\xa3\xd3\x4b\x8b\x42\xbf\x6c\x19\x61\x9f\xc7\x76\xc4\x08\x1f\xa0\xb3\x1d\x62\x59\xc0\xa5\xd2\xb7\xd2\x5c\x2d\xab\x84\xd1\xee\xa6\xe5\x14\xeb\x03\x36\x54\x9f\x5c\x17\xa9\x1c\xd5\x15\x62\xb3\x56\xba\x79\xb0\xc6\x4a\xbd\xd7\xe3\x71\x7c\xeb\x58\x5e\xac\xea\x3b\xe5\xec\x62\xe5\x96\x02\xb8\xce\xbb\xeb\x30\x5e\xd7\xbd\xa7\x0a\x24\x74\xdc\x55\x83\xb9\x9c\xd9\x47\x3c\x63\xa3\x4e\xd0\xcb\x82\xe9\x88\xe5\x12\x33\x5c\xc0\x67\xc6\x7d\xaf\x53\xe3\x3a\x1a\x31\x36\x21\xbf\xd6\x84\x3b\x2d\x72\x7f\x1a\x91\x0a\x9c\xe3\x65\xee\x55\xe0\x7b\x10\x79\x1b\xec\x78\xc4\xc7\x6f\x6c\x13\xa9\x28\xdb\x97\x6d\x1f\x1e\x7e\x51\xa1\x82\x20\xe1\x88\x50\xb5\x98\x17\x90\xee\xfc\x34\xf5\xc4\x9f\xb5\x8f\x35\x57\x6b\x71\xd4\x6e\x9c\xf0\xed\x22\x57\xdd\x12\x85\x03\xba\xc9\x16\x51\xf1\xd9\x6f\x50\xb2\x68\xc7\x3e\x97\x3c\xc5\xc3\x4d\xd2\x74\xb4\xd5\x06\xf7\xc7\x0f\xff\xc4\x66\x70\xfa\xb0\x0f\xb0\x2f\x02\x68\x0b\x4a\x2b\x14\x4b\xb6\x99\xc8\xb5\x92\xbc\x53\xab\xf3\xfe\x47\xbd\x20\xbd\x31\x73\x35\x56\x65\x53\x5b\x6c\x44\x4b\x5a\x4a\x0b\x1b\xc5\xd9\x7d\xb9\x14\xb9\x14\xa8\x47\x0a\xcf\x31\xfb\xdc\xa0\xea\x2a\x5e\x16\x9b\x1f\x16\x51\x9a\x0c\x16\x41\x83\x8c\xd1\x99\x79\xed\x0c\xdf\xea\x6b\x40\xa5\xfc\x4a\xd5\x25\xc3\xdf\x18\xfe\x36\x65\x6f\x78\x74\x2f\x64\xcc\xb2\xb4\x5c\x27\x44\x10\x83\xe6\x7e\x52\xbd\xd8\x57\x1b\x86\xb6\x05\x96\x6f\x8e\xa1\xb9\xdc\xf2\x7b\x24\x26\x27\x23\xd2\xdc\x1c\xba\xe8\x05\x9d\xab\x64\x91\x34\xe7\xee\xde\xd1\x72\xe7\x61\xb3\x98\xfa\xdc\xd3\x25\xe6\xcb\x3d\x6e\x14\xa1\x8c\x2a\x9e\x9a\x11\x0b\xd7\xcd\xd6\x06\x8f\x97\xe5\x5a\x71\xca\xb4\x54\x19\x5c\xbd\x10\xc2\x03\x72\xfd\x52\x32\x0e\x54\x60\x2f\x34\x2b\x33\x6b\x9f\x41\x6c\x29\x05\xa4\x0f\x0e\x81\xf9\x21\x4b\xa2\x7b\xc4\x96\x42\xf6\x04\x73\xcd\x6b\x08\x18\x32\xe1\x41\x8e\x6d\x5b\xc3\x0a\x89\x70\x8e\xc3\xad\x34\xb8\xf9\xf7\xcc\xd3\x81\x99\x21\xc5\x46\xc8\xc5\x01\x14\xf1\xc3\x07\xad\x92\x05\x42\x66\xb0\x8b\xd1\xb9\x2e\x2c\x65\x42\xa2\x80\xfe\x8e\xed\xf8\x8f\x93\x55\xcd\x8c\x4e\x34\xd3\xbc\x48\xb4\xd9\xcb\x5a\x7b\xdc\xd3\x0f\x1d\xd3\xeb\x7c\x1c\xe7\x51\x0b\xdf\x51\xad\x2f\x5c\xa6\xd9\x94\xbd\x83\xc8\x46\x70\x33\x50\x8e\x3d\xa8\x6b\xc3\x2a\x36\xa2\x93\x46\xf7\x29\x20\x9a\xb6\x05\xc1\xf3\xbd\x01\x2b\x97\x55\x38\x65\x33\x1f\x51\x46\xfe\x24\x8c\x15\xef\x69\x91\x48\xb5\x38\x64\xf2\x0d\x0a\xbe\x00\xea\x0a\x26\x10\x03\x4b\x4a\x9b\xbf\x7b\x55\x10\x57\xcd\x47\x48\xdc\xe7\xf7\x42\xf6\x79\xd8\x87\xd7\x10\x43\x20\xbd\x2e\x01\x17\x5b\x51\x18\x5e\x39\xa4\x82\xc3\x97\x9d\xa7\xac\x4a\x56\xa7\xa6\xcb\xcd\x35\x24\xba\xa7\x74\x41\x8c\xb0\x11\xe9\xd5\xe3\x46\xe9\x70\x9d\xd9\xf1\xc3\x9b\x6c\x5e\x3a\xe5\x07\x48\xb7\x74\x1d\x8c\x38\x4b\xa9\x42\x4e\x2c\xa8\xb5\x5b\xa4\xe8\xd6\x71\xe3\xcd\xec\x16\x0a\xdd\x00\xc8\x04\x5b\x54\x73\x35\xff\xfc\xa3\xbe\x82\x15\xfb\x14\xec\x2b\xed\x52\xdb\xc7\x67\x3e\x1d\x18\xf3\x75\x98\x5e\xaf\xd3\xcd\x63\xc7\x17\x94\xa9\x98\xf9\xe9\x35\x5e\x94\xfb\xeb\x37\xab\x26\xe6\x3d\xa8\x6d\xfb\x66\xf6\x87\x00\x18\xc6\x96\x65\x92\xc6\x48\x9f\x17\x18\x84\xca\x5a\x1c\xc0\x69\x0f\xc7\x7f\xa2\xdd\x79\xd2\x3e\xc7\x3e\xaa\xf8\x98\x89\x35\x9e\x22\xb5\x39\xaf\x07\xe4\x8d\xe8\x10\xbc\xb3\xdd\xdf\x13\x99\xea\x46\xfc\xc7\x0b\x5d\x15\x71\xeb\xa9\x30\x40\xbc\x96\xe5\xea\x06\x14\xa3\xba\x58\x88\x02\x31\x15\x9b\x56\x6c\xc6\xd9\x7c\xc6\x25\xb9\x75\x0d\x0a\x21\x86\xfc\xf1\xcf\xd9\x9f\x6f\xae\x2e\x4f\xb6\x3c\xd7\x1b\x0e\x2c\x0f\xb6\xac\x89\x15\xe1\xc4\xeb\xb1\x45\x32\x24\x72\x2e\x4f\xd8\x5a\x4d\x10\x37\xf3\x9a\x6d\x8a\x22\xd3\xaf\x4f\x4f\xd7\x49\xb1\x29\x97\xd3\x48\x6d\x4f\x7d\xd7\x9c\xf2\x2c\x39\x5d\xa6\x6a\x79\x9a\x0b\xc8\x9c\x38\x79\x35\xfd\xe1\x15\x8c\xcc\xe9\xc3\xab\x53\x40\x4b\x4c\xd7\xea\x1f\xde\xff\xf0\xcf\x7f\xfc\x27\x53\x70\xb6\x2b\x36\x4a\xbe\x26\x50\x4e\x6f\xd9\x27\x68\x95\x9f\xe2\x2b\xb5\xaf\xfc\xf3\xf4\xfb\xb0\x1a\xf4\xe8\x56\xc5\x22\xd5\xa7\x0f\xaf\x16\x76\x60\xa6\xd9\xee\xf7\x5c\x83\xaf\x96\x6b\x70\x9f\x14\xbf\xe7\x1a\x7c\xd5\x5c\x83\xe1\x16\x8e\xdb\x63\x80\xbc\xd9\xef\x8f\xe6\xef\x6e\x8f\xb4\xae\xf7\x7d\xfb\x50\xcb\xe1\x10\x66\x82\x1d\x71\x44\xdc\x8b\x51\x57\xec\x5a\x73\xdd\xd5\xa1\xc3\xc5\x36\x56\x40\xa5\xd3\x98\x1f\x45\x7c\x01\xc8\xbe\x24\x02\x72\x7e\x74\x09\x66\x3c\x69\xcb\x20\x20\x04\xeb\x31\xfd\xf7\x9c\x32\x13\x4f\xad\x2f\x41\xcd\x3d\x50\x5b\x22\xc5\xb7\x2d\xde\x56\x3d\x5a\x4d\x89\xa7\x50\x62\x18\xa8\xcd\xed\x08\xe6\x71\xf2\x40\x5d\x6c\xbd\x3a\xaa\xb1\xe1\xfa\x30\xe0\xf6\x0c\x69\x5c\x5d\x9c\x0e\x51\xaf\x89\xb6\x1f\xb4\x07\x87\x65\xc6\x31\xe7\x90\x25\xe0\xcb\xca\x3c\x53\x5a\xe8\x29\x7b\x57\x53\xaf\xf5\x60\xf4\xeb\x77\x67\xec\xd5\x8f\xff\xfc\xc7\xb9\xfc\xae\xe5\xdc\x86\xfd\x5e\xe5\x6b\xc2\xc6\xc3\x69\xbd\xe5\xba\x10\xf9\x69\xbe\x8a\x4e\x71\x97\x3b\x35\xef\x9f\xd0\x47\x4f\xd4\xea\xc4\xd1\xcc\x9f\x10\xe3\xf6\x74\x1b\x8f\x23\x8d\xa9\x4c\x3d\x3c\x6b\xe8\xa0\xd1\x70\x28\x21\xbd\x9c\x5a\x39\x41\x11\xcc\x5d\x44\xed\x21\xb5\x6a\xf9\x8f\x37\xa9\x5a\xea\x97\x8e\xd4\x92\x6b\xfb\x0d\xcf\x32\xd7\xbd\x34\x9f\x46\x71\xc2\x4e\x91\xe7\x74\x54\xd8\xbd\x24\xbc\x8e\x8c\xe9\xf8\xf6\xc5\xe6\x8f\x7b\xe4\xd8\xe1\xb9\x2a\xa5\x65\xec\x57\x52\xa8\x15\x40\x37\xc0\x12\xb6\xc8\x33\xf0\xd5\x02\x9e\xc9\xf1\xe9\xe4\x22\xc3\x03\x06\xa2\x0a\xdd\xdd\x7d\xa4\x6a\xc5\xbe\x7e\x7e\x0e\xd5\x8a\x63\xfb\x9d\x36\x94\xaf\xd4\xe1\xc7\xc2\xc3\x71\x29\x8d\x41\x55\x98\xe7\xf7\x46\x50\xdd\x3e\xe0\x15\xe5\x3c\x41\x7c\xc6\x73\x30\xd2\xc4\x49\xa1\x4e\x80\x88\x0c\xe8\xad\x50\x47\xa6\x0b\x56\x01\x91\xe7\x31\xc7\xa4\x79\x7e\x40\x3d\xd1\x30\xff\x2d\xa8\x28\xd9\x24\x1a\x69\x99\x09\x66\x9b\x48\x29\x72\x8a\xa9\xed\x3d\x51\x47\xc6\xa5\xc3\xa1\xec\x47\xd9\xfa\x9b\x68\xa8\xf1\xe1\x72\xac\x78\xb0\x09\x4c\x19\x58\x9f\x1b\xb5\x55\xc6\x9c\x51\xa5\x0e\x7e\xc4\xdb\x0b\x1c\xc2\x9d\xb6\xd7\x96\x67\x48\x3c\xfa\xf5\x5a\x63\x96\x96\xf9\x09\x9d\x7a\xe1\x43\xa3\x64\x93\x96\x55\xa1\x98\x3d\xf5\x77\x0a\x1f\xfd\xf3\x06\x50\x0f\xa8\x3d\x0a\xf2\xdf\xc4\xdb\x9f\xfc\xa7\xb9\xd7\x98\x29\xe5\x6e\x0a\xee\xe4\x46\x90\x0e\xf2\xeb\x86\x90\x34\x6b\xcd\x77\x32\x60\x94\xdb\x91\x63\xe0\x12\x47\x86\x0c\x00\x97\x98\x4a\x61\x73\x28\x4e\x5a\x93\x28\xba\xd6\xa5\x15\xdb\x8e\x17\x96\x03\x7a\x5c\x55\x6f\x5c\x01\x44\xf7\xdc\xac\xb7\xa7\xd0\x83\x8c\x1b\xec\x63\xdc\x10\xac\x6d\xd1\x05\xdc\x1c\xbf\x18\x41\x34\x6b\x4c\xdf\xc1\x47\x70\x72\x36\x7a\x30\x58\x0b\x5d\x1d\x38\xce\xc5\xd6\xe7\xb1\x6a\xc3\xec\x22\xeb\xa8\xcf\xc8\x34\xb5\x6c\x5c\x1e\xdd\x8b\x0f\x6e\x7d\x21\xa4\x71\x59\xc2\xef\x97\x57\xb7\x21\x5a\x23\xc1\xd6\x9e\x44\x1b\x11\xdd\x83\xc3\x04\x8f\x3c\x27\xc4\x4b\x0c\xa7\x73\xe9\xe5\x1c\x0b\x65\xa1\x07\x3b\xa7\x70\xe1\x54\x5e\x54\xce\xe2\x44\x67\x29\xdf\x41\x90\x57\x62\xee\x95\x0f\x10\xbb\xa4\x45\xb3\x15\xec\xf3\x17\x0f\x1f\x69\x33\x2a\x33\xff\xde\xd8\xbe\xf4\x60\x5a\xdf\x99\xcd\xfd\x80\x69\xb1\xe5\xb2\x48\xa2\xb9\xdc\x0a\x2e\x43\x54\x1e\x05\xb9\x4d\x27\xc7\x4a\x10\x07\xfc\x6a\x25\xa2\xc2\x93\xc8\x82\xf1\xee\x7a\x6a\xdf\x1a\x1c\xd7\x76\xb7\xf2\x7a\x9b\xfe\x13\xe0\x7e\x31\x46\x9f\xab\x07\xda\x86\xed\xd1\x78\x60\xf0\x06\xe4\x3f\xe9\xc8\xb5\x97\x41\xf8\x97\x9d\x53\x6c\x29\x8a\x47\x01\x1c\x29\x94\xd4\xdd\x66\xe3\x1f\x2d\x01\x73\x9c\xa2\x7b\xbb\x16\x7e\x80\x04\xc3\x05\x16\x82\xc9\x1c\x99\x9b\xac\xb1\xb2\xbd\xa0\x34\x73\xf0\xf6\xbc\x20\xbf\xd5\x0b\x38\xa6\xcd\xed\x31\x7f\x10\xf1\x5c\x56\xa9\xf2\xc8\x66\xf4\x0b\x8e\x79\x71\xc3\xa7\xd9\x6d\x6c\x1f\x0f\xf2\xe5\x9f\x03\x3d\x90\x27\x06\x76\x89\xd4\x3d\x62\x8b\xd8\xe8\xe7\xbc\x55\x59\x9d\xd7\xd0\xba\x1f\x00\xc9\x12\xda\x8a\x97\x91\xd6\x69\x05\x4f\xe1\x26\xa5\x23\x02\x43\x96\x50\x07\x80\x25\xbf\x64\xc3\xd3\xd9\x56\xc6\x5c\x5a\x86\x8c\x55\x99\x22\xf3\x73\x57\x1e\x02\xf1\x02\xda\x6c\xbe\xaf\x97\xd5\xe9\xfc\x6a\x2c\xd0\x8b\x74\xb0\x87\x00\x8c\x8c\x7b\x9d\x9d\xf5\x42\xea\x12\x4c\x0a\x2b\x15\x07\x8e\xe7\xb5\x28\xe0\x34\x8f\xcb\x14\x09\x1f\xc0\x63\x0e\x1c\x83\x3c\x4d\x59\x52\xe8\xb9\x74\x94\x88\x98\x6c\x00\x3b\xac\x75\xa9\x5b\x95\x71\xe9\xb4\xca\xe1\x67\x2e\xc1\x0e\x4b\xa2\xa4\x68\x40\xb8\x77\xa1\xbc\x52\x96\x09\x8e\xf9\xc9\x38\x6c\x73\x19\xde\xb9\xea\x83\x40\xc9\xbc\x3c\x4d\x78\x8f\x0a\xfc\x53\x4c\xdd\x99\xf9\xc4\x41\xb8\x05\x6c\x9d\xb9\x70\x59\xe5\x64\xac\x2d\x71\xa2\x10\xd2\xd2\xdc\x6a\x0a\x6d\x7d\xe4\xfe\xde\x0a\x79\x0a\x51\x99\xf2\x1c\x13\x34\x56\x65\xca\x92\x55\x20\x02\x0d\x63\x80\x84\x78\x66\xb8\x22\x05\x67\xb5\xf5\x92\x6b\xbe\x15\x01\x17\x07\xb9\x77\xd2\x00\x43\x81\x2c\xff\x18\x9c\x37\x65\xbd\x9c\xb2\xb7\x9e\xf2\x13\x47\x18\xd6\x44\x40\xa4\x9b\x68\xdc\xfe\x5c\x7d\x83\x34\x72\x68\x9d\xa9\xa2\x92\x66\x45\xba\x55\xd7\x31\x82\x20\xc8\x31\x0e\xa0\x61\xe5\x58\xfa\x51\xc3\xad\x34\x12\xe6\xd5\x1a\x6c\xc3\x2d\x88\x8e\x0a\xda\x53\x61\x64\x25\x43\x12\xe2\x03\x2a\xea\x48\x9e\x5b\x2a\xbb\xed\xd1\x9c\x86\x71\x1c\x59\xd5\x40\xc1\x6d\x7c\x45\x83\x99\x13\xc2\x71\x86\xf4\xec\x9a\x17\x63\xb1\x39\x2e\x19\x67\x7c\x45\x5b\x71\x50\x43\xaa\x09\xbb\xc7\xc8\x7a\x3a\xa1\xfd\x03\x2a\xea\xe4\xfc\xbd\xc6\x0b\x6c\x15\x82\x47\x9b\x6a\x5e\xbc\x65\xaf\x75\x2d\x80\xbc\x28\x58\x8f\xe3\x53\xfa\x67\x7e\xce\x81\x88\x1d\x33\xd5\x9f\xb2\x2b\x29\x10\x39\xa7\x56\xc1\xa1\x42\x15\x20\xb5\x3b\x10\x10\x71\xbb\xdc\xd2\x54\x4c\xde\x5b\xba\x20\xb3\xe4\x26\x8c\xfb\xd2\x61\xd7\xc3\x69\x83\xbb\x48\x87\x2d\xd9\x26\xb7\x73\x84\x79\x39\x2c\xe9\xbe\xfd\xce\x1f\x00\x50\xc7\xef\x00\x6d\xed\x18\x3e\x2c\xbd\x48\x72\x77\x8b\xb3\xf0\xf1\xea\xbc\x61\x08\x27\xdd\xd7\xbf\x1f\x37\x55\x14\xe2\x08\x71\xba\xbb\xcb\xb7\xe7\xef\x2e\x2e\xab\x8a\x72\x7f\xb9\x3b\xbf\xab\xfe\xe5\xfa\xee\xf2\xf2\xe2\xf2\x4f\xe1\x9f\x6e\xee\xce\xce\xce\xcf\xdf\x56\x9f\x7b\x37\xbb\x78\x5f\x7b\xce\xfc\xa9\xfa\xd0\xec\xcd\xd5\x75\x4d\xc3\xce\x0a\xd0\x05\x7f\xba\xbd\xf8\x70\xfe\x76\x71\x75\x57\x91\xc1\x7b\xfb\xaf\x97\xb3\x0f\x17\x67\x8b\x96\xfa\x5c\x9f\x9f\x5d\xfd\x72\x7e\xbd\x47\xc5\xce\xb7\xb7\xb5\x4b\x9f\x02\x3e\x76\xb0\xa6\xe1\x8c\xad\xf2\x44\xc8\x38\xdd\x21\xf6\xde\xde\x6c\x6b\x60\xda\xf0\xec\x4d\xb6\x42\x95\xc7\x40\xe8\x6f\x37\x82\xa9\x07\x91\x03\xb3\x11\x96\x46\x34\x08\x3e\x8b\xba\xfe\xd5\x5c\x14\x79\x33\x2a\xd0\x9b\x29\x54\xe4\x3b\x97\x8b\xd6\x57\x1d\xcf\x8a\x47\x1f\x61\x99\xc8\xfb\xea\x02\x96\x51\x5e\x66\x45\xb2\xec\x4e\x8a\x18\x9d\x4c\x3c\xf4\xee\x8d\x1c\xae\xed\x84\x57\x97\xed\x1b\x63\x25\x37\xe0\x18\xe0\x31\x94\x70\xa8\x54\xa7\x7b\xdb\x82\x35\xb3\x72\x99\x26\x11\x4b\xe2\xba\x3f\x85\x72\xfc\xc1\x65\x5c\xa7\x7a\xce\x44\x0e\xa6\xaa\xb9\x01\x64\xb9\x38\xe1\x65\xb1\x41\x5a\x42\x4a\x45\x20\x61\x8e\xb9\xd4\x22\xca\x05\xc6\x02\x84\x06\x27\x2d\x6a\x34\x06\x5f\x82\xca\x10\x2b\x47\x0c\x04\x60\xd3\x40\x76\xa3\x23\x46\x80\x6f\x62\xe9\x23\x9c\xa4\xf8\x7c\x6f\xd7\x50\x8d\x13\x5d\x17\xe8\x87\x13\x1e\x7f\xb4\x4a\x8f\xa6\xdd\x66\xa7\x76\x4a\x87\x38\xc8\x36\x77\xa3\xbd\x19\xfb\xe6\x58\x38\x51\xaa\xc9\x0c\x54\x3a\xfd\x74\x96\x0b\x38\x44\x08\x0a\x60\xfd\x17\x00\x5d\xa1\x5c\x0f\x48\xf1\x30\x57\xb5\xa5\xd8\xf0\x74\x85\x16\x87\x19\x9a\x76\xa6\x04\x2c\xff\x56\xdd\x0b\x79\x8d\x03\xf6\x55\xb6\x43\x89\x37\x1f\xcf\xd3\xe2\x3c\x42\xde\x85\x69\xea\x68\x67\x95\xcd\x75\x03\x63\xaa\xc0\x7b\x42\xf0\x33\xa6\x74\x78\x16\x76\x9b\x26\xb7\x5a\x25\xbf\x99\x02\xe7\x52\xb4\xf2\x50\x03\x5e\xc8\x32\xe6\xb9\x7d\x19\xb0\x51\x48\x3b\x76\x2f\x24\x68\x44\xa2\x84\xfc\xde\x39\x3b\xce\x7f\xde\x1c\x8b\x1e\x87\x3e\xf8\xfc\x92\x8a\x74\x66\x18\xe5\xb1\xfd\x54\x60\x8e\x8d\xe3\x15\x80\x79\x73\xf6\xfe\xe2\xfc\xf2\x76\x71\x76\x7d\xfe\xf6\xfc\xf2\xf6\x62\xf6\xfe\x66\xe8\xf2\x7b\x8a\xbc\xa8\xda\xea\xab\xa7\x07\xb9\x1d\xe2\x94\x56\x9e\x4f\xcf\x75\x8d\xf2\xcb\x0e\x86\x64\x7f\xed\x93\x38\x5b\xc4\x89\x8e\xcc\xf1\xb7\x5b\x08\x19\x03\x81\xff\x41\x53\xb5\xbd\xa8\x7a\x2b\xdc\x13\xcc\x3d\x61\x77\x10\x3c\xed\x1e\xec\x8c\x76\xbf\x03\xea\x0e\xdc\x90\xb9\x30\x8b\x3f\xae\xf0\x26\x4c\xf7\xab\x36\x99\xe2\x8e\x6b\x5b\xb5\x88\x7a\x9b\xb0\xbe\x89\xd6\x25\xd0\x33\xd8\xc7\x00\x72\xd8\xd1\x2b\xc4\xaa\x1a\xaa\x08\x24\x81\x02\x36\x4b\xf4\x5c\x6e\xb9\x8c\x79\xa1\xf2\x5d\x47\x13\x87\x6d\x9e\xe1\xb2\xa9\x6e\xa1\xe1\x91\x2d\x85\x88\xed\x28\xe0\xa3\x5c\xd6\xa7\x12\x6a\x0d\xdc\x5e\xfd\x7c\x7e\x79\xb3\x38\xbf\xfc\x65\xf1\xf1\xfa\xfc\xdd\xc5\x5f\x1d\x12\x32\xe3\xba\x4d\xf1\x36\xcb\x85\xd9\x5d\x2c\x75\x53\xeb\xfe\x82\x32\xb4\xb6\x1c\x92\x1e\x4c\x56\x73\x69\x77\x96\xdc\x17\xbf\xc9\x55\xb9\xde\xb4\x17\x54\xaf\xe5\xc7\xd9\xed\x4f\x07\x55\x13\x88\xf5\x50\xab\x12\x57\x5b\x13\x11\x9a\xac\x68\xdf\x43\x18\x69\xad\x7a\x40\x0f\x09\x8f\xb6\x45\x19\x3a\x76\xb4\x83\x6e\x2f\xcd\x4d\xab\xd7\xf8\x6f\x79\xbc\x6b\x02\xdd\x06\xfb\x66\xe5\x18\x01\x84\x32\x4a\x1e\x37\x4a\x7b\xdd\xf2\xb7\xca\x09\xf6\xc3\x49\x2a\xd6\x6b\x11\xe3\xf4\xaa\x17\x4c\x3e\x38\xda\x02\x23\x7f\xae\xb7\xf5\x22\x89\x92\x1e\x71\x30\x3b\xbc\xd7\xf0\x0d\xfc\xa3\x7b\xa5\x7d\xaf\x38\x23\x72\x1c\x88\x6f\x16\x5c\x76\x04\x92\x1f\x9a\x08\xcd\x41\x5b\xd1\x55\xce\x5c\xf2\x13\x39\x4c\x6c\xc8\xc0\xaf\x83\x2e\xc0\xcb\xf1\xb8\x50\x57\x8f\x6b\x91\xa5\x3c\x12\x2e\x87\x01\x59\x4d\xe1\x5e\x7f\x48\x00\x8f\xa4\x5f\x25\xf9\x5b\x02\x49\x58\xaf\x76\xd5\x36\x05\xc0\x73\x7b\x6d\xf7\xe3\xe7\x77\xad\xf4\x5e\xdc\x88\xcb\x10\x1c\xcd\xa8\xbd\x47\xd0\x77\xf4\x45\x81\xa0\x65\x27\x2c\x79\xd4\x74\xa8\x7d\xf9\x17\x1a\x78\xbc\x33\x57\x1d\xdd\xdc\xb2\x85\xba\xe9\xe1\x4c\xc7\x3e\x7f\x61\x51\xe4\xbd\x04\xc3\x4f\x11\x8e\xf8\x98\xab\x6d\xa2\xc5\xac\x28\xf2\x64\x59\x86\x0a\xab\x23\x01\x73\x95\xcb\x89\x6f\x70\x96\xab\xb8\x8c\x2c\x25\x10\xb4\xd6\xc3\x7e\xc8\xcb\x67\xad\x8e\x98\x9d\x98\xd9\x47\x37\x37\x11\x9f\x00\xa0\x1f\x39\xab\xda\x62\x6c\x76\x63\xec\xf0\xfd\x7d\xb4\x47\xf9\x31\x53\xb2\x65\x52\x74\x77\xa6\x9d\x03\xc3\x12\x6b\x99\x7d\x1c\x2c\xe0\x0e\xd4\x14\x4d\x97\x25\xc7\x00\x7a\xd5\x46\xe9\x62\x00\x71\x47\xcd\x38\x70\xd7\x30\x6c\x4c\x35\x63\x06\xed\x86\x0d\xd7\x68\xce\x17\xd1\xa6\x5a\x71\x68\x4d\x95\x09\xb5\x5e\x5d\x67\x1e\x1f\xe7\x36\x19\x14\x46\x9b\xa0\xa3\x21\x21\xc7\x76\x45\xd5\xd2\x49\xf4\x8e\xf3\x76\x87\x16\xa3\xbb\xd1\xe1\x61\x00\xfb\x68\xca\x4b\x19\x6d\x58\x96\x72\x4c\x26\xdf\x70\x8d\x53\xda\x62\x49\xf8\x32\x49\x93\x02\x58\x7a\x30\xc4\x59\xeb\x61\x73\xcd\xe3\xf9\xbd\x25\x3b\xe7\x9e\x92\xa9\x6f\xd2\x1f\x89\xd9\x75\xad\xfa\xa2\xa8\x5d\xbf\x64\xc3\x6d\x68\xd8\xb4\x24\xc4\xae\x1f\x0e\xb3\x11\xc3\xb4\xf4\x6d\x19\x37\xb2\x54\xe2\xc7\xfa\xeb\x95\xfe\x6e\xb1\x5e\xc6\x23\x56\x48\xc5\x63\xc4\xe9\x53\xd7\xf8\x68\x5d\x59\xab\x54\xf1\x0e\x9d\x79\x5b\x36\x4a\x76\x74\x95\x1d\xab\x72\xd9\x45\x12\x8f\xb5\xea\x2f\xbd\x2f\x18\x62\xd7\xed\x53\x39\x4b\xc3\x0d\x90\x17\xa2\x48\xc6\xf9\x7b\x82\x46\xf3\x42\x9c\xc0\xeb\xed\x85\x53\x86\xe1\xe0\x36\x37\x26\x9a\x17\x8e\x72\x46\x1b\x60\x09\xdb\x66\x57\xed\x74\x3e\x06\x13\x7e\xe4\x78\x25\x72\xcf\x54\xda\xaf\x45\xf3\xc7\x1f\x5a\xba\xa5\xd1\xe8\xbf\x94\xdc\xec\x87\x57\xab\x1b\xe4\xca\x39\xa6\xd1\x45\xd2\x5c\x56\xed\xdb\x4f\xfd\xab\xb7\xd5\xf0\x5a\x38\xf1\x07\x67\x22\xb7\xb5\xe6\xc6\xbc\x3d\x7c\x17\xba\xa8\xb8\xd1\xb2\x3c\x51\xc0\x19\xa3\x56\xc8\xc0\xd8\x4d\x21\xdc\xfa\xdd\x23\x7a\xf2\x73\x29\x4a\x61\x26\xd0\xb2\x8c\xd7\x4d\x2f\xf7\x08\x4b\xd9\x37\x69\xa3\x1e\xd9\xb6\x8c\x36\xcc\x16\xce\x62\x91\xf2\x5d\xa5\x69\x60\x24\x16\x0a\xf8\x3c\x47\x51\x67\x05\x2c\xcc\x51\xa9\x0b\xb5\x05\x80\xb1\x2f\x37\x2f\x25\xac\x72\xc6\xed\xea\x6a\xdb\xdf\x2b\xec\x72\x07\x86\x36\x6f\x3e\x9e\x9f\x5d\xbc\xbb\xa8\xc5\x15\x67\x37\x3f\x87\xff\xfe\xf5\xea\xfa\xe7\x77\xef\xaf\x7e\x0d\xff\xf6\x7e\x76\x77\x79\xf6\xd3\xe2\xe3\xfb\xd9\x65\x25\xfa\x38\xbb\x9d\xdd\x9c\xdf\xee\x09\x30\x36\xbf\xda\x3d\x10\x3c\x20\xbf\xb3\x90\x67\xab\xec\x60\xfd\x0c\xf4\xd5\xd7\x6c\x66\xa9\x00\x2b\x64\x95\x36\x48\x0c\xa8\x92\x14\xc1\x71\x18\x4b\x7e\xcb\x0b\x7e\xc6\x0b\x9e\xaa\xf5\x94\xcd\x18\x01\xc2\x11\xe8\xaf\x8d\x85\x44\x3c\x69\x66\x74\xb0\x08\x63\x26\x45\xfe\x0e\xef\xa5\x6b\xd5\x8a\x18\x0a\x53\x11\x8a\x9c\xd8\xac\xb6\xb9\x3c\x7f\x10\xb2\x28\x81\x84\x96\xa7\x29\xa3\xcf\xda\x07\x82\x8c\x7d\x5b\x4b\x9d\x6c\x93\x94\xe7\x5e\x65\xf4\x8a\xca\x82\x5b\x8a\xad\xab\x23\x68\x6a\xa6\x83\xdb\x8b\xdc\xdd\x05\x83\x7a\x9f\xbd\xbf\x00\xbb\x2f\x2a\xac\x84\x96\xfd\xf8\x5c\x22\x03\x1e\x7d\x71\xcb\x21\xf9\xa4\x50\xe4\x59\xc5\xcf\xd3\xc3\xdd\x13\xf1\x28\x2a\x70\x1b\x83\x78\xae\x1b\xa5\xab\xa4\xfd\x8f\x73\x59\xe4\xbb\xc1\xc6\xdc\x2d\x64\x5b\x6b\x30\xc8\x09\xcb\x56\x55\x1e\x45\xc7\x17\xb3\xa5\x5f\x82\x85\x67\x81\x96\x14\x97\x71\xe1\x17\xc4\xb5\x74\x5c\x3a\x52\x73\xf2\x7e\xab\xfd\x10\x12\xe2\x40\x2f\x2c\x55\x29\x63\x4d\xa8\xbb\x6d\x22\x4f\xb7\xfc\xb7\x97\xb6\xa5\x48\x30\xe1\xf4\x7f\x80\x3c\x4c\xa4\xe6\xfa\xb5\x33\x9b\x5c\x7f\x77\xcd\x65\x4f\x7f\xed\x37\x91\xed\xce\x0a\x77\x3d\x7f\x31\x47\xfc\xe0\x83\xd8\xb5\x8d\x5f\x43\xc3\x8d\x85\x44\xe4\x50\x48\x96\x0b\xf3\xa0\x03\x27\xa6\x88\x39\x75\xff\x86\x24\x84\x8a\xce\x6c\xfb\xde\x1d\xc6\xfb\x8f\x5a\x36\xad\x48\x83\xe1\x86\xcf\x60\x11\x3e\xfa\x92\x19\x33\xc4\x1d\x58\x97\x37\x25\x5d\x50\x40\xd5\x0c\xd6\x7f\xa8\x25\x5b\x41\x06\x12\x26\xda\xb1\x5c\x40\x88\x03\x86\xc2\xaa\x46\x00\xc5\x54\x03\xcc\x60\xa7\x40\x2a\x34\x38\xfe\xa5\xb9\x63\x8a\xcf\x25\xc5\x6e\x5f\x7d\x3f\xee\x9c\x2d\x90\x7a\x1c\xb9\x66\xeb\xa4\xdc\xee\x2c\x87\x7a\x95\x32\x69\xe3\x9d\xbb\x2e\xa5\x39\x8a\x9f\x02\xf6\x32\x3c\xae\x59\xfb\x28\xfd\x73\x6f\x92\x90\x75\xc9\xe7\xf8\xfc\xb3\xd1\x88\xfe\x52\x63\x0f\xa5\xcf\x01\x24\x9d\x4a\x0f\x0f\xb4\x25\x8f\xee\x1f\x79\x1e\xa3\xdf\x16\x70\x28\x53\xf6\x93\x7a\x14\x0f\x22\x9f\xb0\x48\xe4\x05\x27\xea\x2e\x0d\x81\x78\x58\x50\x54\xce\x5c\x42\x86\x06\xf2\xa0\x49\x5d\xe6\x82\x15\xc9\x7a\x63\x2e\xd1\x01\x8c\x42\xe5\x66\x3b\x2a\x90\xb5\x31\x13\x11\x91\x25\x75\x74\xc0\x2a\xe5\x0f\x4d\x2e\xb2\x43\x58\x1e\xd8\x85\x4b\x33\xb5\x71\x4a\xab\xc4\xd3\x07\x7c\xa1\x0e\xa3\x4d\x13\xe9\x6d\x26\x6c\xad\x52\x2e\xd7\xd3\xe9\x14\x58\xe7\x5f\x8e\x9a\xe8\x54\x60\x18\xf9\x74\xf0\xea\x54\x29\x2d\xd2\x9d\x23\xf8\x71\x09\x30\x80\xb8\xfc\xad\x10\x52\x27\xe8\xe7\x69\x99\xfe\x37\xf5\xa8\xc0\x97\x0d\xa2\xb4\x5f\xcf\x47\xa7\x57\x76\x94\x03\xc2\x7e\x23\x4a\xc2\xe7\xdb\x6f\x5e\x07\xa5\x0b\xb7\x97\x25\x95\x1c\x9b\x03\xfb\x8b\x4a\x3a\x62\xf8\x07\xf1\xee\xb5\x96\x44\x24\x25\x07\xe5\x0d\xb6\xf7\x59\x23\x95\xf3\x88\x2c\xce\x9e\x84\xcc\x91\xb9\x98\x43\x1c\x01\x37\xf5\xe1\x1e\xbd\x2c\xf6\x6b\x0d\xb5\x36\x68\x64\xae\xab\x4f\x4a\x1f\x63\x3a\x61\xba\x5c\xba\x83\x1b\x97\xcb\x7c\x05\x77\x7a\x1c\x84\x03\x2a\xd1\x0e\xc8\xc1\xf2\xe1\x12\x47\x1e\x15\x44\x47\x74\xa1\x72\xbe\x16\x6c\x2b\xe2\xa4\xdc\xb6\x6e\x36\xae\xba\xc7\xe0\xfe\x54\x5a\x6e\xbb\x69\xfc\x8e\x35\xa0\x7d\x25\xf1\xbf\xce\xe0\x73\x83\x0d\xe8\x99\x83\xb4\x5b\xc9\x37\xaa\x2f\xfa\xfe\xa9\xaf\xcd\x49\x99\x27\x1a\x08\x27\x0f\x49\x79\x74\xc5\x60\xd1\x10\x39\xdd\x65\xe8\x73\xae\x8c\xee\x89\x0d\x69\xd1\x2b\x1a\x47\x15\xc2\xad\xdd\x87\x42\x1d\x4d\x38\x5e\xf8\x29\x57\x65\x83\x6c\x67\x50\x84\x1b\xcc\xc6\x80\x06\x9e\xe0\x4e\x50\x20\x61\x32\x0a\xc5\x56\x36\x89\xee\x5e\x04\xb4\x64\x31\x10\xc4\x3f\x22\xc7\xcd\xcf\x3f\x6a\x8b\xde\x20\x80\x8d\xb7\x58\x0a\xff\x11\x0c\x88\x3c\xbc\xb2\xb8\x2a\x6c\x21\x16\x01\xe4\x61\x31\x97\x45\x6b\x01\x1e\x76\x08\x65\xe1\x2b\xbf\xf0\x32\x6d\x7f\x9c\xca\x87\x47\x51\x40\x70\xf6\xeb\x0d\xc3\xae\x26\x2a\xf1\xbc\xaf\xa2\x41\x21\xfb\x91\x5d\xd0\x5d\x8b\x03\x2c\xc1\xca\x38\x60\xa7\x5b\x2e\x79\xd3\xed\xa2\x88\x36\xde\xf2\x00\xf2\x34\x47\xfa\x46\xea\xb0\xd4\xce\xad\x27\x47\x47\xd0\x6c\x88\x3e\x4c\xd6\x52\x85\xba\x1e\x4a\x0a\x88\x4c\x99\x0d\x48\x85\xc5\xb2\xa4\xd8\x0f\xf1\x1a\xc9\x18\xb6\x6f\xaa\x15\x0a\xa1\x3b\xd4\xce\x4a\x80\x11\xae\x14\x09\xf2\x0c\x59\x7c\x2c\xde\x89\x48\x6c\xb4\x4e\x9a\x5d\x65\x6e\x98\xcb\xea\xa7\x1a\x9d\x64\x31\x58\x49\x2e\x90\xeb\x56\x1b\xeb\xad\x48\x1e\xcc\x42\x6d\x4e\x6b\x37\x41\x61\x07\x68\xce\xbd\xb9\xc4\x6a\x07\x84\xb9\xf7\x62\xa7\x43\x65\x53\x9a\x51\xac\x6b\x42\x26\xa6\x3d\x34\x5e\xfb\x87\x02\x3a\x6e\x91\x7b\x7d\xb2\x61\x67\x19\x7e\xf4\x83\x79\xb9\x07\xdc\xd9\x28\xdc\xcc\x41\x9f\xa5\xe8\x7d\x8a\xb4\x4d\xf8\x7e\xa6\x31\xf4\xf8\x2d\x40\xe7\x85\xf8\xbb\x30\xe5\x04\x2e\xbe\xe6\x7e\x3b\x97\xc4\xa9\x1d\x1c\x72\x66\xc3\x69\x0e\x1b\xa5\x4e\x23\x93\xef\xae\x42\xfb\x02\xb4\x87\x96\x02\xb2\xfa\x49\x1b\x6c\xb5\xc2\xd8\x73\x09\x9f\xc6\xe4\x52\xeb\xc3\x6b\xfd\xe0\x81\xa0\x40\x1a\xdc\x4e\x20\x60\x90\xc1\x85\x4f\x12\xf3\x1f\x4a\xe4\xe2\xed\x27\x12\xa6\xfb\x66\xb2\x15\x83\x67\x11\x78\x37\xe7\x67\xd7\xe7\xb7\x5f\x0c\x28\x68\x51\x7a\xa3\x91\x82\xb6\x9e\x6f\xcf\xdf\xcd\xee\xde\xdf\x2e\xde\x5e\x5c\x3f\x07\x54\x90\x7e\x3a\x00\x2b\x78\x43\x54\xfd\x67\x4a\x16\xe2\xb7\xa3\xce\xe4\xbc\x94\x0b\x3e\x22\x67\xc5\x89\x75\xf4\x99\x3b\x58\x68\x53\x6a\xc0\xe9\x00\x10\xef\x24\x9e\x68\x4e\x59\x60\xe5\x9d\x86\xab\x24\x4d\x21\x85\xd7\xb9\xd7\x29\x3d\xcc\x74\x2a\xec\x3f\x96\x6a\x93\xf6\xd4\xb9\x5c\x56\x94\x20\xc0\xe5\xb7\x31\x97\x60\x4c\xde\xcd\x4c\x07\xe4\x09\xa4\x46\xf6\xa9\x11\xac\x13\x29\x7c\x35\x50\xce\xbe\x94\xac\x93\x42\x9a\x06\xf1\x39\x21\x51\x64\x78\x0d\xb5\x35\xed\x8c\xab\xcc\x4f\x6b\x7e\xda\x1f\x5d\x0b\x71\x11\x27\x12\x0d\xd3\xca\x6a\xbe\x69\x9f\xba\xa7\x7e\x09\x40\xbf\x9b\x91\xe4\x10\x83\x00\xc5\x78\x3f\x90\x34\x10\xa8\x52\xe4\x83\x13\xf7\x09\x42\x87\xd4\xaa\xd6\xcf\x66\x2b\x34\x7d\x9d\x40\xa4\x82\x13\x2b\x49\x94\x96\xba\x10\x39\xb9\x4d\x66\xbf\xde\xcc\xe5\x1b\x73\x7c\xbd\xa4\x53\x88\x94\x6c\xf0\x13\x08\x5c\x51\x95\xef\x5b\x0b\x25\xdc\xc1\xbe\x43\x1f\xf5\x56\x70\xa9\x19\x2c\x8d\x34\x15\xb9\x9f\x19\x58\x1f\x21\x62\x52\x74\x05\x1a\x56\xff\xfe\x4b\x46\xa8\x44\xd3\x15\xa6\xbe\xf4\x6b\x2e\xb6\xaa\x68\xce\xa7\xae\x0c\x71\x80\x0a\x3f\xe7\xcc\x69\xc9\x58\x19\x3a\x8b\x08\x65\xdd\x3a\x89\xaa\xf9\x23\x83\xe6\xd2\x2d\x16\xf7\xfb\x54\x7a\xc2\xa9\x34\xe0\x5c\x0f\x4f\x09\xb6\x51\x66\x03\x75\x32\x2f\x3e\xcc\xec\x18\x2a\x52\x00\x7d\x99\x6e\x6c\x3d\x75\x6a\x52\x87\xc7\x60\x3f\xa0\xa8\xe3\xa0\xb5\xb3\x16\x2a\x1c\xaf\xa9\x65\x63\x3b\xbd\x2a\x8a\xcf\x43\x39\x37\xb3\x20\x43\xa9\x0a\x4b\x1e\xe1\x70\x7d\x04\x52\x34\x0f\x38\xd6\x92\xde\x3a\x12\x13\x88\xb5\x52\x16\x47\x2a\x91\xdd\x86\x60\xc8\x4a\x3a\x2d\xd6\x22\x4c\xc4\xb7\xc9\xf7\x8e\xbc\x63\xcc\xe4\x3b\x5c\xeb\xb2\x3a\xe7\x1c\x11\xe4\x41\x60\x87\xcb\xab\xcb\xf3\x10\xaa\x70\x71\x79\x7b\xfe\xa7\xf3\xeb\x4a\x22\xf6\xfb\xab\x59\x25\x99\xfa\xe6\xf6\xba\x96\x43\xfd\xe6\xea\xea\xfd\x79\x03\xf3\x70\x7e\x7b\xf1\xa1\x52\xf8\xdb\xbb\xeb\xd9\xed\xc5\x55\xe5\xb9\x37\x17\x97\xb3\xeb\x7f\x0d\xff\x72\x7e\x7d\x7d\x75\x5d\xfb\xde\xdd\x59\x3f\x7a\xa2\xd2\x8c\x76\xf7\x8f\x0f\xce\x06\x9c\x98\xad\xcb\xb8\xaa\x05\x7a\xc4\x2a\x1e\x88\x3c\xdb\x37\x1d\x6d\x9e\x75\x1c\x52\xe5\xe3\xc2\x30\x55\x1d\x35\xeb\x9e\x5e\xbc\xb4\xd2\x75\x19\x3f\x6e\xdb\x33\xa7\xda\xe2\x29\x90\x80\xbd\x06\xa0\xfb\x4a\xcd\x71\x4b\x5a\xc1\xd8\xb5\x19\x44\xb0\xd6\xbc\x53\xba\x47\xc6\xcf\x5e\x53\xfb\x8d\x7d\xf5\xf4\x1c\x4c\x7b\xa8\x6c\x9e\x8a\xc6\xa2\xaf\xd2\xc1\xc7\x6c\x96\x78\x12\x5b\x43\xc1\xfe\x18\x1c\xdc\xd0\x0c\x33\x73\x82\xe9\xd8\xa5\x32\xd9\x9e\x6f\xd2\x4f\x9b\x36\xb6\xfe\xf4\x91\x66\xdd\x6b\x1c\x1b\x23\xea\x0d\x5c\x47\x63\xea\x7d\xcb\xf5\xfd\xd8\x7a\xd3\x47\x9a\xf5\x06\xb3\xef\xa0\x7a\x83\xc3\xbb\x68\xe7\x3f\x19\xb1\x89\x85\xc5\x54\xab\xe7\x92\xb3\xdd\x23\x81\x98\xeb\xb0\x3a\x9a\x05\xf0\xbc\xd7\xcb\x8c\x0f\x0f\x64\x40\x6d\xdc\x72\xe5\x35\x3a\xf0\x1b\xf8\x15\x5a\xb8\xcc\x05\xbf\x8f\xd5\x23\x8d\x47\x1d\x19\xca\x06\xed\xe6\xd5\x0e\x32\x7b\xb8\x3d\x22\x8a\x9c\x22\x50\x88\x52\xf3\xc5\x03\x4c\x2e\x21\x42\x6b\xb4\xc1\x02\x15\xd2\x3a\x83\x0c\x70\xf6\x48\x3f\x3a\x73\x89\xd6\x7c\x9b\x92\xa9\x19\x55\x53\x23\xe2\x7c\x80\xa6\x3a\x1b\x1a\x83\xeb\x3a\x18\x58\x4a\xe0\x28\x73\x00\xd3\x2d\x73\xb8\x33\x41\x87\x24\x12\x9c\xc9\xb9\xb9\xf0\xe4\x22\x4a\xb4\x08\xd4\x9c\x5a\x4f\xec\xcf\xc7\x69\x3f\x14\xbc\x68\x75\xbb\x0e\xf6\x87\xf3\xa8\x28\x79\xca\x3e\x97\x22\xdf\x11\x75\x1e\xfa\x2a\xf1\x2f\x11\x97\x98\x29\x52\x88\x6d\x06\xe9\xd8\x61\x8a\xc3\x5c\xfe\x0a\x40\x09\x1c\x82\x17\x9a\xfd\x09\x20\x0f\xf6\x61\x3a\x84\xb7\xbc\x80\xb3\xf8\x2f\xf8\x0d\xf7\xdb\x74\x2e\x2b\xea\x28\xc1\x5b\x15\xa1\x94\xe9\x5c\x5a\x79\x82\x58\x45\x7a\x0a\x37\xbe\xa9\xca\xd7\xa7\x24\xec\x6b\x26\xbb\xba\x5f\x2a\x75\x7f\x2a\xe4\x29\xf8\xa4\x8a\x53\x5e\x16\xea\x14\xe0\x52\x38\xfe\xfa\xd4\xea\x7f\x5a\x01\x55\x7d\xba\x49\x1e\x04\xfc\xbf\xe9\xa6\xd8\xa6\xff\xa0\xb3\xcd\x6f\x27\xeb\x34\x3f\x31\xef\x9e\x84\xef\x9e\xd8\x77\x4f\xec\xbb\x27\xe6\x35\xfc\x7f\xd9\x0e\xc3\x3b\xe2\x37\x6e\xce\xb2\xc9\x5c\x26\x52\x8b\xbc\x00\xeb\xe7\x31\x4f\x0a\x2f\x43\xb3\x63\x2f\xfe\xeb\xbf\xd8\x34\xe7\x8f\x98\xca\xf8\x96\x17\xfc\x23\xfa\x17\xff\xfb\xbf\x5f\x40\x40\x15\x93\x7a\x32\x9e\x7f\x2e\x45\x31\x97\x5a\x98\x45\xc8\xfe\xbf\xb9\x84\x08\xec\x76\xb7\x28\xd0\xef\x8a\x3e\xc8\x58\xb3\x7f\xc1\x32\x2f\x90\x46\x32\xd6\xa6\xa4\x8e\x74\x82\x84\xa7\x2d\x92\xd1\x1d\x2e\xfa\xcf\xe9\x5b\x7a\x7e\xc4\xb2\xfe\x9c\x56\x57\xb5\x15\x42\xd1\x9f\x53\x38\x40\x53\xc5\x2d\x58\x8b\xb9\xc9\x0b\xf7\x64\xaa\x5c\xdb\x1a\x69\x40\x03\x9e\x35\x4c\xdf\xbe\x56\x6e\x90\xca\xda\x7a\xee\x1b\xdb\x08\xc4\x0a\x7c\x1c\x02\xa2\xe7\x89\x59\x21\x37\xe8\x09\x05\xcb\x0d\x5b\x0e\x36\x29\x85\xce\x5d\x79\xe8\xb8\xd0\x7f\x7c\x7d\x7a\x3a\x61\x6b\x0d\xff\xb3\xfc\x0c\xff\x03\xe8\xa1\xa7\x62\x63\x6d\x74\xa6\x03\xc2\x35\x47\x79\xff\x48\x3c\x05\x8a\xee\x4b\x10\x80\xd7\xa6\xe9\x9b\x52\xc6\xa9\xf0\x29\x90\x95\x90\x48\xaa\xac\x64\x3d\x3a\xc6\xea\x52\x2b\x30\xc6\x4b\x11\x71\xb3\xf1\x35\xbe\x8d\xe0\x52\xb5\x2a\x84\x44\x6f\x58\xee\x95\xd8\x38\x7a\xae\xc0\x2c\x06\x28\x24\x2f\x08\x72\x2e\xe0\x8f\xf0\x11\x60\xd4\x9e\xd4\x7f\x62\x3b\x55\x12\x39\x34\x50\x9e\xc6\x22\x4a\x81\x81\xdf\xd2\xbe\xb0\x5c\x14\x65\x2e\x19\x67\x19\x97\x31\xd7\x30\x03\x57\x39\x44\x3b\x73\xc6\x9b\x15\x9d\x20\x1c\x57\x95\x05\x90\x19\x21\xb2\x20\xec\x09\x64\xef\x0e\xea\x3c\x09\x2a\x81\x67\x02\x90\x08\x37\x5e\x9c\xce\xa5\xd5\x0a\x23\x2c\x1c\x7a\xca\x22\x95\xed\x88\xaa\xa6\xde\xe9\x89\xf5\x9c\x51\x77\x4f\x3c\xde\xa4\xfe\xec\x84\x25\xd5\xd0\x1a\x10\x85\x17\x81\xda\xb1\xd5\x8b\xfe\x4e\xc8\x48\xc5\x22\xd7\x2f\xcd\x32\x4c\xdc\xbd\x03\xed\x87\x44\xfb\xc1\x80\x5d\xca\x1c\x6e\xe4\x2d\x34\xc5\x3b\x45\x1d\xd3\x3b\x15\x6a\xe9\x36\x3b\x67\xff\x52\xf9\xd6\x51\x30\x6d\xf5\xa5\xff\xfc\xa2\x88\x98\x10\xd7\x69\xef\x9c\x87\xbb\x20\x70\xc9\x86\x3b\x2e\x16\x8a\x36\x0e\x19\x27\x56\x5a\x36\x29\x40\xbd\x2e\x17\xba\x98\x4b\x3a\x81\x27\x6c\x25\xb8\xb1\xf3\x26\x2c\xd2\x0f\xb8\x19\xe3\x71\x5f\x3c\x2a\x8f\xc1\xb1\xba\x24\x00\x86\xad\x14\xee\x9d\xc4\xf8\x18\x20\x0a\x78\x54\x20\xc0\xa0\x53\x85\xdc\x9a\x2a\xd0\x59\xad\x1b\xe2\x01\xfd\x60\x65\x2e\xea\x92\x52\xa1\xca\x0a\xf4\xc4\x0e\x03\xc5\xac\x5e\x0f\xfc\xc1\x6c\x3c\xd8\x3a\x84\x81\x04\x9b\x23\x58\xdc\x84\xa5\xc5\x75\xe6\x63\xb8\x21\xd7\x38\xf8\x66\xba\x16\x55\x4f\x47\x40\x05\x0e\xf3\x5b\x98\x57\xf7\x3a\xac\xb4\xc8\xad\x06\x07\xb6\x15\x99\x01\x37\x49\x1e\x9f\x64\x3c\x2f\x76\x76\xfa\xa6\xc9\x12\xa8\xfb\xd3\xe4\x5e\xb0\x59\x9e\xab\xc7\xa7\xee\x85\xce\xad\xa5\xeb\x86\x7d\x0c\x92\x7d\xec\x2d\xbf\x95\x17\xb4\xee\xee\x38\x8c\x83\xb4\xcb\xf1\xd1\xfa\x9d\x5c\x14\xf9\x6e\x61\x26\xe2\x36\xeb\xdc\x29\x06\x25\x4d\x0c\x37\x72\xc7\xd1\x9b\xd6\x5c\x18\x9d\xf4\xa6\x95\x51\xfd\x76\xe8\x4d\x5b\x98\x4b\x9b\xf4\xa6\x17\x97\x17\xb7\x17\xb3\xf7\x17\xff\xb7\x56\xe2\xaf\xb3\x8b\xdb\x8b\xcb\x3f\x2d\xde\x5d\x5d\x2f\xae\xcf\x6f\xae\xee\xae\xcf\xce\xfb\xf9\x8a\x9a\xb5\xf7\x26\xf8\x09\x0b\xbf\xf3\x9a\xdd\x06\x40\x0d\x4c\x36\x20\xfb\x9b\xb4\x2b\x61\x56\x99\xc5\x9c\xc8\xf5\x04\x16\xea\x6b\x76\x9e\xe7\x17\x5b\xbe\x16\x1f\xcb\x34\x05\x38\x15\x66\xf6\x9c\xe5\x02\x2e\x9e\x13\xf6\x51\xc5\x17\xc1\x7b\x90\x8e\xd8\xda\x0c\xf8\x3e\x8f\xe3\x5c\x68\x8d\x9f\x9f\xd0\xf7\x03\xf0\x90\x4b\x75\x24\xf0\x1c\x7f\xe0\x49\x6a\xee\x6f\xaf\xd9\x1b\x1e\xdd\xab\xd5\x0a\xd3\x67\x26\x2e\x71\x8a\x7d\x2e\x55\xc1\x99\xf8\x2d\x02\x8e\xae\xf6\x79\xf2\x5e\xad\xbf\x02\x54\x79\x40\x78\xaa\xe3\x92\x02\x1a\x65\x8b\xf6\xe3\xbc\x7d\x23\xa0\x56\x7e\xc0\x57\xdf\xe1\x9b\xed\x0e\xca\x22\x7d\x82\xf4\xf8\xf7\x6a\xdd\xae\x18\x03\xd6\x35\xc9\xdc\x50\x20\x21\x22\xb2\x0d\xb5\x66\x3a\x91\xf7\x73\xf9\xeb\x46\x48\xa6\xca\x1c\xff\x04\xd7\x7c\x63\x66\xa6\xa5\xde\x08\x90\x90\x9d\xb0\x47\xc1\xb6\x7c\x87\x66\x33\xdc\x09\x9c\xcc\x05\x4c\x19\x38\x45\xcc\xdb\x69\x22\xcd\x6e\x91\x25\x36\x2f\xa1\x3e\xf4\x4f\x71\xe3\xb2\x0c\x75\xfc\x78\x02\xd9\xbe\xf3\xb4\x82\xcf\x03\x57\x99\xc7\x4d\x5a\x80\x10\xed\xdc\xa0\xa2\xa9\xd4\x7d\x99\x79\x2e\xcb\x17\x36\x38\x09\xdd\xfd\xa0\x92\x98\xc5\x65\x96\x26\x91\xdb\x77\x1f\x55\xde\x49\xd8\x8b\x09\x34\xc3\x4f\x9d\x7a\x5a\x58\x5f\xc3\x5a\xb2\x73\x02\x24\x5d\x0f\x75\xef\x33\x93\x17\xb3\x44\x46\x69\x09\xfa\x60\xa5\x16\xf9\x49\x91\x27\xeb\x35\x18\xe0\x36\xd7\xef\xdb\x67\x37\xf6\xec\x89\xc7\xa7\xb5\x85\x49\xe7\xa9\x5a\x27\x11\x4f\x43\x70\xb3\x47\x45\x38\xfa\x54\xbb\xec\x49\x3d\x15\xf2\x20\x6c\x85\x3a\x19\x90\xb2\x5c\x00\x83\xef\x02\xb6\xf2\x05\x6d\x77\xc7\xd4\x7b\xc5\xcc\x05\x1d\xeb\x15\x92\x9b\xda\xf0\x82\x3d\xe1\xfc\xb7\xad\x84\x16\x98\x98\x28\xaf\xcd\xd4\xa3\x14\x39\x58\xb0\x00\xfb\x30\x2d\x95\x0a\x6c\x13\x27\xab\xe5\xf0\xc9\x56\x56\x6e\xe5\x80\xd8\x98\x39\xbb\x4e\x1e\x84\xfc\xf2\x6c\xd4\xc1\x07\x22\x1e\x6d\xc4\xc2\xda\xe5\x4f\xbd\x65\xb9\x03\x60\xe4\x66\x65\xf5\x2d\xc2\xad\xd4\x85\x37\xe1\xea\x84\x35\x6e\xee\x5d\x18\x48\xec\xc9\xc8\x32\x95\x58\xc4\x22\xba\xff\xe2\x5b\xb3\x07\x59\xd9\x8a\x30\xce\xde\x8a\xe8\x9e\xdd\x5d\x5f\x60\x36\x70\x52\x30\xb3\x15\xe8\x8d\xd7\xeb\xe9\xbc\xbb\x15\x7c\xfd\x0c\x8c\x4e\x43\x05\x87\x3c\xc7\xbc\x93\x59\x33\x15\x22\x40\x14\xe4\x4b\x9a\x4d\x92\x72\x69\x00\x08\xc6\x0b\x2b\x43\x03\x8e\x78\xa6\xb7\xa0\x3a\x53\x16\x81\x54\x5b\xca\x97\x22\xed\x60\x5c\xcc\x54\xbc\xb0\x71\x92\x63\xc1\x3c\x8d\xb2\xac\x1f\x83\xa2\x8e\x36\x8f\x81\x1b\x8b\xf5\x96\x1e\x64\xf7\x3f\xea\x80\x5e\x43\x85\xc4\xcf\x70\xaf\xe7\x1a\xd2\xbb\x57\xc9\xda\x46\xdb\x92\x15\x69\xe3\x60\x42\x3f\xa8\xc0\x9b\xfd\xd2\x94\xf4\x51\xc5\x04\xd3\x73\x24\x66\xc6\x0a\x12\xe4\x3d\xf1\xb8\x8a\xb0\x0a\x4e\x29\x5f\x83\x6f\x40\x17\x82\xc7\x4c\xad\xc8\x9b\x98\x65\x69\x02\x94\xbe\x31\xb2\x87\x03\x7b\x86\xae\xa2\xe3\xc3\xd2\x6c\x65\x03\x92\x8f\x8f\x16\x88\xd7\x1b\x6f\xf4\x41\x2e\xd3\xae\xaa\x93\xbb\x6e\x53\x1d\xab\x02\xe7\xf2\x91\x0e\xbd\x42\xf7\x7b\xd3\xd6\xa9\x5a\x42\x47\x75\x83\xe2\x7a\x36\x68\xb3\x3b\xe5\x49\x3c\xe6\x78\xb7\x7d\x72\xe5\x5e\xed\xab\xe0\x95\xf5\x74\xb8\x2f\xd9\x61\x66\x44\xb8\x1e\x46\xf0\x6b\x69\xec\xfb\xee\xda\x10\x20\xd4\x2e\x42\xe8\xac\xf1\x82\xe4\x06\x60\x55\xb8\xed\xb8\xe3\x5a\x5d\x6d\xcb\x51\x03\xdd\x24\x46\xd9\xd3\x97\x9e\x4b\xa5\x7f\x90\x8f\x60\xb7\xc0\x95\xeb\x28\x2e\xba\x1d\x69\x32\x16\xf1\xe2\x80\x36\x9c\xd3\xbb\xc3\xda\xe2\x7a\x1a\xab\x07\x2e\x2f\x79\x62\x4e\xc6\x98\xe7\xb1\x6f\xc7\x04\x6e\xc4\x11\xcf\xc0\xeb\x0c\x5e\xfc\x87\x57\x53\xfb\x8d\x6b\x9f\x4c\x63\xb6\x07\x4c\x71\x47\xb8\xb2\x6a\xd1\xea\xd8\x37\x8f\xdc\x24\x45\x34\xb3\x99\x39\x7e\xba\x56\xd2\x4c\x06\xcd\xdd\xfa\x0c\xb3\xfb\xd5\x31\x93\xeb\x39\xf6\x8e\xb2\x50\x3e\xb8\x01\xed\xb9\x00\x7a\xd4\x30\x81\x0d\x36\xc8\x8b\xb8\x03\x18\x61\xad\x4d\xbb\x09\x8d\x80\xdb\x8e\x02\xfc\x66\xb9\xb0\x61\xb2\x9d\x28\x1c\x8d\x41\x6a\xf5\xcf\x20\x0a\xe4\x5a\x5d\xe5\x71\xb1\x54\x0d\x8e\x7b\x0b\x62\x36\x64\xd9\x46\x6a\x9b\x29\x09\x28\x1c\x4c\xca\x9a\x4b\x2a\xdc\xaa\x58\xbb\x40\x52\x25\xb3\x6f\x42\xfe\x3b\xcc\x13\x11\x5a\xa5\x0f\x14\x31\x0c\xc4\x16\x40\xff\xce\x54\xf0\xcc\x5c\x85\xcc\xc5\x1f\x42\xd9\x74\x90\x01\xf0\xbd\x26\xe5\x9c\x8b\x75\xa2\x0b\x11\x26\x43\x86\xef\x3f\x99\xea\x66\xc5\x57\xd0\xd7\xf5\x9d\xaa\x9b\xfb\x8c\x7e\xb3\x3f\x8d\xa8\xcf\x2e\x13\xf1\x85\x7b\xaf\x7f\x32\xd4\xf2\xd5\xfd\x76\x58\x39\xef\x70\x0e\xe0\x65\x47\x23\xb3\x95\x76\x32\x09\x6e\x90\x88\x73\x88\x7b\xfc\x9e\x19\xa2\x75\xc9\x73\x2e\x0b\x21\xf4\x5c\x52\x9c\x15\x19\xda\x42\x12\x92\x1a\xee\xcf\x99\xf2\x91\xd2\x05\x12\x1e\xc1\x2b\x2b\x9e\xa4\x65\xde\x79\xbb\xc6\x59\x79\x10\xcb\x42\x5f\x2f\x9d\x41\xb1\xac\x6d\xd0\x5c\xbe\x6e\xb0\x8a\x1c\x49\x48\x3d\x4a\x5a\x4d\x67\xed\x68\x82\x3d\x5c\x86\x8f\xb7\x73\xad\x76\xa4\xf0\xfe\xa8\x17\x99\x1a\xb1\xe3\xfd\xfc\xa3\xfe\xa8\x3a\x92\x9f\xf5\xe7\x86\x0b\xb0\x07\x2d\xf0\xb9\x4b\x38\x82\xeb\x7b\x08\xb4\xed\xf3\x3c\x0c\x22\x9f\xdc\x1b\x8e\xeb\xdc\xbb\x60\xd6\x6e\xb8\x8c\x53\x73\x23\xe7\x45\xed\x04\xf2\xb0\x66\x73\x03\x28\xec\xe6\xd8\x9d\xc3\x06\x29\x21\x8b\xa8\x91\x4f\xb8\xaf\x9f\x6a\x89\x88\xbd\xd0\xc1\xda\x57\xaa\xe9\x81\x6d\x69\x29\xde\x86\x21\xb9\x56\xb7\x60\xbf\xba\xfd\x72\x1e\xd6\xfd\x0b\x99\x2f\xd5\xb5\xb6\x4a\xd6\xdf\xc0\xbd\xf9\x43\xf3\x48\x88\x68\xcf\xa1\x83\xda\x81\xf9\x8f\xdc\x75\x20\x6f\xca\xec\xda\x21\xdf\xf4\x5c\x92\x6c\x35\x06\xd3\x21\x8a\x8a\xf4\x62\x9a\xbd\x72\xc9\xb4\xaf\xfe\xd1\x92\x4b\xed\xd8\x0a\x26\x15\x30\xb8\xa9\x28\x2a\x73\x88\x74\x93\x37\x8e\x09\x3c\x84\xf5\x28\xde\x14\x30\x3d\x1c\x3e\x09\xed\xc4\x36\x33\xc9\xb9\x5f\x2b\x8d\xba\x05\xaf\x1b\x0a\x70\xbb\x43\x9f\x74\x95\x72\x5d\x30\x5d\x88\xac\x75\xfb\xad\x58\x97\x55\x8d\xf9\x23\xec\x4b\xaf\x70\x3f\x70\xe9\x8c\x38\x8c\x66\x81\x87\xe4\xcf\x37\x57\x97\x2c\xe3\x3b\x80\xfa\x15\x8a\xe1\xa3\xc0\xaf\x59\xdf\xa8\xf6\x8d\x40\xb5\xf1\xd5\x5d\x05\xfb\xd4\x62\x86\xdb\xdd\xf1\xf4\xc5\xa6\xb1\x08\x73\x86\xa6\xa4\xd9\xb3\x72\x95\x9e\x64\x29\x97\x01\x9a\x5b\x4f\x59\xed\xf3\x61\xf8\xde\x05\xf2\x08\x20\x05\x15\x00\x0f\x19\xcd\x85\xbc\x6c\xc5\xfb\x56\x65\xf3\x8f\x8a\xd8\x77\xee\x11\xbd\x38\xc6\x0f\xa8\x56\xc1\x23\xb3\x4c\x90\x2c\xc2\xa2\x10\x1c\x90\x85\x6b\xc0\x98\x8e\x18\xa8\x7e\x8d\xff\xb9\xb4\x12\xce\xea\x51\xb3\x18\xe9\x34\xca\x44\x6f\xc0\xe5\x8c\x31\x1e\xc0\x7b\xd1\xfe\x82\x60\x94\x9c\x4b\x6d\x06\x14\xdc\xd4\xe2\x41\x90\xaf\xaa\x12\x5f\xbd\x78\xfb\xde\x41\x36\x70\x90\x48\x65\xaf\xa3\xeb\x03\x0b\xf4\x98\x9b\x5a\xab\xe8\xfc\x7e\xad\x86\x0f\x3c\xeb\x4b\x04\x3c\xba\xc4\x7d\xa3\xe4\xc8\x84\xea\xe6\x35\xc8\xef\x82\xf0\x56\x25\x1b\x30\xec\xbd\x3b\x79\xe4\xf6\xd3\xca\xe7\xbd\x5f\x27\x62\xf0\x6d\x73\x58\x72\xe7\x88\xbd\x27\xe0\xf5\x73\x70\x29\x77\x31\x30\xab\x1c\x64\x8d\x80\x90\x0c\x9d\xa3\x53\x76\x23\x04\xfb\x04\x3d\x65\x3e\xf6\x89\x64\xf3\x00\x01\x5a\xf0\xa4\x55\xd5\x08\x9e\xbe\x90\x2b\x75\xdc\x66\x90\xaf\x1b\x08\xc3\xa3\x7a\xa5\xbd\x9e\xc7\x62\x18\x21\x3b\x55\x3e\x2f\xa5\x42\x6b\xbb\xf6\x20\x16\x3f\x7a\xe7\x03\x25\x5a\xda\x9a\x9a\xf3\x19\x86\xf8\x10\xd2\xae\xda\x24\x31\xad\x9c\x20\x11\xf5\xbd\x54\x8f\x12\x6d\x01\xfa\x12\xfb\xce\xac\x3f\x38\xc0\xd0\x27\x8e\x66\x41\x89\xbb\xe1\x4b\x60\xc6\x9e\xb9\x7f\xb3\x1b\x0c\xff\x61\x9d\x41\xef\x46\x83\xf1\x43\x4a\x35\xb0\x9b\x7f\x37\x9b\xb0\x37\x13\x76\x36\x61\xd3\xe9\xf4\xe5\x04\xc5\xb8\xa9\x46\xf8\x0a\x82\x01\x0b\xbe\x36\x65\x93\x02\xc8\x2a\xf8\x00\x68\x5a\x99\xc3\xca\x12\xc0\x71\xff\x54\xe0\x62\xb1\x4d\xc0\xb4\x54\xca\xa1\x21\xa8\x44\xb4\x51\x89\xaf\x14\xa0\x6e\x45\xa4\x72\x8b\xdb\xd5\x85\xca\x2d\x06\xf1\x81\xe7\x3c\x91\x90\xad\xcf\x9b\x08\x6c\xfa\x72\xc0\xd7\x2d\x7e\xe3\x5b\x68\x7f\x22\x1d\x65\xa9\xe9\xa6\x5b\x57\xff\x62\x97\x51\x8c\xe1\x31\x4f\x8a\xc2\x9c\xce\x7a\x2e\x6f\xd8\xeb\x7f\x61\xb3\x2c\x4b\x05\x9b\xb1\xbf\xb1\x37\x5c\x72\xc9\xd9\x1b\xf6\x37\x76\xc6\x65\xc1\x53\x55\x66\x82\x9d\xb1\xbf\x99\x6e\x33\xe5\x5d\x2a\x73\x1c\xee\x26\x8c\x33\x59\xa6\x78\xea\x7f\x67\xf1\x7d\x2f\x5d\xbb\xb8\x1f\x9d\xa5\x28\x1e\x85\x90\x4c\xab\x2d\x1d\x85\x7f\x75\x61\x26\x9d\xc8\x75\x2a\x0a\x9a\x0f\x55\x24\x26\x7e\xe0\x04\x5a\xfa\x7a\x2e\x9d\xd3\xf2\xaf\xa6\xc6\x7f\x65\x7f\x63\x97\x65\x9a\x9a\x2a\x99\x8d\xc6\x4c\xa4\xd7\xcc\x66\xc6\x08\x39\x7d\x4c\xee\x93\x4c\xc4\x09\x87\xdc\x18\xf3\xaf\xd3\x5b\x18\xed\x45\xe9\x69\x10\xc3\x35\xed\x34\x84\x8e\xd9\x7a\x9e\x25\xcf\xde\x29\x5c\xd9\xc1\xef\xb9\xe2\x56\x5f\x1d\x6f\x11\x79\xf2\x57\x5a\x0f\x64\xb0\xa2\xfe\x53\xa8\xa4\x75\xd0\x16\x50\x3b\x6c\x6d\x59\x2d\x47\x41\x78\xa8\x1f\xbb\xc9\x82\x2c\xdc\x93\xdf\x21\x07\xc8\x51\x0d\xdd\x72\x1b\x62\x3a\x15\xe6\x02\xb0\x25\x3d\x7b\xd3\xa0\x40\x9f\x13\xd1\xf9\xa5\xaa\x8b\x57\xe9\x62\x95\x0c\x12\x10\xac\x55\xf6\x8e\x9c\x34\x98\x73\x68\x96\x69\x92\x9e\x9a\xa5\x7a\x7a\xa9\xa4\xb9\xb6\xea\x64\x8d\x8c\x53\x70\xbb\x47\xe5\x3d\x6b\x14\xdc\x56\x4d\xd6\x60\x09\x80\x7d\x60\xaa\x84\x68\xc5\xc2\xec\x02\x66\x08\xd2\xdd\x5c\x9a\x37\xe8\x44\x82\xcc\x85\xc4\x11\x13\xe3\xd7\x88\xf7\xd7\x7e\x8b\x36\xe4\xa0\xf0\x96\x09\xd6\x97\x16\x7d\xc4\x84\xa3\x2c\xbc\x23\xdc\xff\x97\x01\x29\x1f\x95\x66\x19\x5b\x30\x94\xbd\x14\xa9\x92\x6b\x33\x2b\xba\x36\x01\xb5\xe5\xc9\x31\x58\xa1\xb0\x0a\x58\x58\x67\x0d\xcc\x61\x49\x8f\xd0\x90\x98\x73\x32\x89\xfd\xfd\x5e\x97\x4b\x63\x47\x38\xd7\xb3\x3b\x0d\xa9\x71\x5d\x39\xe2\xc7\x41\x06\xee\xb4\xc8\x81\x39\x1b\x41\x2b\x2e\xac\x81\x07\xa7\xe7\x4f\xc1\x16\x0d\x5b\x54\xbd\x58\xe7\x76\x57\x08\x85\x4d\x1a\xd9\xf2\x03\xe6\xe3\xd7\x84\x3d\x0f\x01\x43\xbf\x9b\x5d\xbc\xaf\x3d\xd7\x04\x43\xb7\x20\xa6\x6f\x2f\x3e\x9c\xbf\x5d\x5c\xdd\xdd\x36\x9e\x33\xa5\xd1\x9f\xf6\xe0\xa1\x3b\x7b\xef\x29\x10\xa1\x9f\x51\xc1\x67\xa1\x56\x36\x39\x76\xf8\x99\xde\xd0\x50\x1a\x06\x3c\x0a\x79\xd3\x43\xad\xa1\xe6\xc4\xe9\x4c\xf1\x97\x0b\x0a\xbd\x0c\xab\x6c\xbd\xc3\xae\xe4\x3b\x7c\xfd\xa3\x4a\x93\xa8\x1f\xc7\x68\x8f\xab\x8d\x7a\x6c\x01\x86\x2d\x05\x00\x7b\xc9\xff\x43\x95\x42\x0b\xbd\x10\x51\xe1\x43\x8b\xcd\xc6\xfd\xaf\xc6\x4e\xed\xbf\x83\xa3\x47\xd9\x75\x1b\x68\xaa\xba\x60\x25\x9c\xad\xc0\x99\x0a\x52\x01\xe8\x6b\x85\xf4\x48\x08\xe2\x47\x9c\x5c\xd0\x95\x9e\x87\x0d\xfa\x71\xa3\x52\x73\x17\x93\x31\xf1\xcf\xce\x65\x26\xf2\x48\x01\xe6\x08\xa9\x0d\x14\x8b\x36\x49\x1a\x7b\x3d\x9e\xef\x00\xa4\x0d\x50\xca\x97\xa4\xb4\x28\x5c\x30\xdd\x16\xdf\x73\xea\xda\x69\x67\x25\xe8\x8f\xf3\x40\x3d\x1d\xea\xb2\x6f\xda\xff\x4a\xe8\x40\xec\x0a\x62\x8c\xaa\x85\x45\x4d\xa7\x57\xea\x33\xca\xc3\x0b\x22\xf1\x2b\xab\xe7\x6a\x2f\x4e\x45\x6d\x5c\x69\x9a\xd5\xbb\x12\x78\x84\x11\x18\x87\x98\x20\x2d\xa0\x3a\x5b\xc1\xd1\x16\xf3\xac\x9e\x34\xa8\x73\xe9\x03\xc1\x2f\x74\x68\x97\xb5\x8e\x33\xd2\xe4\x5a\x5c\xe7\x84\xbd\xa8\x34\xf4\x05\xf0\xcc\x4a\x05\xdf\xa3\x60\x5d\xa5\x6b\x60\xba\x4e\x58\x52\xcc\x65\xa2\x71\x66\xe6\x22\x15\x0f\xa6\x76\xa1\xb3\x98\xe0\x4b\xf6\xee\x6c\x9b\x0d\xa9\x01\xdc\x66\x94\xd3\xb4\xa1\x45\x98\x87\x7c\xa5\x18\xa5\x8a\x85\x36\x76\x23\x28\xad\x88\xdf\xcc\x02\x48\x20\x16\x82\x38\x97\x58\x48\x5b\x3f\x80\xbf\xa0\x3c\xf1\x5c\x5e\xac\x20\xad\x17\x92\x89\xe3\x18\x6f\xa1\x56\x7b\xc3\x91\xc7\x25\xe4\x1c\x56\x74\x27\xb7\x03\x41\x42\xa1\xb8\x92\xc4\x83\xc8\x77\x05\x38\x75\xa1\x5f\xa5\xe0\xc5\x86\x25\xc5\x04\x58\xff\xec\x4e\x39\x97\x3c\x26\xf1\x79\x2a\xce\x74\x0d\xcc\xfb\x9e\x71\xa6\xdf\x97\xea\xa1\xcf\xb0\x3d\x16\xc8\x87\xab\x3a\x4b\xb9\x5c\xe0\x09\xf2\x15\xa0\x7c\x81\x86\x6b\x57\x4c\xb7\x5c\x2e\x1c\x53\xd1\x93\xd4\x33\x90\xf9\x0e\x95\x95\x8d\x1d\x6b\x3f\x34\xc1\xc9\xe0\x99\xca\xed\xf5\xc4\xf9\x69\x08\x46\x91\x33\x1b\x6a\x1e\xbe\x0b\x78\x94\x1f\xaf\x41\x2e\xec\x6c\xdd\x07\xf3\xb3\x33\xe0\x5b\x05\x62\x0d\x19\xf9\xda\x19\x52\x1f\xf6\xf1\x18\xa0\x86\x85\x78\x10\x0e\x68\x4f\xb5\x9e\x17\x0b\xd4\xe9\x47\x69\x62\x82\x6c\x6b\x83\x70\x1f\xe6\xc3\x08\xf4\xc3\x39\x37\x4f\xbb\x46\x6f\x78\x0f\x53\x2d\x70\xd9\xa7\xf4\x51\xc3\x3e\x35\xd4\x53\xe2\xd3\xe9\xa1\x5e\x53\x76\x21\x99\x35\xf7\x26\xec\x05\x4e\x2c\xfd\x82\x5c\x90\x24\xf4\x4c\xb1\xf3\x98\x56\x0f\x25\x20\xd7\x31\x27\x98\x06\xe2\x97\x1b\x46\x82\x7a\xd9\x2a\x9f\xb5\x5f\xde\x24\x90\x86\x72\x08\xd3\x00\x46\x11\x97\x58\x00\x1d\x92\x78\xed\xde\xa1\xd1\xae\xbc\x37\xdb\x37\xd8\xc6\xbb\xd8\x1b\xfb\xa2\xe9\xa2\xac\xa4\xf3\xd4\xfe\xce\x54\x3e\x97\xb6\x34\x72\x49\x6a\x94\xc7\xaa\x17\x15\xa0\xe2\xc9\xe6\x0f\x66\x2a\x80\x18\xac\x22\x1a\x08\xed\x79\x4a\xdd\xfa\x2e\x00\xa0\x88\xa5\xf0\x12\xf5\x53\x36\xf3\x5f\x33\x86\x87\x99\xe0\x5b\x3c\xe6\xeb\xb4\x9b\x69\x6a\x3a\x25\x29\x2c\xcb\x67\x90\xb1\xa2\x4b\xe0\xaa\x5d\x95\x66\x33\x0a\x08\x7d\xe7\xd2\x74\x1e\x5b\x25\x00\xe5\xa6\x7e\x99\xcb\x0f\x4a\x5b\x82\x04\xed\xfb\xc3\x82\x65\xa9\xdb\x5e\x38\x61\x38\xfa\xc3\x5b\x38\xb4\xc9\xe7\x8f\x54\x47\xee\x68\x81\x54\x25\x62\x39\xd9\xa9\x32\xf7\x8d\x8a\xb8\x9c\xcb\xff\x30\xdd\x83\xe2\xe4\x4e\xd9\x5f\xad\x70\x09\xc3\x08\x42\xb0\xe4\x13\x16\xfa\xdd\x3f\xbe\xfc\xf4\x12\x53\x0b\x4a\x0d\x5a\x9c\x93\xea\x01\xe2\xb8\xdd\xcb\x34\x85\x48\xb4\x6d\x81\xe3\x17\xf1\x9f\xe8\x85\xe5\xd0\xa5\x6e\x21\xab\x26\xc6\x90\x85\xde\x37\x83\xbd\xf3\x79\xc6\x22\x5e\x44\x9b\x13\x6b\xcb\xd1\x36\x66\x4f\x3f\x1a\x3e\x14\xe5\x33\x96\x56\x3b\xbd\xb9\xb9\x70\xe6\x5b\x47\xb8\x58\x99\x2f\xa6\x09\x00\xac\xb9\xad\x6b\xfd\x38\x3e\x58\x9c\x9c\x5e\x5b\xdf\xdb\x79\xee\x71\xab\xb4\xe7\x6f\x9c\xe4\x25\x97\x7c\x2b\x62\xf6\x02\x92\xe0\x5e\xd8\xc1\x9f\xcb\x6c\x39\x4d\x77\xab\x82\x58\xbb\x4c\xa7\x4c\x41\x93\x6a\xcf\x29\xb7\x88\x9b\xd7\xa4\x3d\x9d\xdd\x79\xd1\x6a\xb7\x75\x5c\xdf\xb8\x2f\x0d\x37\x58\xd0\xc7\xe5\x7a\xe7\xa6\x0a\x11\xaa\x92\xe3\x73\x7d\x3f\x61\xcb\x9c\x4b\x90\x13\x89\x43\xa3\xca\xaf\x4e\xb8\x3c\x23\x25\x96\xcd\x8a\x91\x3c\xdd\x41\x3a\xc0\x64\x2e\x91\x3f\x0c\x88\xa6\x77\x51\x9a\x44\x6c\x9d\xf3\x6c\x53\xb3\x83\xc4\x83\x90\x05\xa8\xd2\x5e\x0b\xae\x8f\x8b\xd6\xe7\xf5\x12\xd8\xe0\x78\xca\x4c\xc2\xed\x83\xcb\x1a\xe3\x2b\x54\xaf\xe3\x68\x01\xb8\x96\x88\x17\xe3\xd8\x5e\xf6\x72\x92\x56\x98\xee\x88\x76\x09\x22\x90\xa6\x71\xcc\x7e\x75\x5f\xf8\x1b\xfb\x95\x88\x48\x2c\xc0\xef\xd8\x90\xbd\x23\x36\x39\x8a\x9e\xf2\xa2\x6a\x45\x72\xcf\xd8\xe2\x3d\xd7\x98\xd3\x45\x9e\x0a\x8b\x8a\x76\x1b\xc7\x84\x54\x03\x81\xba\x8e\xfd\xa5\x5c\xaa\xd4\x72\xff\x5d\xbc\x65\x2a\x07\xd9\x8d\x42\xd1\x9f\x92\xb8\xcb\x3a\x48\x64\x2c\x7e\x3b\x8a\x80\xa3\xff\xa0\xb7\x66\xb3\xf9\x4c\xa0\xee\x50\x6f\x2c\xec\x4e\xb9\x30\x87\x70\x61\x6f\xc6\x8d\xa7\x74\x1d\xb9\x38\x4b\x8b\x0d\xc0\x09\x11\xb1\xef\x3b\x75\xcb\x77\x2c\xda\x70\xb9\x0e\x5c\x13\x80\xee\x12\x99\xca\x51\x9e\xf2\x01\x98\xee\x54\x6e\x13\x9c\x29\x6d\x97\xd2\x06\x5c\x20\x01\xd1\xba\xca\xe6\xe6\xf2\xf5\x3a\x17\x6b\xe0\x9c\x98\xcb\x0a\xf1\x00\xb0\xfc\x59\x65\x0c\xfc\x4e\x5f\xde\xf6\xd3\x90\x9f\x74\xdd\x06\x8b\x7c\xe7\xb2\x5e\x49\xdb\xd5\xaf\xe7\x7a\xb7\x4e\x58\x22\xa6\x13\xf6\x83\x47\x28\x8b\x48\x49\x97\x36\xdb\x91\x33\x59\x73\xf9\xb3\x3d\x57\x87\x26\x4b\x4a\x7b\xdd\xe1\xb7\x86\x42\x6c\xeb\xa4\xe9\xcd\x3b\x2e\x78\x51\x8e\x38\x83\x48\x05\xfc\xcc\xbc\x7c\x83\xef\xf6\x82\xf8\x11\x3e\x6c\x19\xaa\xcc\xf3\xe6\xe4\x34\xdf\xf6\x0c\xd6\x6d\x7d\xbd\xd7\x81\x9c\xaa\x6e\x07\xf2\x53\x98\xea\x96\x86\x64\xbf\x0f\x39\xed\xa0\xd6\xe8\x69\xd3\x58\x17\xb1\x05\xf9\x52\x9e\x82\xae\x5f\x63\x5b\x76\x80\x2c\x57\x71\x19\x89\xd8\xac\x5c\xb8\x0f\x21\x22\xc6\x31\x7c\x54\x36\xc9\xb6\x83\xb6\x42\x53\x04\xa7\xee\x97\xf2\x39\x0c\x62\x86\x76\xdd\x7f\xd7\xe1\x6f\xb0\x16\x5f\x5b\xa7\x87\xeb\x13\xfb\x29\x1f\x79\x4e\xb9\xcf\x57\xf9\x9c\x55\x9e\xac\x13\xc9\x0b\x95\xb3\xef\x5c\x1e\xef\x4b\x27\x02\xd5\x6d\x21\x8c\xdc\x26\x2a\x5d\x84\xdb\xc4\x17\x35\x3c\xda\x26\xa9\x79\x4a\x17\x7c\x9b\x85\x0c\xa9\x4e\x62\x9b\x7a\x26\xc5\x4e\x70\xb6\x09\xf8\x4e\x13\xed\x93\xf8\xe6\x92\x22\x0e\x38\x6e\x2a\x0f\x29\xbe\x3b\xcf\xe6\xac\x2c\x16\x07\xb2\xfe\xe0\xcb\xe3\x1c\x4f\x04\x43\xf8\xc0\xb3\x7e\x1e\x15\x4e\x2e\x07\xcc\x62\x72\xba\xe3\xd6\x52\xa9\xce\xcf\x7e\x11\x8d\x91\xbc\xae\xf5\xd0\xf9\xf5\x7b\x1b\x28\xf2\xf7\xc1\xca\x05\x0b\x06\x02\x09\x25\x31\x2b\x04\xaf\xf6\x6e\x5b\x33\xa7\xb8\x25\x5f\x39\x4b\x55\x19\x33\xda\xd4\x28\x0c\x9f\x4f\xf1\x74\x04\x86\xd7\xe9\xb4\x2b\xcb\x65\xa4\xb8\xaf\xdb\x7f\xe0\xbd\xf6\x15\x08\xbf\x75\xec\xc0\xbd\x4b\x9f\x7a\xf6\xd9\x86\x9e\x7a\x1a\xc6\xde\x6d\xc7\xa3\xc6\xde\x79\xc1\x81\x6e\x6e\x9c\x83\x14\xee\xa3\x49\x9c\xc2\x7a\x0b\x03\x08\x2d\x84\xb8\x95\xc0\xac\xbe\x3f\xfa\x73\x36\x29\xbd\xff\x53\x19\xcf\x85\x2c\x16\xf0\xc5\x71\x1f\x83\x8f\x7c\x84\xd7\x2b\x06\xd3\x20\x47\xf0\xbf\xdd\x2a\xf4\xef\x5b\x6e\x99\x7f\x67\x37\xe4\xd3\x32\xfb\x55\x02\x20\x52\x7d\xcf\xbe\x4b\x00\x73\x14\xc4\x42\xdd\xc0\x75\x0c\x17\x35\xe8\x80\xde\x0b\x1a\x54\xd9\xda\x07\x35\xc8\xd7\x1e\x42\xd5\x50\x0a\xb9\xf7\x28\x45\xd8\x6c\xb5\xf6\x6f\x01\xdf\xfc\x65\xe5\xdf\xc0\x0d\x6a\xc6\x2f\x65\xff\x29\x72\xe5\xd3\x41\xd0\x59\x15\x16\xdc\x6b\xaf\x1f\x2e\x95\x8b\xf6\x38\x8a\xb4\x86\x2a\x85\xf0\x17\xa2\xef\x41\x8f\xc2\x72\x67\xaf\x23\x1d\x21\xa4\x4c\x44\x8b\x0e\x49\x8a\x41\x55\x09\x2e\x9e\xa1\xc4\x44\x52\x3b\xcc\xec\x02\x3d\x05\x7f\x05\x11\x64\x6e\x79\x46\xf8\x3e\x82\x12\xd7\x83\x37\x53\x68\xc4\xbf\xfd\xf5\xdf\xa7\x5d\x22\xe8\x50\xf5\xb1\x70\x29\x57\xf9\x77\x79\x22\x64\x0c\xc1\x58\x1e\x37\xd5\x92\x64\xc5\x3b\x5f\xd9\x9e\xcd\x34\x7c\x92\xf4\xd0\xf6\xa3\x56\x2f\x70\x12\x7d\x81\x88\xbe\xdf\x64\xdd\xf2\xad\xc4\xfb\xba\x4c\x09\xbd\x88\x77\x92\x6f\x9b\xb2\xf1\xcf\x5a\xc7\x5d\x22\xd2\x18\xaa\x48\x5f\xdf\x17\x95\x8a\x45\x74\x3f\xd6\x26\x38\x98\xeb\x5d\x44\xf7\xec\xa7\xdb\x0f\xef\x51\xda\x33\xd1\x73\x79\xc9\x8b\xe4\x41\xdc\xe5\xa9\x0b\x07\x10\x79\x4f\x9e\xda\x35\x52\xe5\x1e\x0e\x78\x6e\x2c\x51\xb1\x35\x1c\x42\x6a\xf8\xed\xee\x64\x59\x46\xf7\xa2\x38\xcd\xb9\x8c\xd5\x16\x9b\x71\xaa\xcb\xd5\x2a\xf9\x6d\x5a\xf0\xbc\x83\x27\x1e\xfd\x08\x5f\xd1\xce\xf5\xea\x3f\x85\xb7\x79\xd1\xd4\x7d\x84\xac\x4f\xd2\x94\xae\x18\xb7\x70\x05\xce\xf9\x56\x00\xd1\x1f\xab\x6a\x2c\x40\x29\x98\x48\x09\x52\x84\x5a\x13\x82\x5e\x91\xd0\xf1\xa7\xc0\xb8\xff\x14\xd4\xaa\x2a\xb6\x6d\x2b\xe5\xe5\xfd\xb6\xfc\x1e\xef\x87\xeb\x5c\x68\x3d\x61\x5a\x41\x8d\xe7\xd2\x62\xd1\x6d\xbe\x14\xe0\x5e\x80\x2a\x34\xdd\xb1\x48\x65\x09\xa8\x21\xba\x76\x6d\xd4\x23\xf8\xe9\xc3\xb4\x41\x10\xb0\x2d\x65\x91\xa4\x8c\xaf\x0a\x72\xe2\x03\x2f\xba\xd5\x41\xd2\xd3\xb9\x84\x50\x6c\x04\xcd\x07\x88\x84\x0b\xbf\xb8\x46\x68\xb6\xe2\x51\x92\x26\x05\xb1\x35\x41\x92\x11\x37\xed\x35\xe7\x81\xe9\xcb\x9c\xef\x78\xea\x2f\x56\x3c\x2d\x7d\xa6\xe4\x89\x16\x3d\x6c\x80\x89\x5e\xa0\x83\xe0\x4b\xd0\x8f\x25\x61\xf0\x01\x99\x93\x67\xe6\xe3\x97\xb5\x53\xf4\x0f\xe1\xff\x56\xee\xe1\x7d\x56\xc1\x11\x17\xf2\x63\x0e\xc7\xe6\x95\xdb\x89\x07\x7b\x3b\x23\x89\x2d\x3e\xb8\x62\x8a\x7b\x12\x69\x77\x3c\x42\xcc\xa4\xe3\xd2\x3f\xb5\x92\x4f\xcd\x2f\x8c\xe8\xbd\x76\x23\xf1\x0b\xb9\x33\xba\xe8\xac\x87\x54\xdf\x7a\xe3\x3f\x2a\x95\x1e\xeb\x91\xa7\x0c\xfd\x44\xc9\x05\xa8\xa0\x1e\x73\x9d\xc4\x09\xe0\x1c\x5b\x17\x6f\x5d\xcc\xdd\xf1\x43\x57\xb5\x93\x08\x0e\x46\x55\x80\x8d\x0c\x2a\xd1\x83\x14\xd7\x59\x0b\xe8\x62\x24\xe2\x1d\xca\x40\xb4\x96\x35\xed\x9b\x21\x82\x80\xac\x81\xfb\x3a\x02\x87\x66\xad\x86\xa3\x9c\x75\xa8\x59\x5a\xfb\x94\x73\xdc\x85\x5c\xbb\xae\x1f\x83\x6f\xdb\xfe\x44\xa5\x74\x73\x7e\x92\x15\x3f\x97\x81\xc5\x8e\x04\x59\x36\xa5\xc0\xf5\x5a\x9b\x3f\xaf\x32\x0d\x8f\xf6\xe7\x1d\x43\xa8\xde\xbb\x73\xbe\x0d\xa5\xd1\x00\x0b\x12\xa9\xed\x32\x91\x36\x45\x9d\x9c\xdc\x70\xd5\x98\x59\xbe\x4a\x17\x90\xb0\x57\x06\x14\xcc\xa8\xf5\xbd\x33\x73\x42\xea\xcf\x70\xcb\xda\x77\x1d\x0f\xef\x77\x4f\xcb\xfd\xde\x11\x69\xac\xb7\xc0\x1c\x20\xe9\x23\xdf\x69\x90\x0f\x16\x66\x57\x5c\xa1\x63\xb7\x5a\xff\x49\x60\x7e\x58\x2e\x54\xd2\xe2\x2f\x49\x55\x9c\xda\x92\x20\x41\x89\x48\xad\x50\xb2\x27\x7e\x7a\xa1\xdb\x3b\xe7\xeb\xc4\x6a\xf2\xde\x58\x0d\x06\xa1\xff\x67\x84\x67\x7a\x9c\xc0\x47\xfa\xa2\x83\x63\x12\x2d\x46\x82\x09\x41\xe2\x96\x0b\x51\x4f\xd8\x96\x27\x92\x96\x01\x8a\xd1\xc5\x62\x59\xae\xd7\x9d\x2e\xd2\x6f\x3f\xd6\x52\x5d\x27\x7f\xf7\xbe\xf0\x5e\xea\xb2\xa7\xf0\x16\x5f\xd8\x2f\xa1\xfb\xda\xdc\xfb\xbe\x8c\x83\xf8\x2b\x7a\xe3\x5b\x43\x62\x8d\x49\xf4\x34\xde\xf8\x8b\x21\xde\x78\x8b\xed\x82\x14\x3b\xba\x4e\x5b\xfc\xcd\xef\x6e\xfa\x2f\xe3\xa6\x1f\x34\x29\x90\xd6\x65\x91\x54\x0d\xf4\x9e\x1a\x1e\x48\x83\xe7\x98\x61\xa1\x56\x24\xec\xaf\x85\x8c\x35\x5b\xf2\xe8\x19\x78\xf1\xe0\x74\x3c\xde\x1f\xb8\x07\xfc\x72\xa3\xb6\x82\xc1\xa7\x34\xca\x98\x30\xca\x62\x9c\x00\x5a\xd5\x34\xd0\x23\x46\x08\x8f\x02\xc7\x29\x22\x57\x62\x6f\x54\x7f\x27\xc5\x23\x33\xa7\xd5\x24\x84\xef\x05\xc3\x03\xfa\x56\x2f\x8d\x75\x58\xc1\xfa\x3b\xca\x86\x5c\xac\x79\x1e\x43\x86\x09\x2d\xc9\x94\x47\xf7\xe6\xbf\xa1\x7e\xf4\x45\x82\x18\x5a\xaa\x7d\x84\xbd\xfa\xd2\x12\x19\x21\x33\x1a\xa1\x19\x7d\xfd\xf0\x75\xcd\x78\x94\x2b\x8d\x4e\x23\x27\x0b\x0b\x19\xce\x60\xc0\x3e\x24\x71\xc9\x53\xfc\x62\xa7\xa7\x7d\x2c\x7c\xad\x0e\x38\x0a\x14\x9c\x9a\x68\x36\x1a\x0e\xe4\x28\x82\x6e\x9c\xce\xe5\x5b\x17\x30\x79\xcd\xee\xb4\x20\x94\x99\xb6\x1c\xd8\xbd\x35\x7d\x36\xf3\xa1\x81\x09\xec\xb4\x21\x7a\x3a\xc0\x82\xac\x83\x8e\xd0\xdd\x3d\xb1\x87\x5d\xf1\x98\x41\x19\xcd\x12\x7b\x11\xc8\x48\xfb\x6e\xc1\x7b\x42\x2e\x78\xbc\x0b\xa9\xd9\x12\xc9\x20\x4a\xc7\x78\xbc\x4d\xa4\x59\x04\x56\xaa\xd0\x9d\x34\x96\xb5\x1c\x21\xc7\xa0\xe8\x93\xa6\xb5\x4d\x50\x33\x29\x8c\x71\xc9\xf3\x24\xdd\xc1\x7d\x22\xcb\xc5\x49\xf0\x9d\x60\x7c\x28\xe3\x09\xf8\xd7\x89\x46\xa4\xd4\x62\x55\xa6\x78\xeb\x80\x7b\xb9\x6b\x00\xed\x48\x77\x17\x13\x63\x70\x14\xa4\xa3\x11\x7c\x18\xd5\xe9\x9e\x22\x7b\xa4\x11\xad\x1c\x17\x71\xf3\xd4\x81\x39\x80\xdc\x37\xea\xd1\xa6\xba\x3d\x72\x8f\x65\xee\x3a\x5d\x9f\x2c\xca\xd2\x6f\x87\xda\x1b\xa0\xdd\xa7\xb0\xf3\xe3\x4a\x68\x8d\x7e\x13\xb1\xdb\x9b\x12\x09\xcd\x21\x81\x57\xef\xb9\x2e\x35\x66\xcc\x99\xb1\x84\xf3\xcb\x3a\x3a\xaa\x8e\x6b\xe6\x5a\x97\x68\x25\xd9\xbc\xfc\xfe\xfb\x3f\x0a\xf6\x3d\xa4\x10\xd2\x7d\x04\xe3\x63\x40\x1e\x88\xa5\xc3\x96\xed\x3e\x20\x90\x59\xb0\x31\x22\xac\x0d\xa2\x6a\xf3\xf5\x01\xe4\xc9\xa3\x0d\xd3\xe5\x12\x11\x8c\x9c\x42\x2c\x5c\x3a\x12\xe2\xf7\x0a\xc0\x88\x78\xb2\xdb\xda\xff\x2f\x09\x28\xa0\xe4\xc1\x5c\x66\x0a\x79\xb2\x01\xfa\xb9\x14\x6c\xcb\xf3\x7b\x50\xb0\x44\xf7\x3c\xf0\x82\x7f\x97\x88\x69\x35\xbc\xf0\xb2\x52\x1f\x0a\xe8\x20\xff\x2d\xcb\x4b\x29\xad\x24\x0f\x33\x86\xa9\xf7\xf5\x4f\xe6\x72\x59\x86\x77\xcf\x4a\xb0\xc0\x4f\x2d\x08\x18\xc0\x66\xab\x80\x2b\x84\x2a\xc5\xb5\xaf\xd7\x94\x0d\x88\x1a\xcc\xe5\x13\x87\x0d\xf6\x39\xfc\x3e\x92\x0d\x66\x9d\x79\x41\xbe\x02\x34\x37\x54\x8d\x85\xe1\xc0\x69\x0f\x46\xce\x47\x90\x8e\x9d\xb0\x9f\x92\x07\x31\x61\x37\x19\xcf\xef\x27\xec\x2d\x86\xff\xfe\xac\x96\x6d\x3e\xbc\x06\xa1\xc4\xd1\x7e\xbc\xc3\xdc\x58\x7d\x44\x2b\xed\xd6\xff\xaf\x0d\x62\x00\xd6\x15\xfb\xfe\x9f\x89\xc8\xeb\xe0\xfa\xf8\x7b\xf7\x44\xec\x09\x53\xff\x0e\x5e\xfb\xbb\xbc\x15\xf7\xd3\x7c\xfc\x21\xfc\x5f\xbb\x7f\x59\x8b\x0b\x6c\x4f\xda\xe5\x5a\x51\x69\xdf\x56\x62\x73\x12\xd7\x0f\xe5\x66\x7e\xf3\xb0\xa5\x40\xe9\xe3\xb1\x4b\x6d\x1f\x01\xba\xa7\x57\x6d\x7f\x9d\xa5\x4a\x97\x79\xff\xe2\xbf\xae\xd6\xda\x7e\xbd\x85\xea\x13\x26\xdb\x76\x29\x80\xb5\x60\x28\xfc\x04\x1f\x5b\xfc\x87\x5a\x2e\x00\x6b\x75\xdc\x0a\x6f\x2b\xce\xea\x41\xb9\x88\x18\x55\xd5\x9f\x90\x37\x99\x00\xc6\x29\x6f\x8a\xfa\x80\x40\x6d\x86\x39\xd7\xc8\x5c\x5a\x02\x6e\xcc\x98\xcd\x73\x01\x4c\xc1\xb9\x00\x99\x33\x96\xf1\xdc\x01\x1e\xac\x45\x14\xdc\x7c\x3c\x28\x26\xcc\x72\x83\x64\x55\xba\x6f\x2d\x85\x90\xae\xb7\xc7\x98\x12\xc6\x3a\xa8\xf7\x3e\xa1\xdd\x1e\x85\xe5\x61\xef\x90\x64\x6c\xbc\x17\xdc\x05\xc1\xe4\x5e\x8b\x22\xd8\xcd\x6b\xa6\x45\x65\x69\x56\x22\x54\xdf\x14\xe2\xbf\x35\x06\x5d\x23\xe7\xaa\x38\x50\x06\xc5\xf4\x9e\xc2\x5f\xfe\x91\x17\x1b\xbc\xd0\x6e\x55\x21\x70\xcf\x44\x96\x20\x9c\x2f\xe8\x75\x5e\xa6\x6a\x09\xfa\x62\xe6\x97\xae\xbb\x61\x44\x4b\x7b\x50\xd7\x35\x07\x6c\xc8\xce\x60\x76\x13\xc8\xb4\xcd\x85\x06\xc2\x95\x66\x94\x6a\x28\x3e\x79\xdc\xa5\xbb\x59\x5d\xb3\xe9\xbf\x6d\x5c\xb6\x9b\x0c\xfd\x66\x59\x03\x58\xf5\xfc\x80\x0c\x9a\x86\xde\x01\x91\x15\x53\x18\x18\xf9\x4a\x6b\xed\xb5\x32\xd6\x73\x39\xc3\x5f\x82\x43\x80\x7b\xc9\x1d\x87\x07\x25\xc5\x52\xb7\xfe\x30\x7d\x95\xcd\x42\x04\x22\x79\x08\x26\xde\x97\x09\x97\x81\x09\x64\x35\xca\x22\xc9\x05\x93\x80\x42\x98\x4b\x5d\x2e\x4f\x3c\x31\x89\xb9\xc5\x3d\x00\x99\x8e\x16\x19\x87\xab\x0c\xf0\x15\x9d\xb4\x1c\xc3\xe8\x99\xf4\xd2\x19\x96\xc0\x8f\xa7\xb4\xf9\x43\xae\x24\x66\xc6\xbb\xb6\xbb\x72\xcc\x65\x0d\x6e\xd1\x16\xae\x84\x87\x5d\xdf\x7e\x01\xe2\x3e\x90\x81\x79\x8d\x28\x8a\xaf\x7d\x80\x87\xd1\xd0\xa1\x47\x37\xc4\xd3\xe6\xf2\xff\xb7\x67\x43\x37\xa8\x78\xc4\x4c\x37\x3d\x63\x8e\xa8\x4e\xb0\x73\xa5\x6e\xf6\x0a\x19\x18\x81\xdd\x95\x6a\x4c\xf9\xb6\x52\xb9\xc5\xb5\x84\x0a\x0f\x8a\xd2\x65\xe1\xd7\x87\x44\x07\x74\xdf\xf0\xb5\x1b\x21\xd8\xeb\x5c\xac\x5e\x7f\xca\xc5\x6a\x61\x47\x7a\x0a\x0d\x9a\x9a\x16\x35\x49\xbf\x07\x4e\x0e\x9d\x29\xd9\x4e\x7e\xb8\x87\x9c\xb4\xd6\x24\x2c\x27\x68\x53\xb2\x62\x5e\xdb\xd1\xb4\x07\x18\x20\x44\x5c\x67\x23\x6f\xd4\xec\x8b\x1f\x73\x5d\x48\xb0\x01\x50\xab\x0e\x4d\xc4\xbf\xff\xe3\xad\xd2\x67\x43\x8e\xb7\xdb\x2a\x64\xc6\x6e\xf6\x5c\xba\x03\xaf\x1b\x17\xfa\x65\xd1\xe9\x30\x80\x3a\xe3\x8f\x92\x78\x6c\x46\xb9\x9e\x86\x1d\x6b\x35\x00\x51\x70\xac\x35\x30\x70\x7e\x95\x49\xeb\xe9\x4b\x9c\xac\xde\x24\xd0\xde\xe6\x69\x1a\x6a\x2a\xf8\x48\xdb\x5c\xfa\xbc\x54\x63\xb5\xa6\xa9\x75\xe1\x55\xec\x0d\x62\x2d\x8a\x21\x21\x58\x4c\x2c\xe9\x0a\xd1\x15\x52\x3c\xec\x64\xc9\x41\xd8\xd5\x49\x2a\xed\x5b\xcd\x4f\x75\x89\xfc\xc6\xf2\xa2\xf7\x44\x9e\xf1\xb3\x8b\x7b\xd1\x80\x33\xef\xad\x6b\x7b\xa4\x23\xa0\x94\x80\xc5\x6c\x77\xd9\x88\xe7\xb9\x45\xf9\xd3\x57\x99\xb9\x2b\xad\x78\x54\x71\x73\x76\xd4\x73\x23\xa2\xfb\x4c\x25\x72\xf4\x5e\x54\xa1\xb8\x80\xc9\x5e\x30\x5f\x9a\xbb\x1d\x0e\x3a\x1c\x2b\xf6\x24\x36\x44\x03\xbc\xc2\x42\x43\x3d\x19\x1b\x67\x4e\x38\xb7\x7b\xda\x3d\xb5\xff\x42\xf8\xb3\xe1\x19\x7c\xb1\x2d\xf1\xa1\xda\xa9\xc2\x5b\x1c\x3b\x15\x26\x50\xde\xc8\xfe\x1a\xd8\xd9\x9c\x55\x28\x0c\x5b\xbb\x14\x5c\x90\xbf\x7b\x86\x7e\xf7\x0c\xfd\x0f\xf7\x0c\x7d\x49\xb7\x10\x60\x63\x9e\xd3\x27\xd4\x13\x20\x3f\x62\x39\xba\xaf\x1e\xab\x22\x8f\xd6\xf1\x24\xd0\x00\x0e\x32\x1d\x9b\x40\x7f\x4b\x84\x51\x80\x56\x7c\x74\x2f\x64\x67\x8c\xde\xd2\x17\x75\xca\x31\x3e\x2d\x82\xa5\x8d\x7d\x29\x78\xbb\x1f\xca\xe2\xa1\x4e\x44\x1a\xdc\x46\x08\x62\xd6\x09\xd8\x9e\xa6\xe1\x27\x00\x1a\x53\xb9\x23\xb6\xd6\x94\x85\x87\xc1\x48\xa4\x49\x42\xb0\x54\x8d\x0a\x7a\x28\x26\xce\x7e\x78\x91\x29\x95\xb6\x42\xe3\x9e\xb4\x03\x1b\x89\x32\x43\x3b\xef\x02\x8d\x51\x1d\x02\xc6\x6c\x2f\xfa\xa4\x0b\x9f\xa2\x81\xf9\x18\xa0\x85\x01\xb3\x29\x2e\x21\x97\xd2\x77\xc7\xd4\x67\x38\x71\xe7\x70\x21\x8c\xd8\x52\x44\x1c\x74\x20\x2d\x78\x2f\xe2\x2e\xfb\x24\x24\x45\x6a\xa4\x83\xe8\xe6\x77\x3a\xa2\x96\x50\xee\x22\x69\x13\xbe\x18\xbb\xb8\x6a\x16\x82\x85\x96\x63\xcd\x2d\x92\xc4\xd2\x2e\xee\xd3\x37\xb5\x1c\xd3\x8b\x28\xe5\x7a\xa0\x61\xdd\xba\xef\x5c\x50\x41\x67\x50\xce\xf0\x8d\xf4\x27\x48\xc7\xd9\x0e\x44\xee\xcc\xe5\xcc\xc9\x5e\x7a\xec\x97\x43\xee\x61\xb8\x14\x31\x8b\x8d\xa1\x41\x2e\x47\x7f\x73\x99\x30\x5d\x46\x1b\x60\xab\xac\xee\x53\xe1\xbe\xd5\x5c\xb1\x93\xb9\x34\x17\x22\x70\xb5\x6c\x39\xe4\xc5\x3f\x1a\x63\x55\x27\xff\x29\x1c\x3c\x8b\xc8\xbb\x42\x44\x16\x5e\x9c\x94\x6c\x45\xaf\x59\xe2\x50\x04\x58\x78\x4c\x49\x99\xc5\xbc\x10\xd3\xb9\x47\xdb\x24\xe8\xe9\xb4\x28\x0f\x32\x99\x75\xd8\xb0\x10\xc7\x58\xdb\x69\xd3\x64\x25\xa2\x5d\xd4\xd0\x01\xea\xa7\x89\xf8\xfd\xda\xf6\x6d\x5d\xdb\x90\x65\x17\x73\x06\xc7\x74\x2d\x55\xf5\xda\xbf\x7e\x5c\xe7\x0a\x16\xd4\x44\x8f\xe8\xe7\x2f\x78\xed\x6c\xb1\x81\xc7\xd9\xf3\x83\xef\x41\xfd\xc7\x99\xbf\xd8\xfa\xc3\x3a\xa0\x40\x68\x98\x85\x61\x70\xb1\x08\xa7\x8e\x31\x68\x07\x87\xf5\xbb\x59\x66\xbe\x29\x70\xd2\x90\x8b\xab\xb1\xb8\x1d\x5c\xe9\xd2\x5a\xda\x52\xe0\x79\xd7\x63\x71\x07\xac\xee\xbc\x78\xa1\x5d\xaf\x57\x77\x40\x8b\xfd\x7f\x9f\xe8\xe2\x97\x9a\x66\xe8\x61\xa2\xa3\xcf\x66\x9a\xda\xaa\x62\x35\x87\x5a\x54\xd7\x55\x9b\x47\xad\xec\x9c\x83\xcb\x93\x55\x89\x33\xf5\x1e\x73\x0f\xfa\xe4\xfa\xeb\x13\x1e\x4d\x8f\x39\xcf\x32\x91\xdb\x83\xbc\x61\x6b\x81\xe4\x1a\x7c\x05\x34\x13\x37\x02\x85\x9b\x6b\xb7\x5c\xb3\x95\xd4\x8a\x86\xc7\xa0\xeb\xa6\xed\x23\x77\x59\xa6\x69\xe7\xc8\xed\x57\x72\xba\xbc\x7b\xff\x7e\xf1\xcb\xec\xfd\xdd\xb9\x6d\x7e\xab\x32\x52\xf0\x58\x67\x9f\xb8\x9a\x50\x9f\x78\xed\x45\xf3\x59\x61\xc5\xa3\x95\x6f\x35\x3a\xb9\xca\x34\xad\xaa\x66\xcd\xe5\x27\x2a\x07\x60\xda\xa8\x08\x6a\xfa\x8d\xf5\x76\x5c\xf5\xfb\xf0\xd8\x27\x53\xf8\x27\x7c\xf7\x84\xf9\x46\xbc\x06\x6d\x47\xd2\x8c\x6b\xef\x57\xca\x86\x39\x62\x39\x20\x18\xb8\x6b\x39\x3c\xb5\x2e\xe0\x61\xcb\xe3\x4e\x02\x23\xb9\x88\xad\x9c\xdf\x93\xac\x0e\xec\xbb\x4f\xd5\x38\xb5\xdb\xcb\x63\xbc\xd2\x40\xb9\x13\x54\x73\x03\x8d\x6a\x2f\x78\x36\x97\xe8\x03\x35\x75\x2a\x54\x77\x9d\xd8\x05\x99\xb7\x29\x97\xeb\x92\xaf\x8d\x75\x6b\x3f\x3e\x97\xdb\x64\xbd\x41\x1e\x90\x32\xf3\xf8\x64\xce\x24\xd0\xc5\xd4\xa6\x50\x0d\x9f\x9c\xc8\xb9\xa4\x36\xc9\xb5\x2f\x1e\xb1\xb2\x7f\xbe\x71\xcd\x21\x50\x3a\x16\x44\x82\x74\x72\x2e\x71\x70\x91\x9f\xc4\x46\x42\xe0\xc6\xc2\x8b\xfa\xd4\xe5\x10\xbb\x44\xd1\x76\xb3\xa7\xaf\x21\x26\x33\x97\x2e\x45\x17\x3d\x47\xd4\x86\x40\xb8\x04\xab\xb4\x7f\x3f\xb1\x83\x61\xd7\x04\xd5\xad\x7d\xd6\x1f\x7d\x06\x98\x05\xb7\x18\xa1\x3e\xdd\xdc\xc6\x06\x7a\x0b\x79\xb0\x71\x74\xf1\x36\x40\x5e\x76\x7b\x6d\x6c\xbb\xf0\x99\x4e\x68\xab\x2a\x97\xe9\x88\x2a\xe1\xf3\xbd\x95\xc2\x2d\xb9\xbf\x52\x03\xae\xc3\xd7\xb5\xa5\x65\xa6\x69\xdf\x67\x97\x4a\x75\x8c\xcb\x13\x06\x14\x2b\x95\xa2\x17\xf6\x75\x46\x19\x15\x87\xcc\x97\x01\xc9\x8a\xf5\x2e\xb2\xbb\x4f\x5f\x85\xd2\x44\x1f\x54\x1d\x6f\x3f\x0d\xae\x91\xb3\x10\xe8\xb0\x1b\xb5\xc3\xd2\x39\x57\xd9\x60\x3b\xb6\x49\x8a\x27\x59\x19\xe7\x04\xb7\x17\xb3\x78\x50\x63\xd9\xcc\xff\x89\x9b\x44\x13\x3f\x72\x13\xa8\x64\x54\xe6\xda\x6c\x97\xb4\xdf\xd1\xae\xad\x72\xc6\xe7\xd2\xa6\xaa\xd9\xed\x78\x66\xc1\xb9\xb9\xfb\x2b\x26\x80\x66\xc8\xa7\x0f\x16\x6b\xc1\x94\x14\x76\x37\x9c\x4b\xab\xfd\x3d\x61\x7c\xa9\xad\xa4\x36\x97\x3b\xa7\x73\x9d\x38\xfa\x22\x2e\x19\xa0\x9e\xf7\xef\x79\x35\x33\xa0\x72\xce\xff\xc1\xfc\xdf\x7f\xff\xe1\xff\x05\x00\x00\xff\xff\xf9\x47\xac\x28\x8a\xb5\x04\x00") func adminSwaggerJsonBytes() ([]byte, error) { return bindataRead( @@ -93,7 +93,7 @@ func adminSwaggerJson() (*asset, error) { return nil, err } - info := bindataFileInfo{name: "admin.swagger.json", size: 306659, mode: os.FileMode(420), modTime: time.Unix(1562572800, 0)} + info := bindataFileInfo{name: "admin.swagger.json", size: 308618, mode: os.FileMode(420), modTime: time.Unix(1562572800, 0)} a := &asset{bytes: bytes, info: info} return a, nil } diff --git a/flyteidl/gen/pb-java/flyteidl/core/Tasks.java b/flyteidl/gen/pb-java/flyteidl/core/Tasks.java index 9a464e24c5b..d4416668c67 100644 --- a/flyteidl/gen/pb-java/flyteidl/core/Tasks.java +++ b/flyteidl/gen/pb-java/flyteidl/core/Tasks.java @@ -2302,6 +2302,1676 @@ public flyteidl.core.Tasks.Resources getDefaultInstanceForType() { } + public interface GPUAcceleratorOrBuilder extends + // @@protoc_insertion_point(interface_extends:flyteidl.core.GPUAccelerator) + com.google.protobuf.MessageOrBuilder { + + /** + *
+     * This can be any arbitrary string, and should be informed by the labels or taints
+     * associated with the nodes in question. Default cloud provider labels typically
+     * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+     * 
+ * + * string device = 1; + */ + java.lang.String getDevice(); + /** + *
+     * This can be any arbitrary string, and should be informed by the labels or taints
+     * associated with the nodes in question. Default cloud provider labels typically
+     * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+     * 
+ * + * string device = 1; + */ + com.google.protobuf.ByteString + getDeviceBytes(); + + /** + * bool unpartitioned = 2; + */ + boolean getUnpartitioned(); + + /** + *
+     * Like `device`, this can be any arbitrary string, and should be informed by
+     * the labels or taints associated with the nodes in question. Default cloud
+     * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+     * 
+ * + * string partition_size = 3; + */ + java.lang.String getPartitionSize(); + /** + *
+     * Like `device`, this can be any arbitrary string, and should be informed by
+     * the labels or taints associated with the nodes in question. Default cloud
+     * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+     * 
+ * + * string partition_size = 3; + */ + com.google.protobuf.ByteString + getPartitionSizeBytes(); + + public flyteidl.core.Tasks.GPUAccelerator.PartitionSizeValueCase getPartitionSizeValueCase(); + } + /** + *
+   * Metadata associated with the GPU accelerator to allocate to a task. Contains
+   * information about device type, and for multi-instance GPUs, the partition size to
+   * use.
+   * 
+ * + * Protobuf type {@code flyteidl.core.GPUAccelerator} + */ + public static final class GPUAccelerator extends + com.google.protobuf.GeneratedMessageV3 implements + // @@protoc_insertion_point(message_implements:flyteidl.core.GPUAccelerator) + GPUAcceleratorOrBuilder { + private static final long serialVersionUID = 0L; + // Use GPUAccelerator.newBuilder() to construct. + private GPUAccelerator(com.google.protobuf.GeneratedMessageV3.Builder builder) { + super(builder); + } + private GPUAccelerator() { + device_ = ""; + } + + @java.lang.Override + public final com.google.protobuf.UnknownFieldSet + getUnknownFields() { + return this.unknownFields; + } + private GPUAccelerator( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + this(); + if (extensionRegistry == null) { + throw new java.lang.NullPointerException(); + } + int mutable_bitField0_ = 0; + com.google.protobuf.UnknownFieldSet.Builder unknownFields = + com.google.protobuf.UnknownFieldSet.newBuilder(); + try { + boolean done = false; + while (!done) { + int tag = input.readTag(); + switch (tag) { + case 0: + done = true; + break; + case 10: { + java.lang.String s = input.readStringRequireUtf8(); + + device_ = s; + break; + } + case 16: { + partitionSizeValueCase_ = 2; + partitionSizeValue_ = input.readBool(); + break; + } + case 26: { + java.lang.String s = input.readStringRequireUtf8(); + partitionSizeValueCase_ = 3; + partitionSizeValue_ = s; + break; + } + default: { + if (!parseUnknownField( + input, unknownFields, extensionRegistry, tag)) { + done = true; + } + break; + } + } + } + } catch (com.google.protobuf.InvalidProtocolBufferException e) { + throw e.setUnfinishedMessage(this); + } catch (java.io.IOException e) { + throw new com.google.protobuf.InvalidProtocolBufferException( + e).setUnfinishedMessage(this); + } finally { + this.unknownFields = unknownFields.build(); + makeExtensionsImmutable(); + } + } + public static final com.google.protobuf.Descriptors.Descriptor + getDescriptor() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_GPUAccelerator_descriptor; + } + + @java.lang.Override + protected com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internalGetFieldAccessorTable() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_GPUAccelerator_fieldAccessorTable + .ensureFieldAccessorsInitialized( + flyteidl.core.Tasks.GPUAccelerator.class, flyteidl.core.Tasks.GPUAccelerator.Builder.class); + } + + private int partitionSizeValueCase_ = 0; + private java.lang.Object partitionSizeValue_; + public enum PartitionSizeValueCase + implements com.google.protobuf.Internal.EnumLite { + UNPARTITIONED(2), + PARTITION_SIZE(3), + PARTITIONSIZEVALUE_NOT_SET(0); + private final int value; + private PartitionSizeValueCase(int value) { + this.value = value; + } + /** + * @deprecated Use {@link #forNumber(int)} instead. + */ + @java.lang.Deprecated + public static PartitionSizeValueCase valueOf(int value) { + return forNumber(value); + } + + public static PartitionSizeValueCase forNumber(int value) { + switch (value) { + case 2: return UNPARTITIONED; + case 3: return PARTITION_SIZE; + case 0: return PARTITIONSIZEVALUE_NOT_SET; + default: return null; + } + } + public int getNumber() { + return this.value; + } + }; + + public PartitionSizeValueCase + getPartitionSizeValueCase() { + return PartitionSizeValueCase.forNumber( + partitionSizeValueCase_); + } + + public static final int DEVICE_FIELD_NUMBER = 1; + private volatile java.lang.Object device_; + /** + *
+     * This can be any arbitrary string, and should be informed by the labels or taints
+     * associated with the nodes in question. Default cloud provider labels typically
+     * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+     * 
+ * + * string device = 1; + */ + public java.lang.String getDevice() { + java.lang.Object ref = device_; + if (ref instanceof java.lang.String) { + return (java.lang.String) ref; + } else { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + device_ = s; + return s; + } + } + /** + *
+     * This can be any arbitrary string, and should be informed by the labels or taints
+     * associated with the nodes in question. Default cloud provider labels typically
+     * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+     * 
+ * + * string device = 1; + */ + public com.google.protobuf.ByteString + getDeviceBytes() { + java.lang.Object ref = device_; + if (ref instanceof java.lang.String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + device_ = b; + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + + public static final int UNPARTITIONED_FIELD_NUMBER = 2; + /** + * bool unpartitioned = 2; + */ + public boolean getUnpartitioned() { + if (partitionSizeValueCase_ == 2) { + return (java.lang.Boolean) partitionSizeValue_; + } + return false; + } + + public static final int PARTITION_SIZE_FIELD_NUMBER = 3; + /** + *
+     * Like `device`, this can be any arbitrary string, and should be informed by
+     * the labels or taints associated with the nodes in question. Default cloud
+     * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+     * 
+ * + * string partition_size = 3; + */ + public java.lang.String getPartitionSize() { + java.lang.Object ref = ""; + if (partitionSizeValueCase_ == 3) { + ref = partitionSizeValue_; + } + if (ref instanceof java.lang.String) { + return (java.lang.String) ref; + } else { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + if (partitionSizeValueCase_ == 3) { + partitionSizeValue_ = s; + } + return s; + } + } + /** + *
+     * Like `device`, this can be any arbitrary string, and should be informed by
+     * the labels or taints associated with the nodes in question. Default cloud
+     * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+     * 
+ * + * string partition_size = 3; + */ + public com.google.protobuf.ByteString + getPartitionSizeBytes() { + java.lang.Object ref = ""; + if (partitionSizeValueCase_ == 3) { + ref = partitionSizeValue_; + } + if (ref instanceof java.lang.String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + if (partitionSizeValueCase_ == 3) { + partitionSizeValue_ = b; + } + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + + private byte memoizedIsInitialized = -1; + @java.lang.Override + public final boolean isInitialized() { + byte isInitialized = memoizedIsInitialized; + if (isInitialized == 1) return true; + if (isInitialized == 0) return false; + + memoizedIsInitialized = 1; + return true; + } + + @java.lang.Override + public void writeTo(com.google.protobuf.CodedOutputStream output) + throws java.io.IOException { + if (!getDeviceBytes().isEmpty()) { + com.google.protobuf.GeneratedMessageV3.writeString(output, 1, device_); + } + if (partitionSizeValueCase_ == 2) { + output.writeBool( + 2, (boolean)((java.lang.Boolean) partitionSizeValue_)); + } + if (partitionSizeValueCase_ == 3) { + com.google.protobuf.GeneratedMessageV3.writeString(output, 3, partitionSizeValue_); + } + unknownFields.writeTo(output); + } + + @java.lang.Override + public int getSerializedSize() { + int size = memoizedSize; + if (size != -1) return size; + + size = 0; + if (!getDeviceBytes().isEmpty()) { + size += com.google.protobuf.GeneratedMessageV3.computeStringSize(1, device_); + } + if (partitionSizeValueCase_ == 2) { + size += com.google.protobuf.CodedOutputStream + .computeBoolSize( + 2, (boolean)((java.lang.Boolean) partitionSizeValue_)); + } + if (partitionSizeValueCase_ == 3) { + size += com.google.protobuf.GeneratedMessageV3.computeStringSize(3, partitionSizeValue_); + } + size += unknownFields.getSerializedSize(); + memoizedSize = size; + return size; + } + + @java.lang.Override + public boolean equals(final java.lang.Object obj) { + if (obj == this) { + return true; + } + if (!(obj instanceof flyteidl.core.Tasks.GPUAccelerator)) { + return super.equals(obj); + } + flyteidl.core.Tasks.GPUAccelerator other = (flyteidl.core.Tasks.GPUAccelerator) obj; + + if (!getDevice() + .equals(other.getDevice())) return false; + if (!getPartitionSizeValueCase().equals(other.getPartitionSizeValueCase())) return false; + switch (partitionSizeValueCase_) { + case 2: + if (getUnpartitioned() + != other.getUnpartitioned()) return false; + break; + case 3: + if (!getPartitionSize() + .equals(other.getPartitionSize())) return false; + break; + case 0: + default: + } + if (!unknownFields.equals(other.unknownFields)) return false; + return true; + } + + @java.lang.Override + public int hashCode() { + if (memoizedHashCode != 0) { + return memoizedHashCode; + } + int hash = 41; + hash = (19 * hash) + getDescriptor().hashCode(); + hash = (37 * hash) + DEVICE_FIELD_NUMBER; + hash = (53 * hash) + getDevice().hashCode(); + switch (partitionSizeValueCase_) { + case 2: + hash = (37 * hash) + UNPARTITIONED_FIELD_NUMBER; + hash = (53 * hash) + com.google.protobuf.Internal.hashBoolean( + getUnpartitioned()); + break; + case 3: + hash = (37 * hash) + PARTITION_SIZE_FIELD_NUMBER; + hash = (53 * hash) + getPartitionSize().hashCode(); + break; + case 0: + default: + } + hash = (29 * hash) + unknownFields.hashCode(); + memoizedHashCode = hash; + return hash; + } + + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + java.nio.ByteBuffer data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + java.nio.ByteBuffer data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + com.google.protobuf.ByteString data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + com.google.protobuf.ByteString data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom(byte[] data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + byte[] data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom(java.io.InputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + java.io.InputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input, extensionRegistry); + } + public static flyteidl.core.Tasks.GPUAccelerator parseDelimitedFrom(java.io.InputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseDelimitedWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.GPUAccelerator parseDelimitedFrom( + java.io.InputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseDelimitedWithIOException(PARSER, input, extensionRegistry); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + com.google.protobuf.CodedInputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.GPUAccelerator parseFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input, extensionRegistry); + } + + @java.lang.Override + public Builder newBuilderForType() { return newBuilder(); } + public static Builder newBuilder() { + return DEFAULT_INSTANCE.toBuilder(); + } + public static Builder newBuilder(flyteidl.core.Tasks.GPUAccelerator prototype) { + return DEFAULT_INSTANCE.toBuilder().mergeFrom(prototype); + } + @java.lang.Override + public Builder toBuilder() { + return this == DEFAULT_INSTANCE + ? new Builder() : new Builder().mergeFrom(this); + } + + @java.lang.Override + protected Builder newBuilderForType( + com.google.protobuf.GeneratedMessageV3.BuilderParent parent) { + Builder builder = new Builder(parent); + return builder; + } + /** + *
+     * Metadata associated with the GPU accelerator to allocate to a task. Contains
+     * information about device type, and for multi-instance GPUs, the partition size to
+     * use.
+     * 
+ * + * Protobuf type {@code flyteidl.core.GPUAccelerator} + */ + public static final class Builder extends + com.google.protobuf.GeneratedMessageV3.Builder implements + // @@protoc_insertion_point(builder_implements:flyteidl.core.GPUAccelerator) + flyteidl.core.Tasks.GPUAcceleratorOrBuilder { + public static final com.google.protobuf.Descriptors.Descriptor + getDescriptor() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_GPUAccelerator_descriptor; + } + + @java.lang.Override + protected com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internalGetFieldAccessorTable() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_GPUAccelerator_fieldAccessorTable + .ensureFieldAccessorsInitialized( + flyteidl.core.Tasks.GPUAccelerator.class, flyteidl.core.Tasks.GPUAccelerator.Builder.class); + } + + // Construct using flyteidl.core.Tasks.GPUAccelerator.newBuilder() + private Builder() { + maybeForceBuilderInitialization(); + } + + private Builder( + com.google.protobuf.GeneratedMessageV3.BuilderParent parent) { + super(parent); + maybeForceBuilderInitialization(); + } + private void maybeForceBuilderInitialization() { + if (com.google.protobuf.GeneratedMessageV3 + .alwaysUseFieldBuilders) { + } + } + @java.lang.Override + public Builder clear() { + super.clear(); + device_ = ""; + + partitionSizeValueCase_ = 0; + partitionSizeValue_ = null; + return this; + } + + @java.lang.Override + public com.google.protobuf.Descriptors.Descriptor + getDescriptorForType() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_GPUAccelerator_descriptor; + } + + @java.lang.Override + public flyteidl.core.Tasks.GPUAccelerator getDefaultInstanceForType() { + return flyteidl.core.Tasks.GPUAccelerator.getDefaultInstance(); + } + + @java.lang.Override + public flyteidl.core.Tasks.GPUAccelerator build() { + flyteidl.core.Tasks.GPUAccelerator result = buildPartial(); + if (!result.isInitialized()) { + throw newUninitializedMessageException(result); + } + return result; + } + + @java.lang.Override + public flyteidl.core.Tasks.GPUAccelerator buildPartial() { + flyteidl.core.Tasks.GPUAccelerator result = new flyteidl.core.Tasks.GPUAccelerator(this); + result.device_ = device_; + if (partitionSizeValueCase_ == 2) { + result.partitionSizeValue_ = partitionSizeValue_; + } + if (partitionSizeValueCase_ == 3) { + result.partitionSizeValue_ = partitionSizeValue_; + } + result.partitionSizeValueCase_ = partitionSizeValueCase_; + onBuilt(); + return result; + } + + @java.lang.Override + public Builder clone() { + return super.clone(); + } + @java.lang.Override + public Builder setField( + com.google.protobuf.Descriptors.FieldDescriptor field, + java.lang.Object value) { + return super.setField(field, value); + } + @java.lang.Override + public Builder clearField( + com.google.protobuf.Descriptors.FieldDescriptor field) { + return super.clearField(field); + } + @java.lang.Override + public Builder clearOneof( + com.google.protobuf.Descriptors.OneofDescriptor oneof) { + return super.clearOneof(oneof); + } + @java.lang.Override + public Builder setRepeatedField( + com.google.protobuf.Descriptors.FieldDescriptor field, + int index, java.lang.Object value) { + return super.setRepeatedField(field, index, value); + } + @java.lang.Override + public Builder addRepeatedField( + com.google.protobuf.Descriptors.FieldDescriptor field, + java.lang.Object value) { + return super.addRepeatedField(field, value); + } + @java.lang.Override + public Builder mergeFrom(com.google.protobuf.Message other) { + if (other instanceof flyteidl.core.Tasks.GPUAccelerator) { + return mergeFrom((flyteidl.core.Tasks.GPUAccelerator)other); + } else { + super.mergeFrom(other); + return this; + } + } + + public Builder mergeFrom(flyteidl.core.Tasks.GPUAccelerator other) { + if (other == flyteidl.core.Tasks.GPUAccelerator.getDefaultInstance()) return this; + if (!other.getDevice().isEmpty()) { + device_ = other.device_; + onChanged(); + } + switch (other.getPartitionSizeValueCase()) { + case UNPARTITIONED: { + setUnpartitioned(other.getUnpartitioned()); + break; + } + case PARTITION_SIZE: { + partitionSizeValueCase_ = 3; + partitionSizeValue_ = other.partitionSizeValue_; + onChanged(); + break; + } + case PARTITIONSIZEVALUE_NOT_SET: { + break; + } + } + this.mergeUnknownFields(other.unknownFields); + onChanged(); + return this; + } + + @java.lang.Override + public final boolean isInitialized() { + return true; + } + + @java.lang.Override + public Builder mergeFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + flyteidl.core.Tasks.GPUAccelerator parsedMessage = null; + try { + parsedMessage = PARSER.parsePartialFrom(input, extensionRegistry); + } catch (com.google.protobuf.InvalidProtocolBufferException e) { + parsedMessage = (flyteidl.core.Tasks.GPUAccelerator) e.getUnfinishedMessage(); + throw e.unwrapIOException(); + } finally { + if (parsedMessage != null) { + mergeFrom(parsedMessage); + } + } + return this; + } + private int partitionSizeValueCase_ = 0; + private java.lang.Object partitionSizeValue_; + public PartitionSizeValueCase + getPartitionSizeValueCase() { + return PartitionSizeValueCase.forNumber( + partitionSizeValueCase_); + } + + public Builder clearPartitionSizeValue() { + partitionSizeValueCase_ = 0; + partitionSizeValue_ = null; + onChanged(); + return this; + } + + + private java.lang.Object device_ = ""; + /** + *
+       * This can be any arbitrary string, and should be informed by the labels or taints
+       * associated with the nodes in question. Default cloud provider labels typically
+       * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+       * 
+ * + * string device = 1; + */ + public java.lang.String getDevice() { + java.lang.Object ref = device_; + if (!(ref instanceof java.lang.String)) { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + device_ = s; + return s; + } else { + return (java.lang.String) ref; + } + } + /** + *
+       * This can be any arbitrary string, and should be informed by the labels or taints
+       * associated with the nodes in question. Default cloud provider labels typically
+       * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+       * 
+ * + * string device = 1; + */ + public com.google.protobuf.ByteString + getDeviceBytes() { + java.lang.Object ref = device_; + if (ref instanceof String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + device_ = b; + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + /** + *
+       * This can be any arbitrary string, and should be informed by the labels or taints
+       * associated with the nodes in question. Default cloud provider labels typically
+       * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+       * 
+ * + * string device = 1; + */ + public Builder setDevice( + java.lang.String value) { + if (value == null) { + throw new NullPointerException(); + } + + device_ = value; + onChanged(); + return this; + } + /** + *
+       * This can be any arbitrary string, and should be informed by the labels or taints
+       * associated with the nodes in question. Default cloud provider labels typically
+       * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+       * 
+ * + * string device = 1; + */ + public Builder clearDevice() { + + device_ = getDefaultInstance().getDevice(); + onChanged(); + return this; + } + /** + *
+       * This can be any arbitrary string, and should be informed by the labels or taints
+       * associated with the nodes in question. Default cloud provider labels typically
+       * use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc.
+       * 
+ * + * string device = 1; + */ + public Builder setDeviceBytes( + com.google.protobuf.ByteString value) { + if (value == null) { + throw new NullPointerException(); + } + checkByteStringIsUtf8(value); + + device_ = value; + onChanged(); + return this; + } + + /** + * bool unpartitioned = 2; + */ + public boolean getUnpartitioned() { + if (partitionSizeValueCase_ == 2) { + return (java.lang.Boolean) partitionSizeValue_; + } + return false; + } + /** + * bool unpartitioned = 2; + */ + public Builder setUnpartitioned(boolean value) { + partitionSizeValueCase_ = 2; + partitionSizeValue_ = value; + onChanged(); + return this; + } + /** + * bool unpartitioned = 2; + */ + public Builder clearUnpartitioned() { + if (partitionSizeValueCase_ == 2) { + partitionSizeValueCase_ = 0; + partitionSizeValue_ = null; + onChanged(); + } + return this; + } + + /** + *
+       * Like `device`, this can be any arbitrary string, and should be informed by
+       * the labels or taints associated with the nodes in question. Default cloud
+       * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+       * 
+ * + * string partition_size = 3; + */ + public java.lang.String getPartitionSize() { + java.lang.Object ref = ""; + if (partitionSizeValueCase_ == 3) { + ref = partitionSizeValue_; + } + if (!(ref instanceof java.lang.String)) { + com.google.protobuf.ByteString bs = + (com.google.protobuf.ByteString) ref; + java.lang.String s = bs.toStringUtf8(); + if (partitionSizeValueCase_ == 3) { + partitionSizeValue_ = s; + } + return s; + } else { + return (java.lang.String) ref; + } + } + /** + *
+       * Like `device`, this can be any arbitrary string, and should be informed by
+       * the labels or taints associated with the nodes in question. Default cloud
+       * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+       * 
+ * + * string partition_size = 3; + */ + public com.google.protobuf.ByteString + getPartitionSizeBytes() { + java.lang.Object ref = ""; + if (partitionSizeValueCase_ == 3) { + ref = partitionSizeValue_; + } + if (ref instanceof String) { + com.google.protobuf.ByteString b = + com.google.protobuf.ByteString.copyFromUtf8( + (java.lang.String) ref); + if (partitionSizeValueCase_ == 3) { + partitionSizeValue_ = b; + } + return b; + } else { + return (com.google.protobuf.ByteString) ref; + } + } + /** + *
+       * Like `device`, this can be any arbitrary string, and should be informed by
+       * the labels or taints associated with the nodes in question. Default cloud
+       * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+       * 
+ * + * string partition_size = 3; + */ + public Builder setPartitionSize( + java.lang.String value) { + if (value == null) { + throw new NullPointerException(); + } + partitionSizeValueCase_ = 3; + partitionSizeValue_ = value; + onChanged(); + return this; + } + /** + *
+       * Like `device`, this can be any arbitrary string, and should be informed by
+       * the labels or taints associated with the nodes in question. Default cloud
+       * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+       * 
+ * + * string partition_size = 3; + */ + public Builder clearPartitionSize() { + if (partitionSizeValueCase_ == 3) { + partitionSizeValueCase_ = 0; + partitionSizeValue_ = null; + onChanged(); + } + return this; + } + /** + *
+       * Like `device`, this can be any arbitrary string, and should be informed by
+       * the labels or taints associated with the nodes in question. Default cloud
+       * provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc.
+       * 
+ * + * string partition_size = 3; + */ + public Builder setPartitionSizeBytes( + com.google.protobuf.ByteString value) { + if (value == null) { + throw new NullPointerException(); + } + checkByteStringIsUtf8(value); + partitionSizeValueCase_ = 3; + partitionSizeValue_ = value; + onChanged(); + return this; + } + @java.lang.Override + public final Builder setUnknownFields( + final com.google.protobuf.UnknownFieldSet unknownFields) { + return super.setUnknownFields(unknownFields); + } + + @java.lang.Override + public final Builder mergeUnknownFields( + final com.google.protobuf.UnknownFieldSet unknownFields) { + return super.mergeUnknownFields(unknownFields); + } + + + // @@protoc_insertion_point(builder_scope:flyteidl.core.GPUAccelerator) + } + + // @@protoc_insertion_point(class_scope:flyteidl.core.GPUAccelerator) + private static final flyteidl.core.Tasks.GPUAccelerator DEFAULT_INSTANCE; + static { + DEFAULT_INSTANCE = new flyteidl.core.Tasks.GPUAccelerator(); + } + + public static flyteidl.core.Tasks.GPUAccelerator getDefaultInstance() { + return DEFAULT_INSTANCE; + } + + private static final com.google.protobuf.Parser + PARSER = new com.google.protobuf.AbstractParser() { + @java.lang.Override + public GPUAccelerator parsePartialFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return new GPUAccelerator(input, extensionRegistry); + } + }; + + public static com.google.protobuf.Parser parser() { + return PARSER; + } + + @java.lang.Override + public com.google.protobuf.Parser getParserForType() { + return PARSER; + } + + @java.lang.Override + public flyteidl.core.Tasks.GPUAccelerator getDefaultInstanceForType() { + return DEFAULT_INSTANCE; + } + + } + + public interface ExtendedResourcesOrBuilder extends + // @@protoc_insertion_point(interface_extends:flyteidl.core.ExtendedResources) + com.google.protobuf.MessageOrBuilder { + + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + boolean hasGpuAccelerator(); + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + flyteidl.core.Tasks.GPUAccelerator getGpuAccelerator(); + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + flyteidl.core.Tasks.GPUAcceleratorOrBuilder getGpuAcceleratorOrBuilder(); + } + /** + *
+   * Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to
+   * allocate to a task.
+   * 
+ * + * Protobuf type {@code flyteidl.core.ExtendedResources} + */ + public static final class ExtendedResources extends + com.google.protobuf.GeneratedMessageV3 implements + // @@protoc_insertion_point(message_implements:flyteidl.core.ExtendedResources) + ExtendedResourcesOrBuilder { + private static final long serialVersionUID = 0L; + // Use ExtendedResources.newBuilder() to construct. + private ExtendedResources(com.google.protobuf.GeneratedMessageV3.Builder builder) { + super(builder); + } + private ExtendedResources() { + } + + @java.lang.Override + public final com.google.protobuf.UnknownFieldSet + getUnknownFields() { + return this.unknownFields; + } + private ExtendedResources( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + this(); + if (extensionRegistry == null) { + throw new java.lang.NullPointerException(); + } + int mutable_bitField0_ = 0; + com.google.protobuf.UnknownFieldSet.Builder unknownFields = + com.google.protobuf.UnknownFieldSet.newBuilder(); + try { + boolean done = false; + while (!done) { + int tag = input.readTag(); + switch (tag) { + case 0: + done = true; + break; + case 10: { + flyteidl.core.Tasks.GPUAccelerator.Builder subBuilder = null; + if (gpuAccelerator_ != null) { + subBuilder = gpuAccelerator_.toBuilder(); + } + gpuAccelerator_ = input.readMessage(flyteidl.core.Tasks.GPUAccelerator.parser(), extensionRegistry); + if (subBuilder != null) { + subBuilder.mergeFrom(gpuAccelerator_); + gpuAccelerator_ = subBuilder.buildPartial(); + } + + break; + } + default: { + if (!parseUnknownField( + input, unknownFields, extensionRegistry, tag)) { + done = true; + } + break; + } + } + } + } catch (com.google.protobuf.InvalidProtocolBufferException e) { + throw e.setUnfinishedMessage(this); + } catch (java.io.IOException e) { + throw new com.google.protobuf.InvalidProtocolBufferException( + e).setUnfinishedMessage(this); + } finally { + this.unknownFields = unknownFields.build(); + makeExtensionsImmutable(); + } + } + public static final com.google.protobuf.Descriptors.Descriptor + getDescriptor() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_ExtendedResources_descriptor; + } + + @java.lang.Override + protected com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internalGetFieldAccessorTable() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_ExtendedResources_fieldAccessorTable + .ensureFieldAccessorsInitialized( + flyteidl.core.Tasks.ExtendedResources.class, flyteidl.core.Tasks.ExtendedResources.Builder.class); + } + + public static final int GPU_ACCELERATOR_FIELD_NUMBER = 1; + private flyteidl.core.Tasks.GPUAccelerator gpuAccelerator_; + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public boolean hasGpuAccelerator() { + return gpuAccelerator_ != null; + } + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public flyteidl.core.Tasks.GPUAccelerator getGpuAccelerator() { + return gpuAccelerator_ == null ? flyteidl.core.Tasks.GPUAccelerator.getDefaultInstance() : gpuAccelerator_; + } + /** + *
+     * GPU accelerator to select for task. Contains information about device type, and
+     * for multi-instance GPUs, the partition size to use.
+     * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public flyteidl.core.Tasks.GPUAcceleratorOrBuilder getGpuAcceleratorOrBuilder() { + return getGpuAccelerator(); + } + + private byte memoizedIsInitialized = -1; + @java.lang.Override + public final boolean isInitialized() { + byte isInitialized = memoizedIsInitialized; + if (isInitialized == 1) return true; + if (isInitialized == 0) return false; + + memoizedIsInitialized = 1; + return true; + } + + @java.lang.Override + public void writeTo(com.google.protobuf.CodedOutputStream output) + throws java.io.IOException { + if (gpuAccelerator_ != null) { + output.writeMessage(1, getGpuAccelerator()); + } + unknownFields.writeTo(output); + } + + @java.lang.Override + public int getSerializedSize() { + int size = memoizedSize; + if (size != -1) return size; + + size = 0; + if (gpuAccelerator_ != null) { + size += com.google.protobuf.CodedOutputStream + .computeMessageSize(1, getGpuAccelerator()); + } + size += unknownFields.getSerializedSize(); + memoizedSize = size; + return size; + } + + @java.lang.Override + public boolean equals(final java.lang.Object obj) { + if (obj == this) { + return true; + } + if (!(obj instanceof flyteidl.core.Tasks.ExtendedResources)) { + return super.equals(obj); + } + flyteidl.core.Tasks.ExtendedResources other = (flyteidl.core.Tasks.ExtendedResources) obj; + + if (hasGpuAccelerator() != other.hasGpuAccelerator()) return false; + if (hasGpuAccelerator()) { + if (!getGpuAccelerator() + .equals(other.getGpuAccelerator())) return false; + } + if (!unknownFields.equals(other.unknownFields)) return false; + return true; + } + + @java.lang.Override + public int hashCode() { + if (memoizedHashCode != 0) { + return memoizedHashCode; + } + int hash = 41; + hash = (19 * hash) + getDescriptor().hashCode(); + if (hasGpuAccelerator()) { + hash = (37 * hash) + GPU_ACCELERATOR_FIELD_NUMBER; + hash = (53 * hash) + getGpuAccelerator().hashCode(); + } + hash = (29 * hash) + unknownFields.hashCode(); + memoizedHashCode = hash; + return hash; + } + + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + java.nio.ByteBuffer data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + java.nio.ByteBuffer data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + com.google.protobuf.ByteString data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + com.google.protobuf.ByteString data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom(byte[] data) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + byte[] data, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return PARSER.parseFrom(data, extensionRegistry); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom(java.io.InputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + java.io.InputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input, extensionRegistry); + } + public static flyteidl.core.Tasks.ExtendedResources parseDelimitedFrom(java.io.InputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseDelimitedWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.ExtendedResources parseDelimitedFrom( + java.io.InputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseDelimitedWithIOException(PARSER, input, extensionRegistry); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + com.google.protobuf.CodedInputStream input) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input); + } + public static flyteidl.core.Tasks.ExtendedResources parseFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + return com.google.protobuf.GeneratedMessageV3 + .parseWithIOException(PARSER, input, extensionRegistry); + } + + @java.lang.Override + public Builder newBuilderForType() { return newBuilder(); } + public static Builder newBuilder() { + return DEFAULT_INSTANCE.toBuilder(); + } + public static Builder newBuilder(flyteidl.core.Tasks.ExtendedResources prototype) { + return DEFAULT_INSTANCE.toBuilder().mergeFrom(prototype); + } + @java.lang.Override + public Builder toBuilder() { + return this == DEFAULT_INSTANCE + ? new Builder() : new Builder().mergeFrom(this); + } + + @java.lang.Override + protected Builder newBuilderForType( + com.google.protobuf.GeneratedMessageV3.BuilderParent parent) { + Builder builder = new Builder(parent); + return builder; + } + /** + *
+     * Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to
+     * allocate to a task.
+     * 
+ * + * Protobuf type {@code flyteidl.core.ExtendedResources} + */ + public static final class Builder extends + com.google.protobuf.GeneratedMessageV3.Builder implements + // @@protoc_insertion_point(builder_implements:flyteidl.core.ExtendedResources) + flyteidl.core.Tasks.ExtendedResourcesOrBuilder { + public static final com.google.protobuf.Descriptors.Descriptor + getDescriptor() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_ExtendedResources_descriptor; + } + + @java.lang.Override + protected com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internalGetFieldAccessorTable() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_ExtendedResources_fieldAccessorTable + .ensureFieldAccessorsInitialized( + flyteidl.core.Tasks.ExtendedResources.class, flyteidl.core.Tasks.ExtendedResources.Builder.class); + } + + // Construct using flyteidl.core.Tasks.ExtendedResources.newBuilder() + private Builder() { + maybeForceBuilderInitialization(); + } + + private Builder( + com.google.protobuf.GeneratedMessageV3.BuilderParent parent) { + super(parent); + maybeForceBuilderInitialization(); + } + private void maybeForceBuilderInitialization() { + if (com.google.protobuf.GeneratedMessageV3 + .alwaysUseFieldBuilders) { + } + } + @java.lang.Override + public Builder clear() { + super.clear(); + if (gpuAcceleratorBuilder_ == null) { + gpuAccelerator_ = null; + } else { + gpuAccelerator_ = null; + gpuAcceleratorBuilder_ = null; + } + return this; + } + + @java.lang.Override + public com.google.protobuf.Descriptors.Descriptor + getDescriptorForType() { + return flyteidl.core.Tasks.internal_static_flyteidl_core_ExtendedResources_descriptor; + } + + @java.lang.Override + public flyteidl.core.Tasks.ExtendedResources getDefaultInstanceForType() { + return flyteidl.core.Tasks.ExtendedResources.getDefaultInstance(); + } + + @java.lang.Override + public flyteidl.core.Tasks.ExtendedResources build() { + flyteidl.core.Tasks.ExtendedResources result = buildPartial(); + if (!result.isInitialized()) { + throw newUninitializedMessageException(result); + } + return result; + } + + @java.lang.Override + public flyteidl.core.Tasks.ExtendedResources buildPartial() { + flyteidl.core.Tasks.ExtendedResources result = new flyteidl.core.Tasks.ExtendedResources(this); + if (gpuAcceleratorBuilder_ == null) { + result.gpuAccelerator_ = gpuAccelerator_; + } else { + result.gpuAccelerator_ = gpuAcceleratorBuilder_.build(); + } + onBuilt(); + return result; + } + + @java.lang.Override + public Builder clone() { + return super.clone(); + } + @java.lang.Override + public Builder setField( + com.google.protobuf.Descriptors.FieldDescriptor field, + java.lang.Object value) { + return super.setField(field, value); + } + @java.lang.Override + public Builder clearField( + com.google.protobuf.Descriptors.FieldDescriptor field) { + return super.clearField(field); + } + @java.lang.Override + public Builder clearOneof( + com.google.protobuf.Descriptors.OneofDescriptor oneof) { + return super.clearOneof(oneof); + } + @java.lang.Override + public Builder setRepeatedField( + com.google.protobuf.Descriptors.FieldDescriptor field, + int index, java.lang.Object value) { + return super.setRepeatedField(field, index, value); + } + @java.lang.Override + public Builder addRepeatedField( + com.google.protobuf.Descriptors.FieldDescriptor field, + java.lang.Object value) { + return super.addRepeatedField(field, value); + } + @java.lang.Override + public Builder mergeFrom(com.google.protobuf.Message other) { + if (other instanceof flyteidl.core.Tasks.ExtendedResources) { + return mergeFrom((flyteidl.core.Tasks.ExtendedResources)other); + } else { + super.mergeFrom(other); + return this; + } + } + + public Builder mergeFrom(flyteidl.core.Tasks.ExtendedResources other) { + if (other == flyteidl.core.Tasks.ExtendedResources.getDefaultInstance()) return this; + if (other.hasGpuAccelerator()) { + mergeGpuAccelerator(other.getGpuAccelerator()); + } + this.mergeUnknownFields(other.unknownFields); + onChanged(); + return this; + } + + @java.lang.Override + public final boolean isInitialized() { + return true; + } + + @java.lang.Override + public Builder mergeFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws java.io.IOException { + flyteidl.core.Tasks.ExtendedResources parsedMessage = null; + try { + parsedMessage = PARSER.parsePartialFrom(input, extensionRegistry); + } catch (com.google.protobuf.InvalidProtocolBufferException e) { + parsedMessage = (flyteidl.core.Tasks.ExtendedResources) e.getUnfinishedMessage(); + throw e.unwrapIOException(); + } finally { + if (parsedMessage != null) { + mergeFrom(parsedMessage); + } + } + return this; + } + + private flyteidl.core.Tasks.GPUAccelerator gpuAccelerator_; + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.GPUAccelerator, flyteidl.core.Tasks.GPUAccelerator.Builder, flyteidl.core.Tasks.GPUAcceleratorOrBuilder> gpuAcceleratorBuilder_; + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public boolean hasGpuAccelerator() { + return gpuAcceleratorBuilder_ != null || gpuAccelerator_ != null; + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public flyteidl.core.Tasks.GPUAccelerator getGpuAccelerator() { + if (gpuAcceleratorBuilder_ == null) { + return gpuAccelerator_ == null ? flyteidl.core.Tasks.GPUAccelerator.getDefaultInstance() : gpuAccelerator_; + } else { + return gpuAcceleratorBuilder_.getMessage(); + } + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public Builder setGpuAccelerator(flyteidl.core.Tasks.GPUAccelerator value) { + if (gpuAcceleratorBuilder_ == null) { + if (value == null) { + throw new NullPointerException(); + } + gpuAccelerator_ = value; + onChanged(); + } else { + gpuAcceleratorBuilder_.setMessage(value); + } + + return this; + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public Builder setGpuAccelerator( + flyteidl.core.Tasks.GPUAccelerator.Builder builderForValue) { + if (gpuAcceleratorBuilder_ == null) { + gpuAccelerator_ = builderForValue.build(); + onChanged(); + } else { + gpuAcceleratorBuilder_.setMessage(builderForValue.build()); + } + + return this; + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public Builder mergeGpuAccelerator(flyteidl.core.Tasks.GPUAccelerator value) { + if (gpuAcceleratorBuilder_ == null) { + if (gpuAccelerator_ != null) { + gpuAccelerator_ = + flyteidl.core.Tasks.GPUAccelerator.newBuilder(gpuAccelerator_).mergeFrom(value).buildPartial(); + } else { + gpuAccelerator_ = value; + } + onChanged(); + } else { + gpuAcceleratorBuilder_.mergeFrom(value); + } + + return this; + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public Builder clearGpuAccelerator() { + if (gpuAcceleratorBuilder_ == null) { + gpuAccelerator_ = null; + onChanged(); + } else { + gpuAccelerator_ = null; + gpuAcceleratorBuilder_ = null; + } + + return this; + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public flyteidl.core.Tasks.GPUAccelerator.Builder getGpuAcceleratorBuilder() { + + onChanged(); + return getGpuAcceleratorFieldBuilder().getBuilder(); + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + public flyteidl.core.Tasks.GPUAcceleratorOrBuilder getGpuAcceleratorOrBuilder() { + if (gpuAcceleratorBuilder_ != null) { + return gpuAcceleratorBuilder_.getMessageOrBuilder(); + } else { + return gpuAccelerator_ == null ? + flyteidl.core.Tasks.GPUAccelerator.getDefaultInstance() : gpuAccelerator_; + } + } + /** + *
+       * GPU accelerator to select for task. Contains information about device type, and
+       * for multi-instance GPUs, the partition size to use.
+       * 
+ * + * .flyteidl.core.GPUAccelerator gpu_accelerator = 1; + */ + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.GPUAccelerator, flyteidl.core.Tasks.GPUAccelerator.Builder, flyteidl.core.Tasks.GPUAcceleratorOrBuilder> + getGpuAcceleratorFieldBuilder() { + if (gpuAcceleratorBuilder_ == null) { + gpuAcceleratorBuilder_ = new com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.GPUAccelerator, flyteidl.core.Tasks.GPUAccelerator.Builder, flyteidl.core.Tasks.GPUAcceleratorOrBuilder>( + getGpuAccelerator(), + getParentForChildren(), + isClean()); + gpuAccelerator_ = null; + } + return gpuAcceleratorBuilder_; + } + @java.lang.Override + public final Builder setUnknownFields( + final com.google.protobuf.UnknownFieldSet unknownFields) { + return super.setUnknownFields(unknownFields); + } + + @java.lang.Override + public final Builder mergeUnknownFields( + final com.google.protobuf.UnknownFieldSet unknownFields) { + return super.mergeUnknownFields(unknownFields); + } + + + // @@protoc_insertion_point(builder_scope:flyteidl.core.ExtendedResources) + } + + // @@protoc_insertion_point(class_scope:flyteidl.core.ExtendedResources) + private static final flyteidl.core.Tasks.ExtendedResources DEFAULT_INSTANCE; + static { + DEFAULT_INSTANCE = new flyteidl.core.Tasks.ExtendedResources(); + } + + public static flyteidl.core.Tasks.ExtendedResources getDefaultInstance() { + return DEFAULT_INSTANCE; + } + + private static final com.google.protobuf.Parser + PARSER = new com.google.protobuf.AbstractParser() { + @java.lang.Override + public ExtendedResources parsePartialFrom( + com.google.protobuf.CodedInputStream input, + com.google.protobuf.ExtensionRegistryLite extensionRegistry) + throws com.google.protobuf.InvalidProtocolBufferException { + return new ExtendedResources(input, extensionRegistry); + } + }; + + public static com.google.protobuf.Parser parser() { + return PARSER; + } + + @java.lang.Override + public com.google.protobuf.Parser getParserForType() { + return PARSER; + } + + @java.lang.Override + public flyteidl.core.Tasks.ExtendedResources getDefaultInstanceForType() { + return DEFAULT_INSTANCE; + } + + } + public interface RuntimeMetadataOrBuilder extends // @@protoc_insertion_point(interface_extends:flyteidl.core.RuntimeMetadata) com.google.protobuf.MessageOrBuilder { @@ -6006,6 +7676,34 @@ public interface TaskTemplateOrBuilder extends */ flyteidl.core.Security.SecurityContextOrBuilder getSecurityContextOrBuilder(); + /** + *
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + boolean hasExtendedResources(); + /** + *
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + flyteidl.core.Tasks.ExtendedResources getExtendedResources(); + /** + *
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder(); + /** *
      * Metadata about the custom defined for this task. This is extensible to allow various plugins in the system
@@ -6207,11 +7905,24 @@ private TaskTemplate(
 
               break;
             }
+            case 74: {
+              flyteidl.core.Tasks.ExtendedResources.Builder subBuilder = null;
+              if (extendedResources_ != null) {
+                subBuilder = extendedResources_.toBuilder();
+              }
+              extendedResources_ = input.readMessage(flyteidl.core.Tasks.ExtendedResources.parser(), extensionRegistry);
+              if (subBuilder != null) {
+                subBuilder.mergeFrom(extendedResources_);
+                extendedResources_ = subBuilder.buildPartial();
+              }
+
+              break;
+            }
             case 130: {
-              if (!((mutable_bitField0_ & 0x00000400) != 0)) {
+              if (!((mutable_bitField0_ & 0x00000800) != 0)) {
                 config_ = com.google.protobuf.MapField.newMapField(
                     ConfigDefaultEntryHolder.defaultEntry);
-                mutable_bitField0_ |= 0x00000400;
+                mutable_bitField0_ |= 0x00000800;
               }
               com.google.protobuf.MapEntry
               config__ = input.readMessage(
@@ -6638,6 +8349,42 @@ public flyteidl.core.Security.SecurityContextOrBuilder getSecurityContextOrBuild
       return getSecurityContext();
     }
 
+    public static final int EXTENDED_RESOURCES_FIELD_NUMBER = 9;
+    private flyteidl.core.Tasks.ExtendedResources extendedResources_;
+    /**
+     * 
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public boolean hasExtendedResources() { + return extendedResources_ != null; + } + /** + *
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public flyteidl.core.Tasks.ExtendedResources getExtendedResources() { + return extendedResources_ == null ? flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } + /** + *
+     * Encapsulates all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder() { + return getExtendedResources(); + } + public static final int CONFIG_FIELD_NUMBER = 16; private static final class ConfigDefaultEntryHolder { static final com.google.protobuf.MapEntry< @@ -6776,6 +8523,9 @@ public void writeTo(com.google.protobuf.CodedOutputStream output) if (securityContext_ != null) { output.writeMessage(8, getSecurityContext()); } + if (extendedResources_ != null) { + output.writeMessage(9, getExtendedResources()); + } com.google.protobuf.GeneratedMessageV3 .serializeStringMapTo( output, @@ -6828,6 +8578,10 @@ public int getSerializedSize() { size += com.google.protobuf.CodedOutputStream .computeMessageSize(8, getSecurityContext()); } + if (extendedResources_ != null) { + size += com.google.protobuf.CodedOutputStream + .computeMessageSize(9, getExtendedResources()); + } for (java.util.Map.Entry entry : internalGetConfig().getMap().entrySet()) { com.google.protobuf.MapEntry @@ -6890,6 +8644,11 @@ public boolean equals(final java.lang.Object obj) { if (!getSecurityContext() .equals(other.getSecurityContext())) return false; } + if (hasExtendedResources() != other.hasExtendedResources()) return false; + if (hasExtendedResources()) { + if (!getExtendedResources() + .equals(other.getExtendedResources())) return false; + } if (!internalGetConfig().equals( other.internalGetConfig())) return false; if (!getTargetCase().equals(other.getTargetCase())) return false; @@ -6944,6 +8703,10 @@ public int hashCode() { hash = (37 * hash) + SECURITY_CONTEXT_FIELD_NUMBER; hash = (53 * hash) + getSecurityContext().hashCode(); } + if (hasExtendedResources()) { + hash = (37 * hash) + EXTENDED_RESOURCES_FIELD_NUMBER; + hash = (53 * hash) + getExtendedResources().hashCode(); + } if (!internalGetConfig().getMap().isEmpty()) { hash = (37 * hash) + CONFIG_FIELD_NUMBER; hash = (53 * hash) + internalGetConfig().hashCode(); @@ -7158,6 +8921,12 @@ public Builder clear() { securityContext_ = null; securityContextBuilder_ = null; } + if (extendedResourcesBuilder_ == null) { + extendedResources_ = null; + } else { + extendedResources_ = null; + extendedResourcesBuilder_ = null; + } internalGetMutableConfig().clear(); targetCase_ = 0; target_ = null; @@ -7237,6 +9006,11 @@ public flyteidl.core.Tasks.TaskTemplate buildPartial() { } else { result.securityContext_ = securityContextBuilder_.build(); } + if (extendedResourcesBuilder_ == null) { + result.extendedResources_ = extendedResources_; + } else { + result.extendedResources_ = extendedResourcesBuilder_.build(); + } result.config_ = internalGetConfig(); result.config_.makeImmutable(); result.bitField0_ = to_bitField0_; @@ -7311,6 +9085,9 @@ public Builder mergeFrom(flyteidl.core.Tasks.TaskTemplate other) { if (other.hasSecurityContext()) { mergeSecurityContext(other.getSecurityContext()); } + if (other.hasExtendedResources()) { + mergeExtendedResources(other.getExtendedResources()); + } internalGetMutableConfig().mergeFrom( other.internalGetConfig()); switch (other.getTargetCase()) { @@ -8694,6 +10471,168 @@ public flyteidl.core.Security.SecurityContextOrBuilder getSecurityContextOrBuild return securityContextBuilder_; } + private flyteidl.core.Tasks.ExtendedResources extendedResources_; + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder> extendedResourcesBuilder_; + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public boolean hasExtendedResources() { + return extendedResourcesBuilder_ != null || extendedResources_ != null; + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public flyteidl.core.Tasks.ExtendedResources getExtendedResources() { + if (extendedResourcesBuilder_ == null) { + return extendedResources_ == null ? flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } else { + return extendedResourcesBuilder_.getMessage(); + } + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public Builder setExtendedResources(flyteidl.core.Tasks.ExtendedResources value) { + if (extendedResourcesBuilder_ == null) { + if (value == null) { + throw new NullPointerException(); + } + extendedResources_ = value; + onChanged(); + } else { + extendedResourcesBuilder_.setMessage(value); + } + + return this; + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public Builder setExtendedResources( + flyteidl.core.Tasks.ExtendedResources.Builder builderForValue) { + if (extendedResourcesBuilder_ == null) { + extendedResources_ = builderForValue.build(); + onChanged(); + } else { + extendedResourcesBuilder_.setMessage(builderForValue.build()); + } + + return this; + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public Builder mergeExtendedResources(flyteidl.core.Tasks.ExtendedResources value) { + if (extendedResourcesBuilder_ == null) { + if (extendedResources_ != null) { + extendedResources_ = + flyteidl.core.Tasks.ExtendedResources.newBuilder(extendedResources_).mergeFrom(value).buildPartial(); + } else { + extendedResources_ = value; + } + onChanged(); + } else { + extendedResourcesBuilder_.mergeFrom(value); + } + + return this; + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public Builder clearExtendedResources() { + if (extendedResourcesBuilder_ == null) { + extendedResources_ = null; + onChanged(); + } else { + extendedResources_ = null; + extendedResourcesBuilder_ = null; + } + + return this; + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public flyteidl.core.Tasks.ExtendedResources.Builder getExtendedResourcesBuilder() { + + onChanged(); + return getExtendedResourcesFieldBuilder().getBuilder(); + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + public flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder() { + if (extendedResourcesBuilder_ != null) { + return extendedResourcesBuilder_.getMessageOrBuilder(); + } else { + return extendedResources_ == null ? + flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } + } + /** + *
+       * Encapsulates all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 9; + */ + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder> + getExtendedResourcesFieldBuilder() { + if (extendedResourcesBuilder_ == null) { + extendedResourcesBuilder_ = new com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder>( + getExtendedResources(), + getParentForChildren(), + isClean()); + extendedResources_ = null; + } + return extendedResourcesBuilder_; + } + private com.google.protobuf.MapField< java.lang.String, java.lang.String> config_; private com.google.protobuf.MapField @@ -18590,6 +20529,16 @@ public flyteidl.core.Tasks.Sql getDefaultInstanceForType() { private static final com.google.protobuf.GeneratedMessageV3.FieldAccessorTable internal_static_flyteidl_core_Resources_ResourceEntry_fieldAccessorTable; + private static final com.google.protobuf.Descriptors.Descriptor + internal_static_flyteidl_core_GPUAccelerator_descriptor; + private static final + com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internal_static_flyteidl_core_GPUAccelerator_fieldAccessorTable; + private static final com.google.protobuf.Descriptors.Descriptor + internal_static_flyteidl_core_ExtendedResources_descriptor; + private static final + com.google.protobuf.GeneratedMessageV3.FieldAccessorTable + internal_static_flyteidl_core_ExtendedResources_fieldAccessorTable; private static final com.google.protobuf.Descriptors.Descriptor internal_static_flyteidl_core_RuntimeMetadata_descriptor; private static final @@ -18682,76 +20631,82 @@ public flyteidl.core.Tasks.Sql getDefaultInstanceForType() { "re.Resources.ResourceName\022\r\n\005value\030\002 \001(\t" + "\"]\n\014ResourceName\022\013\n\007UNKNOWN\020\000\022\007\n\003CPU\020\001\022\007" + "\n\003GPU\020\002\022\n\n\006MEMORY\020\003\022\013\n\007STORAGE\020\004\022\025\n\021EPHE" + - "MERAL_STORAGE\020\005\"\225\001\n\017RuntimeMetadata\0228\n\004t" + - "ype\030\001 \001(\0162*.flyteidl.core.RuntimeMetadat" + - "a.RuntimeType\022\017\n\007version\030\002 \001(\t\022\016\n\006flavor" + - "\030\003 \001(\t\"\'\n\013RuntimeType\022\t\n\005OTHER\020\000\022\r\n\tFLYT" + - "E_SDK\020\001\"\316\003\n\014TaskMetadata\022\024\n\014discoverable" + - "\030\001 \001(\010\022/\n\007runtime\030\002 \001(\0132\036.flyteidl.core." + - "RuntimeMetadata\022*\n\007timeout\030\004 \001(\0132\031.googl" + - "e.protobuf.Duration\022-\n\007retries\030\005 \001(\0132\034.f" + - "lyteidl.core.RetryStrategy\022\031\n\021discovery_" + - "version\030\006 \001(\t\022 \n\030deprecated_error_messag" + - "e\030\007 \001(\t\022\027\n\rinterruptible\030\010 \001(\010H\000\022\032\n\022cach" + - "e_serializable\030\t \001(\010\022\026\n\016generates_deck\030\n" + - " \001(\010\0223\n\004tags\030\013 \003(\0132%.flyteidl.core.TaskM" + - "etadata.TagsEntry\022\031\n\021pod_template_name\030\014" + - " \001(\t\032+\n\tTagsEntry\022\013\n\003key\030\001 \001(\t\022\r\n\005value\030" + - "\002 \001(\t:\0028\001B\025\n\023interruptible_value\"\220\004\n\014Tas" + - "kTemplate\022%\n\002id\030\001 \001(\0132\031.flyteidl.core.Id" + - "entifier\022\014\n\004type\030\002 \001(\t\022-\n\010metadata\030\003 \001(\013" + - "2\033.flyteidl.core.TaskMetadata\0220\n\tinterfa" + - "ce\030\004 \001(\0132\035.flyteidl.core.TypedInterface\022" + - "\'\n\006custom\030\005 \001(\0132\027.google.protobuf.Struct" + - "\022-\n\tcontainer\030\006 \001(\0132\030.flyteidl.core.Cont" + - "ainerH\000\022(\n\007k8s_pod\030\021 \001(\0132\025.flyteidl.core" + - ".K8sPodH\000\022!\n\003sql\030\022 \001(\0132\022.flyteidl.core.S" + - "qlH\000\022\031\n\021task_type_version\030\007 \001(\005\0228\n\020secur" + - "ity_context\030\010 \001(\0132\036.flyteidl.core.Securi" + - "tyContext\0227\n\006config\030\020 \003(\0132\'.flyteidl.cor" + - "e.TaskTemplate.ConfigEntry\032-\n\013ConfigEntr" + - "y\022\013\n\003key\030\001 \001(\t\022\r\n\005value\030\002 \001(\t:\0028\001B\010\n\006tar" + - "get\"\'\n\rContainerPort\022\026\n\016container_port\030\001" + - " \001(\r\"\255\003\n\tContainer\022\r\n\005image\030\001 \001(\t\022\017\n\007com" + - "mand\030\002 \003(\t\022\014\n\004args\030\003 \003(\t\022+\n\tresources\030\004 " + - "\001(\0132\030.flyteidl.core.Resources\022(\n\003env\030\005 \003" + - "(\0132\033.flyteidl.core.KeyValuePair\022/\n\006confi" + - "g\030\006 \003(\0132\033.flyteidl.core.KeyValuePairB\002\030\001" + - "\022+\n\005ports\030\007 \003(\0132\034.flyteidl.core.Containe" + - "rPort\0225\n\013data_config\030\t \001(\0132 .flyteidl.co" + - "re.DataLoadingConfig\022;\n\014architecture\030\n \001" + - "(\0162%.flyteidl.core.Container.Architectur" + - "e\"I\n\014Architecture\022\013\n\007UNKNOWN\020\000\022\t\n\005AMD64\020" + - "\001\022\t\n\005ARM64\020\002\022\n\n\006ARM_V6\020\003\022\n\n\006ARM_V7\020\004\"\233\002\n" + - "\nIOStrategy\022=\n\rdownload_mode\030\001 \001(\0162&.fly" + - "teidl.core.IOStrategy.DownloadMode\0229\n\013up" + - "load_mode\030\002 \001(\0162$.flyteidl.core.IOStrate" + - "gy.UploadMode\"L\n\014DownloadMode\022\022\n\016DOWNLOA" + - "D_EAGER\020\000\022\023\n\017DOWNLOAD_STREAM\020\001\022\023\n\017DO_NOT" + - "_DOWNLOAD\020\002\"E\n\nUploadMode\022\022\n\016UPLOAD_ON_E" + - "XIT\020\000\022\020\n\014UPLOAD_EAGER\020\001\022\021\n\rDO_NOT_UPLOAD" + - "\020\002\"\363\001\n\021DataLoadingConfig\022\017\n\007enabled\030\001 \001(" + - "\010\022\022\n\ninput_path\030\002 \001(\t\022\023\n\013output_path\030\003 \001" + - "(\t\022A\n\006format\030\004 \001(\01621.flyteidl.core.DataL" + - "oadingConfig.LiteralMapFormat\022.\n\013io_stra" + - "tegy\030\005 \001(\0132\031.flyteidl.core.IOStrategy\"1\n" + - "\020LiteralMapFormat\022\010\n\004JSON\020\000\022\010\n\004YAML\020\001\022\t\n" + - "\005PROTO\020\002\"\236\001\n\006K8sPod\0222\n\010metadata\030\001 \001(\0132 ." + - "flyteidl.core.K8sObjectMetadata\022)\n\010pod_s" + - "pec\030\002 \001(\0132\027.google.protobuf.Struct\0225\n\013da" + - "ta_config\030\003 \001(\0132 .flyteidl.core.DataLoad" + - "ingConfig\"\374\001\n\021K8sObjectMetadata\022<\n\006label" + - "s\030\001 \003(\0132,.flyteidl.core.K8sObjectMetadat" + - "a.LabelsEntry\022F\n\013annotations\030\002 \003(\01321.fly" + - "teidl.core.K8sObjectMetadata.Annotations" + - "Entry\032-\n\013LabelsEntry\022\013\n\003key\030\001 \001(\t\022\r\n\005val" + - "ue\030\002 \001(\t:\0028\001\0322\n\020AnnotationsEntry\022\013\n\003key\030" + - "\001 \001(\t\022\r\n\005value\030\002 \001(\t:\0028\001\"~\n\003Sql\022\021\n\tstate" + - "ment\030\001 \001(\t\022+\n\007dialect\030\002 \001(\0162\032.flyteidl.c" + - "ore.Sql.Dialect\"7\n\007Dialect\022\r\n\tUNDEFINED\020" + - "\000\022\010\n\004ANSI\020\001\022\010\n\004HIVE\020\002\022\t\n\005OTHER\020\003B.flyteidl.core.Resources resources = 1; */ flyteidl.core.Tasks.ResourcesOrBuilder getResourcesOrBuilder(); + + /** + *
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + boolean hasExtendedResources(); + /** + *
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + flyteidl.core.Tasks.ExtendedResources getExtendedResources(); + /** + *
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder(); } /** *
@@ -20446,6 +20474,19 @@ private TaskNodeOverrides(
 
               break;
             }
+            case 18: {
+              flyteidl.core.Tasks.ExtendedResources.Builder subBuilder = null;
+              if (extendedResources_ != null) {
+                subBuilder = extendedResources_.toBuilder();
+              }
+              extendedResources_ = input.readMessage(flyteidl.core.Tasks.ExtendedResources.parser(), extensionRegistry);
+              if (subBuilder != null) {
+                subBuilder.mergeFrom(extendedResources_);
+                extendedResources_ = subBuilder.buildPartial();
+              }
+
+              break;
+            }
             default: {
               if (!parseUnknownField(
                   input, unknownFields, extensionRegistry, tag)) {
@@ -20511,6 +20552,42 @@ public flyteidl.core.Tasks.ResourcesOrBuilder getResourcesOrBuilder() {
       return getResources();
     }
 
+    public static final int EXTENDED_RESOURCES_FIELD_NUMBER = 2;
+    private flyteidl.core.Tasks.ExtendedResources extendedResources_;
+    /**
+     * 
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public boolean hasExtendedResources() { + return extendedResources_ != null; + } + /** + *
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public flyteidl.core.Tasks.ExtendedResources getExtendedResources() { + return extendedResources_ == null ? flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } + /** + *
+     * Overrides for all non-standard resources, not captured by
+     * v1.ResourceRequirements, to allocate to a task.
+     * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder() { + return getExtendedResources(); + } + private byte memoizedIsInitialized = -1; @java.lang.Override public final boolean isInitialized() { @@ -20528,6 +20605,9 @@ public void writeTo(com.google.protobuf.CodedOutputStream output) if (resources_ != null) { output.writeMessage(1, getResources()); } + if (extendedResources_ != null) { + output.writeMessage(2, getExtendedResources()); + } unknownFields.writeTo(output); } @@ -20541,6 +20621,10 @@ public int getSerializedSize() { size += com.google.protobuf.CodedOutputStream .computeMessageSize(1, getResources()); } + if (extendedResources_ != null) { + size += com.google.protobuf.CodedOutputStream + .computeMessageSize(2, getExtendedResources()); + } size += unknownFields.getSerializedSize(); memoizedSize = size; return size; @@ -20561,6 +20645,11 @@ public boolean equals(final java.lang.Object obj) { if (!getResources() .equals(other.getResources())) return false; } + if (hasExtendedResources() != other.hasExtendedResources()) return false; + if (hasExtendedResources()) { + if (!getExtendedResources() + .equals(other.getExtendedResources())) return false; + } if (!unknownFields.equals(other.unknownFields)) return false; return true; } @@ -20576,6 +20665,10 @@ public int hashCode() { hash = (37 * hash) + RESOURCES_FIELD_NUMBER; hash = (53 * hash) + getResources().hashCode(); } + if (hasExtendedResources()) { + hash = (37 * hash) + EXTENDED_RESOURCES_FIELD_NUMBER; + hash = (53 * hash) + getExtendedResources().hashCode(); + } hash = (29 * hash) + unknownFields.hashCode(); memoizedHashCode = hash; return hash; @@ -20719,6 +20812,12 @@ public Builder clear() { resources_ = null; resourcesBuilder_ = null; } + if (extendedResourcesBuilder_ == null) { + extendedResources_ = null; + } else { + extendedResources_ = null; + extendedResourcesBuilder_ = null; + } return this; } @@ -20750,6 +20849,11 @@ public flyteidl.core.Workflow.TaskNodeOverrides buildPartial() { } else { result.resources_ = resourcesBuilder_.build(); } + if (extendedResourcesBuilder_ == null) { + result.extendedResources_ = extendedResources_; + } else { + result.extendedResources_ = extendedResourcesBuilder_.build(); + } onBuilt(); return result; } @@ -20801,6 +20905,9 @@ public Builder mergeFrom(flyteidl.core.Workflow.TaskNodeOverrides other) { if (other.hasResources()) { mergeResources(other.getResources()); } + if (other.hasExtendedResources()) { + mergeExtendedResources(other.getExtendedResources()); + } this.mergeUnknownFields(other.unknownFields); onChanged(); return this; @@ -20982,6 +21089,168 @@ public flyteidl.core.Tasks.ResourcesOrBuilder getResourcesOrBuilder() { } return resourcesBuilder_; } + + private flyteidl.core.Tasks.ExtendedResources extendedResources_; + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder> extendedResourcesBuilder_; + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public boolean hasExtendedResources() { + return extendedResourcesBuilder_ != null || extendedResources_ != null; + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public flyteidl.core.Tasks.ExtendedResources getExtendedResources() { + if (extendedResourcesBuilder_ == null) { + return extendedResources_ == null ? flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } else { + return extendedResourcesBuilder_.getMessage(); + } + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public Builder setExtendedResources(flyteidl.core.Tasks.ExtendedResources value) { + if (extendedResourcesBuilder_ == null) { + if (value == null) { + throw new NullPointerException(); + } + extendedResources_ = value; + onChanged(); + } else { + extendedResourcesBuilder_.setMessage(value); + } + + return this; + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public Builder setExtendedResources( + flyteidl.core.Tasks.ExtendedResources.Builder builderForValue) { + if (extendedResourcesBuilder_ == null) { + extendedResources_ = builderForValue.build(); + onChanged(); + } else { + extendedResourcesBuilder_.setMessage(builderForValue.build()); + } + + return this; + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public Builder mergeExtendedResources(flyteidl.core.Tasks.ExtendedResources value) { + if (extendedResourcesBuilder_ == null) { + if (extendedResources_ != null) { + extendedResources_ = + flyteidl.core.Tasks.ExtendedResources.newBuilder(extendedResources_).mergeFrom(value).buildPartial(); + } else { + extendedResources_ = value; + } + onChanged(); + } else { + extendedResourcesBuilder_.mergeFrom(value); + } + + return this; + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public Builder clearExtendedResources() { + if (extendedResourcesBuilder_ == null) { + extendedResources_ = null; + onChanged(); + } else { + extendedResources_ = null; + extendedResourcesBuilder_ = null; + } + + return this; + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public flyteidl.core.Tasks.ExtendedResources.Builder getExtendedResourcesBuilder() { + + onChanged(); + return getExtendedResourcesFieldBuilder().getBuilder(); + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + public flyteidl.core.Tasks.ExtendedResourcesOrBuilder getExtendedResourcesOrBuilder() { + if (extendedResourcesBuilder_ != null) { + return extendedResourcesBuilder_.getMessageOrBuilder(); + } else { + return extendedResources_ == null ? + flyteidl.core.Tasks.ExtendedResources.getDefaultInstance() : extendedResources_; + } + } + /** + *
+       * Overrides for all non-standard resources, not captured by
+       * v1.ResourceRequirements, to allocate to a task.
+       * 
+ * + * .flyteidl.core.ExtendedResources extended_resources = 2; + */ + private com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder> + getExtendedResourcesFieldBuilder() { + if (extendedResourcesBuilder_ == null) { + extendedResourcesBuilder_ = new com.google.protobuf.SingleFieldBuilderV3< + flyteidl.core.Tasks.ExtendedResources, flyteidl.core.Tasks.ExtendedResources.Builder, flyteidl.core.Tasks.ExtendedResourcesOrBuilder>( + getExtendedResources(), + getParentForChildren(), + isClean()); + extendedResources_ = null; + } + return extendedResourcesBuilder_; + } @java.lang.Override public final Builder setUnknownFields( final com.google.protobuf.UnknownFieldSet unknownFields) { @@ -21205,10 +21474,11 @@ public flyteidl.core.Workflow.TaskNodeOverrides getDefaultInstanceForType() { "re.Binding\022)\n\014failure_node\030\006 \001(\0132\023.flyte" + "idl.core.Node\022B\n\021metadata_defaults\030\007 \001(\013" + "2\'.flyteidl.core.WorkflowMetadataDefault" + - "s\"@\n\021TaskNodeOverrides\022+\n\tresources\030\001 \001(" + - "\0132\030.flyteidl.core.ResourcesB>> 3) { + case 1: + message.device = reader.string(); + break; + case 2: + message.unpartitioned = reader.bool(); + break; + case 3: + message.partitionSize = reader.string(); + break; + default: + reader.skipType(tag & 7); + break; + } + } + return message; + }; + + /** + * Verifies a GPUAccelerator message. + * @function verify + * @memberof flyteidl.core.GPUAccelerator + * @static + * @param {Object.} message Plain object to verify + * @returns {string|null} `null` if valid, otherwise the reason why it is not + */ + GPUAccelerator.verify = function verify(message) { + if (typeof message !== "object" || message === null) + return "object expected"; + var properties = {}; + if (message.device != null && message.hasOwnProperty("device")) + if (!$util.isString(message.device)) + return "device: string expected"; + if (message.unpartitioned != null && message.hasOwnProperty("unpartitioned")) { + properties.partitionSizeValue = 1; + if (typeof message.unpartitioned !== "boolean") + return "unpartitioned: boolean expected"; + } + if (message.partitionSize != null && message.hasOwnProperty("partitionSize")) { + if (properties.partitionSizeValue === 1) + return "partitionSizeValue: multiple values"; + properties.partitionSizeValue = 1; + if (!$util.isString(message.partitionSize)) + return "partitionSize: string expected"; + } + return null; + }; + + return GPUAccelerator; + })(); + + core.ExtendedResources = (function() { + + /** + * Properties of an ExtendedResources. + * @memberof flyteidl.core + * @interface IExtendedResources + * @property {flyteidl.core.IGPUAccelerator|null} [gpuAccelerator] ExtendedResources gpuAccelerator + */ + + /** + * Constructs a new ExtendedResources. + * @memberof flyteidl.core + * @classdesc Represents an ExtendedResources. + * @implements IExtendedResources + * @constructor + * @param {flyteidl.core.IExtendedResources=} [properties] Properties to set + */ + function ExtendedResources(properties) { + if (properties) + for (var keys = Object.keys(properties), i = 0; i < keys.length; ++i) + if (properties[keys[i]] != null) + this[keys[i]] = properties[keys[i]]; + } + + /** + * ExtendedResources gpuAccelerator. + * @member {flyteidl.core.IGPUAccelerator|null|undefined} gpuAccelerator + * @memberof flyteidl.core.ExtendedResources + * @instance + */ + ExtendedResources.prototype.gpuAccelerator = null; + + /** + * Creates a new ExtendedResources instance using the specified properties. + * @function create + * @memberof flyteidl.core.ExtendedResources + * @static + * @param {flyteidl.core.IExtendedResources=} [properties] Properties to set + * @returns {flyteidl.core.ExtendedResources} ExtendedResources instance + */ + ExtendedResources.create = function create(properties) { + return new ExtendedResources(properties); + }; + + /** + * Encodes the specified ExtendedResources message. Does not implicitly {@link flyteidl.core.ExtendedResources.verify|verify} messages. + * @function encode + * @memberof flyteidl.core.ExtendedResources + * @static + * @param {flyteidl.core.IExtendedResources} message ExtendedResources message or plain object to encode + * @param {$protobuf.Writer} [writer] Writer to encode to + * @returns {$protobuf.Writer} Writer + */ + ExtendedResources.encode = function encode(message, writer) { + if (!writer) + writer = $Writer.create(); + if (message.gpuAccelerator != null && message.hasOwnProperty("gpuAccelerator")) + $root.flyteidl.core.GPUAccelerator.encode(message.gpuAccelerator, writer.uint32(/* id 1, wireType 2 =*/10).fork()).ldelim(); + return writer; + }; + + /** + * Decodes an ExtendedResources message from the specified reader or buffer. + * @function decode + * @memberof flyteidl.core.ExtendedResources + * @static + * @param {$protobuf.Reader|Uint8Array} reader Reader or buffer to decode from + * @param {number} [length] Message length if known beforehand + * @returns {flyteidl.core.ExtendedResources} ExtendedResources + * @throws {Error} If the payload is not a reader or valid buffer + * @throws {$protobuf.util.ProtocolError} If required fields are missing + */ + ExtendedResources.decode = function decode(reader, length) { + if (!(reader instanceof $Reader)) + reader = $Reader.create(reader); + var end = length === undefined ? reader.len : reader.pos + length, message = new $root.flyteidl.core.ExtendedResources(); + while (reader.pos < end) { + var tag = reader.uint32(); + switch (tag >>> 3) { + case 1: + message.gpuAccelerator = $root.flyteidl.core.GPUAccelerator.decode(reader, reader.uint32()); + break; + default: + reader.skipType(tag & 7); + break; + } + } + return message; + }; + + /** + * Verifies an ExtendedResources message. + * @function verify + * @memberof flyteidl.core.ExtendedResources + * @static + * @param {Object.} message Plain object to verify + * @returns {string|null} `null` if valid, otherwise the reason why it is not + */ + ExtendedResources.verify = function verify(message) { + if (typeof message !== "object" || message === null) + return "object expected"; + if (message.gpuAccelerator != null && message.hasOwnProperty("gpuAccelerator")) { + var error = $root.flyteidl.core.GPUAccelerator.verify(message.gpuAccelerator); + if (error) + return "gpuAccelerator." + error; + } + return null; + }; + + return ExtendedResources; + })(); + core.RuntimeMetadata = (function() { /** @@ -12770,6 +13066,7 @@ * @property {flyteidl.core.ISql|null} [sql] TaskTemplate sql * @property {number|null} [taskTypeVersion] TaskTemplate taskTypeVersion * @property {flyteidl.core.ISecurityContext|null} [securityContext] TaskTemplate securityContext + * @property {flyteidl.core.IExtendedResources|null} [extendedResources] TaskTemplate extendedResources * @property {Object.|null} [config] TaskTemplate config */ @@ -12869,6 +13166,14 @@ */ TaskTemplate.prototype.securityContext = null; + /** + * TaskTemplate extendedResources. + * @member {flyteidl.core.IExtendedResources|null|undefined} extendedResources + * @memberof flyteidl.core.TaskTemplate + * @instance + */ + TaskTemplate.prototype.extendedResources = null; + /** * TaskTemplate config. * @member {Object.} config @@ -12931,6 +13236,8 @@ writer.uint32(/* id 7, wireType 0 =*/56).int32(message.taskTypeVersion); if (message.securityContext != null && message.hasOwnProperty("securityContext")) $root.flyteidl.core.SecurityContext.encode(message.securityContext, writer.uint32(/* id 8, wireType 2 =*/66).fork()).ldelim(); + if (message.extendedResources != null && message.hasOwnProperty("extendedResources")) + $root.flyteidl.core.ExtendedResources.encode(message.extendedResources, writer.uint32(/* id 9, wireType 2 =*/74).fork()).ldelim(); if (message.config != null && message.hasOwnProperty("config")) for (var keys = Object.keys(message.config), i = 0; i < keys.length; ++i) writer.uint32(/* id 16, wireType 2 =*/130).fork().uint32(/* id 1, wireType 2 =*/10).string(keys[i]).uint32(/* id 2, wireType 2 =*/18).string(message.config[keys[i]]).ldelim(); @@ -12989,6 +13296,9 @@ case 8: message.securityContext = $root.flyteidl.core.SecurityContext.decode(reader, reader.uint32()); break; + case 9: + message.extendedResources = $root.flyteidl.core.ExtendedResources.decode(reader, reader.uint32()); + break; case 16: reader.skip().pos++; if (message.config === $util.emptyObject) @@ -13076,6 +13386,11 @@ if (error) return "securityContext." + error; } + if (message.extendedResources != null && message.hasOwnProperty("extendedResources")) { + var error = $root.flyteidl.core.ExtendedResources.verify(message.extendedResources); + if (error) + return "extendedResources." + error; + } if (message.config != null && message.hasOwnProperty("config")) { if (!$util.isObject(message.config)) return "config: object expected"; diff --git a/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.py b/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.py index 2452ead45e7..ba28b4716f1 100644 --- a/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.py +++ b/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.py @@ -19,7 +19,7 @@ from google.protobuf import struct_pb2 as google_dot_protobuf_dot_struct__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x19\x66lyteidl/core/tasks.proto\x12\rflyteidl.core\x1a\x1e\x66lyteidl/core/identifier.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x1c\x66lyteidl/core/security.proto\x1a\x1egoogle/protobuf/duration.proto\x1a\x1cgoogle/protobuf/struct.proto\"\xd0\x02\n\tResources\x12\x42\n\x08requests\x18\x01 \x03(\x0b\x32&.flyteidl.core.Resources.ResourceEntryR\x08requests\x12>\n\x06limits\x18\x02 \x03(\x0b\x32&.flyteidl.core.Resources.ResourceEntryR\x06limits\x1a`\n\rResourceEntry\x12\x39\n\x04name\x18\x01 \x01(\x0e\x32%.flyteidl.core.Resources.ResourceNameR\x04name\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value\"]\n\x0cResourceName\x12\x0b\n\x07UNKNOWN\x10\x00\x12\x07\n\x03\x43PU\x10\x01\x12\x07\n\x03GPU\x10\x02\x12\n\n\x06MEMORY\x10\x03\x12\x0b\n\x07STORAGE\x10\x04\x12\x15\n\x11\x45PHEMERAL_STORAGE\x10\x05\"\xac\x01\n\x0fRuntimeMetadata\x12>\n\x04type\x18\x01 \x01(\x0e\x32*.flyteidl.core.RuntimeMetadata.RuntimeTypeR\x04type\x12\x18\n\x07version\x18\x02 \x01(\tR\x07version\x12\x16\n\x06\x66lavor\x18\x03 \x01(\tR\x06\x66lavor\"\'\n\x0bRuntimeType\x12\t\n\x05OTHER\x10\x00\x12\r\n\tFLYTE_SDK\x10\x01\"\xf5\x04\n\x0cTaskMetadata\x12\"\n\x0c\x64iscoverable\x18\x01 \x01(\x08R\x0c\x64iscoverable\x12\x38\n\x07runtime\x18\x02 \x01(\x0b\x32\x1e.flyteidl.core.RuntimeMetadataR\x07runtime\x12\x33\n\x07timeout\x18\x04 \x01(\x0b\x32\x19.google.protobuf.DurationR\x07timeout\x12\x36\n\x07retries\x18\x05 \x01(\x0b\x32\x1c.flyteidl.core.RetryStrategyR\x07retries\x12+\n\x11\x64iscovery_version\x18\x06 \x01(\tR\x10\x64iscoveryVersion\x12\x38\n\x18\x64\x65precated_error_message\x18\x07 \x01(\tR\x16\x64\x65precatedErrorMessage\x12&\n\rinterruptible\x18\x08 \x01(\x08H\x00R\rinterruptible\x12-\n\x12\x63\x61\x63he_serializable\x18\t \x01(\x08R\x11\x63\x61\x63heSerializable\x12%\n\x0egenerates_deck\x18\n \x01(\x08R\rgeneratesDeck\x12\x39\n\x04tags\x18\x0b \x03(\x0b\x32%.flyteidl.core.TaskMetadata.TagsEntryR\x04tags\x12*\n\x11pod_template_name\x18\x0c \x01(\tR\x0fpodTemplateName\x1a\x37\n\tTagsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x42\x15\n\x13interruptible_value\"\x85\x05\n\x0cTaskTemplate\x12)\n\x02id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierR\x02id\x12\x12\n\x04type\x18\x02 \x01(\tR\x04type\x12\x37\n\x08metadata\x18\x03 \x01(\x0b\x32\x1b.flyteidl.core.TaskMetadataR\x08metadata\x12;\n\tinterface\x18\x04 \x01(\x0b\x32\x1d.flyteidl.core.TypedInterfaceR\tinterface\x12/\n\x06\x63ustom\x18\x05 \x01(\x0b\x32\x17.google.protobuf.StructR\x06\x63ustom\x12\x38\n\tcontainer\x18\x06 \x01(\x0b\x32\x18.flyteidl.core.ContainerH\x00R\tcontainer\x12\x30\n\x07k8s_pod\x18\x11 \x01(\x0b\x32\x15.flyteidl.core.K8sPodH\x00R\x06k8sPod\x12&\n\x03sql\x18\x12 \x01(\x0b\x32\x12.flyteidl.core.SqlH\x00R\x03sql\x12*\n\x11task_type_version\x18\x07 \x01(\x05R\x0ftaskTypeVersion\x12I\n\x10security_context\x18\x08 \x01(\x0b\x32\x1e.flyteidl.core.SecurityContextR\x0fsecurityContext\x12?\n\x06\x63onfig\x18\x10 \x03(\x0b\x32\'.flyteidl.core.TaskTemplate.ConfigEntryR\x06\x63onfig\x1a\x39\n\x0b\x43onfigEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x42\x08\n\x06target\"6\n\rContainerPort\x12%\n\x0e\x63ontainer_port\x18\x01 \x01(\rR\rcontainerPort\"\xfc\x03\n\tContainer\x12\x14\n\x05image\x18\x01 \x01(\tR\x05image\x12\x18\n\x07\x63ommand\x18\x02 \x03(\tR\x07\x63ommand\x12\x12\n\x04\x61rgs\x18\x03 \x03(\tR\x04\x61rgs\x12\x36\n\tresources\x18\x04 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresources\x12-\n\x03\x65nv\x18\x05 \x03(\x0b\x32\x1b.flyteidl.core.KeyValuePairR\x03\x65nv\x12\x37\n\x06\x63onfig\x18\x06 \x03(\x0b\x32\x1b.flyteidl.core.KeyValuePairB\x02\x18\x01R\x06\x63onfig\x12\x32\n\x05ports\x18\x07 \x03(\x0b\x32\x1c.flyteidl.core.ContainerPortR\x05ports\x12\x41\n\x0b\x64\x61ta_config\x18\t \x01(\x0b\x32 .flyteidl.core.DataLoadingConfigR\ndataConfig\x12I\n\x0c\x61rchitecture\x18\n \x01(\x0e\x32%.flyteidl.core.Container.ArchitectureR\x0c\x61rchitecture\"I\n\x0c\x41rchitecture\x12\x0b\n\x07UNKNOWN\x10\x00\x12\t\n\x05\x41MD64\x10\x01\x12\t\n\x05\x41RM64\x10\x02\x12\n\n\x06\x41RM_V6\x10\x03\x12\n\n\x06\x41RM_V7\x10\x04\"\xb5\x02\n\nIOStrategy\x12K\n\rdownload_mode\x18\x01 \x01(\x0e\x32&.flyteidl.core.IOStrategy.DownloadModeR\x0c\x64ownloadMode\x12\x45\n\x0bupload_mode\x18\x02 \x01(\x0e\x32$.flyteidl.core.IOStrategy.UploadModeR\nuploadMode\"L\n\x0c\x44ownloadMode\x12\x12\n\x0e\x44OWNLOAD_EAGER\x10\x00\x12\x13\n\x0f\x44OWNLOAD_STREAM\x10\x01\x12\x13\n\x0f\x44O_NOT_DOWNLOAD\x10\x02\"E\n\nUploadMode\x12\x12\n\x0eUPLOAD_ON_EXIT\x10\x00\x12\x10\n\x0cUPLOAD_EAGER\x10\x01\x12\x11\n\rDO_NOT_UPLOAD\x10\x02\"\xa7\x02\n\x11\x44\x61taLoadingConfig\x12\x18\n\x07\x65nabled\x18\x01 \x01(\x08R\x07\x65nabled\x12\x1d\n\ninput_path\x18\x02 \x01(\tR\tinputPath\x12\x1f\n\x0boutput_path\x18\x03 \x01(\tR\noutputPath\x12I\n\x06\x66ormat\x18\x04 \x01(\x0e\x32\x31.flyteidl.core.DataLoadingConfig.LiteralMapFormatR\x06\x66ormat\x12:\n\x0bio_strategy\x18\x05 \x01(\x0b\x32\x19.flyteidl.core.IOStrategyR\nioStrategy\"1\n\x10LiteralMapFormat\x12\x08\n\x04JSON\x10\x00\x12\x08\n\x04YAML\x10\x01\x12\t\n\x05PROTO\x10\x02\"\xbd\x01\n\x06K8sPod\x12<\n\x08metadata\x18\x01 \x01(\x0b\x32 .flyteidl.core.K8sObjectMetadataR\x08metadata\x12\x32\n\x08pod_spec\x18\x02 \x01(\x0b\x32\x17.google.protobuf.StructR\x07podSpec\x12\x41\n\x0b\x64\x61ta_config\x18\x03 \x01(\x0b\x32 .flyteidl.core.DataLoadingConfigR\ndataConfig\"\xa9\x02\n\x11K8sObjectMetadata\x12\x44\n\x06labels\x18\x01 \x03(\x0b\x32,.flyteidl.core.K8sObjectMetadata.LabelsEntryR\x06labels\x12S\n\x0b\x61nnotations\x18\x02 \x03(\x0b\x32\x31.flyteidl.core.K8sObjectMetadata.AnnotationsEntryR\x0b\x61nnotations\x1a\x39\n\x0bLabelsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1a>\n\x10\x41nnotationsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"\x92\x01\n\x03Sql\x12\x1c\n\tstatement\x18\x01 \x01(\tR\tstatement\x12\x34\n\x07\x64ialect\x18\x02 \x01(\x0e\x32\x1a.flyteidl.core.Sql.DialectR\x07\x64ialect\"7\n\x07\x44ialect\x12\r\n\tUNDEFINED\x10\x00\x12\x08\n\x04\x41NSI\x10\x01\x12\x08\n\x04HIVE\x10\x02\x12\t\n\x05OTHER\x10\x03\x42\xb0\x01\n\x11\x63om.flyteidl.coreB\nTasksProtoP\x01Z:github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core\xa2\x02\x03\x46\x43X\xaa\x02\rFlyteidl.Core\xca\x02\rFlyteidl\\Core\xe2\x02\x19\x46lyteidl\\Core\\GPBMetadata\xea\x02\x0e\x46lyteidl::Coreb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x19\x66lyteidl/core/tasks.proto\x12\rflyteidl.core\x1a\x1e\x66lyteidl/core/identifier.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x1c\x66lyteidl/core/security.proto\x1a\x1egoogle/protobuf/duration.proto\x1a\x1cgoogle/protobuf/struct.proto\"\xd0\x02\n\tResources\x12\x42\n\x08requests\x18\x01 \x03(\x0b\x32&.flyteidl.core.Resources.ResourceEntryR\x08requests\x12>\n\x06limits\x18\x02 \x03(\x0b\x32&.flyteidl.core.Resources.ResourceEntryR\x06limits\x1a`\n\rResourceEntry\x12\x39\n\x04name\x18\x01 \x01(\x0e\x32%.flyteidl.core.Resources.ResourceNameR\x04name\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value\"]\n\x0cResourceName\x12\x0b\n\x07UNKNOWN\x10\x00\x12\x07\n\x03\x43PU\x10\x01\x12\x07\n\x03GPU\x10\x02\x12\n\n\x06MEMORY\x10\x03\x12\x0b\n\x07STORAGE\x10\x04\x12\x15\n\x11\x45PHEMERAL_STORAGE\x10\x05\"\x91\x01\n\x0eGPUAccelerator\x12\x16\n\x06\x64\x65vice\x18\x01 \x01(\tR\x06\x64\x65vice\x12&\n\runpartitioned\x18\x02 \x01(\x08H\x00R\runpartitioned\x12\'\n\x0epartition_size\x18\x03 \x01(\tH\x00R\rpartitionSizeB\x16\n\x14partition_size_value\"[\n\x11\x45xtendedResources\x12\x46\n\x0fgpu_accelerator\x18\x01 \x01(\x0b\x32\x1d.flyteidl.core.GPUAcceleratorR\x0egpuAccelerator\"\xac\x01\n\x0fRuntimeMetadata\x12>\n\x04type\x18\x01 \x01(\x0e\x32*.flyteidl.core.RuntimeMetadata.RuntimeTypeR\x04type\x12\x18\n\x07version\x18\x02 \x01(\tR\x07version\x12\x16\n\x06\x66lavor\x18\x03 \x01(\tR\x06\x66lavor\"\'\n\x0bRuntimeType\x12\t\n\x05OTHER\x10\x00\x12\r\n\tFLYTE_SDK\x10\x01\"\xf5\x04\n\x0cTaskMetadata\x12\"\n\x0c\x64iscoverable\x18\x01 \x01(\x08R\x0c\x64iscoverable\x12\x38\n\x07runtime\x18\x02 \x01(\x0b\x32\x1e.flyteidl.core.RuntimeMetadataR\x07runtime\x12\x33\n\x07timeout\x18\x04 \x01(\x0b\x32\x19.google.protobuf.DurationR\x07timeout\x12\x36\n\x07retries\x18\x05 \x01(\x0b\x32\x1c.flyteidl.core.RetryStrategyR\x07retries\x12+\n\x11\x64iscovery_version\x18\x06 \x01(\tR\x10\x64iscoveryVersion\x12\x38\n\x18\x64\x65precated_error_message\x18\x07 \x01(\tR\x16\x64\x65precatedErrorMessage\x12&\n\rinterruptible\x18\x08 \x01(\x08H\x00R\rinterruptible\x12-\n\x12\x63\x61\x63he_serializable\x18\t \x01(\x08R\x11\x63\x61\x63heSerializable\x12%\n\x0egenerates_deck\x18\n \x01(\x08R\rgeneratesDeck\x12\x39\n\x04tags\x18\x0b \x03(\x0b\x32%.flyteidl.core.TaskMetadata.TagsEntryR\x04tags\x12*\n\x11pod_template_name\x18\x0c \x01(\tR\x0fpodTemplateName\x1a\x37\n\tTagsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x42\x15\n\x13interruptible_value\"\xd6\x05\n\x0cTaskTemplate\x12)\n\x02id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierR\x02id\x12\x12\n\x04type\x18\x02 \x01(\tR\x04type\x12\x37\n\x08metadata\x18\x03 \x01(\x0b\x32\x1b.flyteidl.core.TaskMetadataR\x08metadata\x12;\n\tinterface\x18\x04 \x01(\x0b\x32\x1d.flyteidl.core.TypedInterfaceR\tinterface\x12/\n\x06\x63ustom\x18\x05 \x01(\x0b\x32\x17.google.protobuf.StructR\x06\x63ustom\x12\x38\n\tcontainer\x18\x06 \x01(\x0b\x32\x18.flyteidl.core.ContainerH\x00R\tcontainer\x12\x30\n\x07k8s_pod\x18\x11 \x01(\x0b\x32\x15.flyteidl.core.K8sPodH\x00R\x06k8sPod\x12&\n\x03sql\x18\x12 \x01(\x0b\x32\x12.flyteidl.core.SqlH\x00R\x03sql\x12*\n\x11task_type_version\x18\x07 \x01(\x05R\x0ftaskTypeVersion\x12I\n\x10security_context\x18\x08 \x01(\x0b\x32\x1e.flyteidl.core.SecurityContextR\x0fsecurityContext\x12O\n\x12\x65xtended_resources\x18\t \x01(\x0b\x32 .flyteidl.core.ExtendedResourcesR\x11\x65xtendedResources\x12?\n\x06\x63onfig\x18\x10 \x03(\x0b\x32\'.flyteidl.core.TaskTemplate.ConfigEntryR\x06\x63onfig\x1a\x39\n\x0b\x43onfigEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x42\x08\n\x06target\"6\n\rContainerPort\x12%\n\x0e\x63ontainer_port\x18\x01 \x01(\rR\rcontainerPort\"\xfc\x03\n\tContainer\x12\x14\n\x05image\x18\x01 \x01(\tR\x05image\x12\x18\n\x07\x63ommand\x18\x02 \x03(\tR\x07\x63ommand\x12\x12\n\x04\x61rgs\x18\x03 \x03(\tR\x04\x61rgs\x12\x36\n\tresources\x18\x04 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresources\x12-\n\x03\x65nv\x18\x05 \x03(\x0b\x32\x1b.flyteidl.core.KeyValuePairR\x03\x65nv\x12\x37\n\x06\x63onfig\x18\x06 \x03(\x0b\x32\x1b.flyteidl.core.KeyValuePairB\x02\x18\x01R\x06\x63onfig\x12\x32\n\x05ports\x18\x07 \x03(\x0b\x32\x1c.flyteidl.core.ContainerPortR\x05ports\x12\x41\n\x0b\x64\x61ta_config\x18\t \x01(\x0b\x32 .flyteidl.core.DataLoadingConfigR\ndataConfig\x12I\n\x0c\x61rchitecture\x18\n \x01(\x0e\x32%.flyteidl.core.Container.ArchitectureR\x0c\x61rchitecture\"I\n\x0c\x41rchitecture\x12\x0b\n\x07UNKNOWN\x10\x00\x12\t\n\x05\x41MD64\x10\x01\x12\t\n\x05\x41RM64\x10\x02\x12\n\n\x06\x41RM_V6\x10\x03\x12\n\n\x06\x41RM_V7\x10\x04\"\xb5\x02\n\nIOStrategy\x12K\n\rdownload_mode\x18\x01 \x01(\x0e\x32&.flyteidl.core.IOStrategy.DownloadModeR\x0c\x64ownloadMode\x12\x45\n\x0bupload_mode\x18\x02 \x01(\x0e\x32$.flyteidl.core.IOStrategy.UploadModeR\nuploadMode\"L\n\x0c\x44ownloadMode\x12\x12\n\x0e\x44OWNLOAD_EAGER\x10\x00\x12\x13\n\x0f\x44OWNLOAD_STREAM\x10\x01\x12\x13\n\x0f\x44O_NOT_DOWNLOAD\x10\x02\"E\n\nUploadMode\x12\x12\n\x0eUPLOAD_ON_EXIT\x10\x00\x12\x10\n\x0cUPLOAD_EAGER\x10\x01\x12\x11\n\rDO_NOT_UPLOAD\x10\x02\"\xa7\x02\n\x11\x44\x61taLoadingConfig\x12\x18\n\x07\x65nabled\x18\x01 \x01(\x08R\x07\x65nabled\x12\x1d\n\ninput_path\x18\x02 \x01(\tR\tinputPath\x12\x1f\n\x0boutput_path\x18\x03 \x01(\tR\noutputPath\x12I\n\x06\x66ormat\x18\x04 \x01(\x0e\x32\x31.flyteidl.core.DataLoadingConfig.LiteralMapFormatR\x06\x66ormat\x12:\n\x0bio_strategy\x18\x05 \x01(\x0b\x32\x19.flyteidl.core.IOStrategyR\nioStrategy\"1\n\x10LiteralMapFormat\x12\x08\n\x04JSON\x10\x00\x12\x08\n\x04YAML\x10\x01\x12\t\n\x05PROTO\x10\x02\"\xbd\x01\n\x06K8sPod\x12<\n\x08metadata\x18\x01 \x01(\x0b\x32 .flyteidl.core.K8sObjectMetadataR\x08metadata\x12\x32\n\x08pod_spec\x18\x02 \x01(\x0b\x32\x17.google.protobuf.StructR\x07podSpec\x12\x41\n\x0b\x64\x61ta_config\x18\x03 \x01(\x0b\x32 .flyteidl.core.DataLoadingConfigR\ndataConfig\"\xa9\x02\n\x11K8sObjectMetadata\x12\x44\n\x06labels\x18\x01 \x03(\x0b\x32,.flyteidl.core.K8sObjectMetadata.LabelsEntryR\x06labels\x12S\n\x0b\x61nnotations\x18\x02 \x03(\x0b\x32\x31.flyteidl.core.K8sObjectMetadata.AnnotationsEntryR\x0b\x61nnotations\x1a\x39\n\x0bLabelsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\x1a>\n\x10\x41nnotationsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"\x92\x01\n\x03Sql\x12\x1c\n\tstatement\x18\x01 \x01(\tR\tstatement\x12\x34\n\x07\x64ialect\x18\x02 \x01(\x0e\x32\x1a.flyteidl.core.Sql.DialectR\x07\x64ialect\"7\n\x07\x44ialect\x12\r\n\tUNDEFINED\x10\x00\x12\x08\n\x04\x41NSI\x10\x01\x12\x08\n\x04HIVE\x10\x02\x12\t\n\x05OTHER\x10\x03\x42\xb0\x01\n\x11\x63om.flyteidl.coreB\nTasksProtoP\x01Z:github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core\xa2\x02\x03\x46\x43X\xaa\x02\rFlyteidl.Core\xca\x02\rFlyteidl\\Core\xe2\x02\x19\x46lyteidl\\Core\\GPBMetadata\xea\x02\x0e\x46lyteidl::Coreb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -44,44 +44,48 @@ _globals['_RESOURCES_RESOURCEENTRY']._serialized_end=471 _globals['_RESOURCES_RESOURCENAME']._serialized_start=473 _globals['_RESOURCES_RESOURCENAME']._serialized_end=566 - _globals['_RUNTIMEMETADATA']._serialized_start=569 - _globals['_RUNTIMEMETADATA']._serialized_end=741 - _globals['_RUNTIMEMETADATA_RUNTIMETYPE']._serialized_start=702 - _globals['_RUNTIMEMETADATA_RUNTIMETYPE']._serialized_end=741 - _globals['_TASKMETADATA']._serialized_start=744 - _globals['_TASKMETADATA']._serialized_end=1373 - _globals['_TASKMETADATA_TAGSENTRY']._serialized_start=1295 - _globals['_TASKMETADATA_TAGSENTRY']._serialized_end=1350 - _globals['_TASKTEMPLATE']._serialized_start=1376 - _globals['_TASKTEMPLATE']._serialized_end=2021 - _globals['_TASKTEMPLATE_CONFIGENTRY']._serialized_start=1954 - _globals['_TASKTEMPLATE_CONFIGENTRY']._serialized_end=2011 - _globals['_CONTAINERPORT']._serialized_start=2023 - _globals['_CONTAINERPORT']._serialized_end=2077 - _globals['_CONTAINER']._serialized_start=2080 - _globals['_CONTAINER']._serialized_end=2588 - _globals['_CONTAINER_ARCHITECTURE']._serialized_start=2515 - _globals['_CONTAINER_ARCHITECTURE']._serialized_end=2588 - _globals['_IOSTRATEGY']._serialized_start=2591 - _globals['_IOSTRATEGY']._serialized_end=2900 - _globals['_IOSTRATEGY_DOWNLOADMODE']._serialized_start=2753 - _globals['_IOSTRATEGY_DOWNLOADMODE']._serialized_end=2829 - _globals['_IOSTRATEGY_UPLOADMODE']._serialized_start=2831 - _globals['_IOSTRATEGY_UPLOADMODE']._serialized_end=2900 - _globals['_DATALOADINGCONFIG']._serialized_start=2903 - _globals['_DATALOADINGCONFIG']._serialized_end=3198 - _globals['_DATALOADINGCONFIG_LITERALMAPFORMAT']._serialized_start=3149 - _globals['_DATALOADINGCONFIG_LITERALMAPFORMAT']._serialized_end=3198 - _globals['_K8SPOD']._serialized_start=3201 - _globals['_K8SPOD']._serialized_end=3390 - _globals['_K8SOBJECTMETADATA']._serialized_start=3393 - _globals['_K8SOBJECTMETADATA']._serialized_end=3690 - _globals['_K8SOBJECTMETADATA_LABELSENTRY']._serialized_start=3569 - _globals['_K8SOBJECTMETADATA_LABELSENTRY']._serialized_end=3626 - _globals['_K8SOBJECTMETADATA_ANNOTATIONSENTRY']._serialized_start=3628 - _globals['_K8SOBJECTMETADATA_ANNOTATIONSENTRY']._serialized_end=3690 - _globals['_SQL']._serialized_start=3693 - _globals['_SQL']._serialized_end=3839 - _globals['_SQL_DIALECT']._serialized_start=3784 - _globals['_SQL_DIALECT']._serialized_end=3839 + _globals['_GPUACCELERATOR']._serialized_start=569 + _globals['_GPUACCELERATOR']._serialized_end=714 + _globals['_EXTENDEDRESOURCES']._serialized_start=716 + _globals['_EXTENDEDRESOURCES']._serialized_end=807 + _globals['_RUNTIMEMETADATA']._serialized_start=810 + _globals['_RUNTIMEMETADATA']._serialized_end=982 + _globals['_RUNTIMEMETADATA_RUNTIMETYPE']._serialized_start=943 + _globals['_RUNTIMEMETADATA_RUNTIMETYPE']._serialized_end=982 + _globals['_TASKMETADATA']._serialized_start=985 + _globals['_TASKMETADATA']._serialized_end=1614 + _globals['_TASKMETADATA_TAGSENTRY']._serialized_start=1536 + _globals['_TASKMETADATA_TAGSENTRY']._serialized_end=1591 + _globals['_TASKTEMPLATE']._serialized_start=1617 + _globals['_TASKTEMPLATE']._serialized_end=2343 + _globals['_TASKTEMPLATE_CONFIGENTRY']._serialized_start=2276 + _globals['_TASKTEMPLATE_CONFIGENTRY']._serialized_end=2333 + _globals['_CONTAINERPORT']._serialized_start=2345 + _globals['_CONTAINERPORT']._serialized_end=2399 + _globals['_CONTAINER']._serialized_start=2402 + _globals['_CONTAINER']._serialized_end=2910 + _globals['_CONTAINER_ARCHITECTURE']._serialized_start=2837 + _globals['_CONTAINER_ARCHITECTURE']._serialized_end=2910 + _globals['_IOSTRATEGY']._serialized_start=2913 + _globals['_IOSTRATEGY']._serialized_end=3222 + _globals['_IOSTRATEGY_DOWNLOADMODE']._serialized_start=3075 + _globals['_IOSTRATEGY_DOWNLOADMODE']._serialized_end=3151 + _globals['_IOSTRATEGY_UPLOADMODE']._serialized_start=3153 + _globals['_IOSTRATEGY_UPLOADMODE']._serialized_end=3222 + _globals['_DATALOADINGCONFIG']._serialized_start=3225 + _globals['_DATALOADINGCONFIG']._serialized_end=3520 + _globals['_DATALOADINGCONFIG_LITERALMAPFORMAT']._serialized_start=3471 + _globals['_DATALOADINGCONFIG_LITERALMAPFORMAT']._serialized_end=3520 + _globals['_K8SPOD']._serialized_start=3523 + _globals['_K8SPOD']._serialized_end=3712 + _globals['_K8SOBJECTMETADATA']._serialized_start=3715 + _globals['_K8SOBJECTMETADATA']._serialized_end=4012 + _globals['_K8SOBJECTMETADATA_LABELSENTRY']._serialized_start=3891 + _globals['_K8SOBJECTMETADATA_LABELSENTRY']._serialized_end=3948 + _globals['_K8SOBJECTMETADATA_ANNOTATIONSENTRY']._serialized_start=3950 + _globals['_K8SOBJECTMETADATA_ANNOTATIONSENTRY']._serialized_end=4012 + _globals['_SQL']._serialized_start=4015 + _globals['_SQL']._serialized_end=4161 + _globals['_SQL_DIALECT']._serialized_start=4106 + _globals['_SQL_DIALECT']._serialized_end=4161 # @@protoc_insertion_point(module_scope) diff --git a/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.pyi b/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.pyi index e33453ecdeb..05452394439 100644 --- a/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.pyi +++ b/flyteidl/gen/pb_python/flyteidl/core/tasks_pb2.pyi @@ -41,6 +41,22 @@ class Resources(_message.Message): limits: _containers.RepeatedCompositeFieldContainer[Resources.ResourceEntry] def __init__(self, requests: _Optional[_Iterable[_Union[Resources.ResourceEntry, _Mapping]]] = ..., limits: _Optional[_Iterable[_Union[Resources.ResourceEntry, _Mapping]]] = ...) -> None: ... +class GPUAccelerator(_message.Message): + __slots__ = ["device", "unpartitioned", "partition_size"] + DEVICE_FIELD_NUMBER: _ClassVar[int] + UNPARTITIONED_FIELD_NUMBER: _ClassVar[int] + PARTITION_SIZE_FIELD_NUMBER: _ClassVar[int] + device: str + unpartitioned: bool + partition_size: str + def __init__(self, device: _Optional[str] = ..., unpartitioned: bool = ..., partition_size: _Optional[str] = ...) -> None: ... + +class ExtendedResources(_message.Message): + __slots__ = ["gpu_accelerator"] + GPU_ACCELERATOR_FIELD_NUMBER: _ClassVar[int] + gpu_accelerator: GPUAccelerator + def __init__(self, gpu_accelerator: _Optional[_Union[GPUAccelerator, _Mapping]] = ...) -> None: ... + class RuntimeMetadata(_message.Message): __slots__ = ["type", "version", "flavor"] class RuntimeType(int, metaclass=_enum_type_wrapper.EnumTypeWrapper): @@ -91,7 +107,7 @@ class TaskMetadata(_message.Message): def __init__(self, discoverable: bool = ..., runtime: _Optional[_Union[RuntimeMetadata, _Mapping]] = ..., timeout: _Optional[_Union[_duration_pb2.Duration, _Mapping]] = ..., retries: _Optional[_Union[_literals_pb2.RetryStrategy, _Mapping]] = ..., discovery_version: _Optional[str] = ..., deprecated_error_message: _Optional[str] = ..., interruptible: bool = ..., cache_serializable: bool = ..., generates_deck: bool = ..., tags: _Optional[_Mapping[str, str]] = ..., pod_template_name: _Optional[str] = ...) -> None: ... class TaskTemplate(_message.Message): - __slots__ = ["id", "type", "metadata", "interface", "custom", "container", "k8s_pod", "sql", "task_type_version", "security_context", "config"] + __slots__ = ["id", "type", "metadata", "interface", "custom", "container", "k8s_pod", "sql", "task_type_version", "security_context", "extended_resources", "config"] class ConfigEntry(_message.Message): __slots__ = ["key", "value"] KEY_FIELD_NUMBER: _ClassVar[int] @@ -109,6 +125,7 @@ class TaskTemplate(_message.Message): SQL_FIELD_NUMBER: _ClassVar[int] TASK_TYPE_VERSION_FIELD_NUMBER: _ClassVar[int] SECURITY_CONTEXT_FIELD_NUMBER: _ClassVar[int] + EXTENDED_RESOURCES_FIELD_NUMBER: _ClassVar[int] CONFIG_FIELD_NUMBER: _ClassVar[int] id: _identifier_pb2.Identifier type: str @@ -120,8 +137,9 @@ class TaskTemplate(_message.Message): sql: Sql task_type_version: int security_context: _security_pb2.SecurityContext + extended_resources: ExtendedResources config: _containers.ScalarMap[str, str] - def __init__(self, id: _Optional[_Union[_identifier_pb2.Identifier, _Mapping]] = ..., type: _Optional[str] = ..., metadata: _Optional[_Union[TaskMetadata, _Mapping]] = ..., interface: _Optional[_Union[_interface_pb2.TypedInterface, _Mapping]] = ..., custom: _Optional[_Union[_struct_pb2.Struct, _Mapping]] = ..., container: _Optional[_Union[Container, _Mapping]] = ..., k8s_pod: _Optional[_Union[K8sPod, _Mapping]] = ..., sql: _Optional[_Union[Sql, _Mapping]] = ..., task_type_version: _Optional[int] = ..., security_context: _Optional[_Union[_security_pb2.SecurityContext, _Mapping]] = ..., config: _Optional[_Mapping[str, str]] = ...) -> None: ... + def __init__(self, id: _Optional[_Union[_identifier_pb2.Identifier, _Mapping]] = ..., type: _Optional[str] = ..., metadata: _Optional[_Union[TaskMetadata, _Mapping]] = ..., interface: _Optional[_Union[_interface_pb2.TypedInterface, _Mapping]] = ..., custom: _Optional[_Union[_struct_pb2.Struct, _Mapping]] = ..., container: _Optional[_Union[Container, _Mapping]] = ..., k8s_pod: _Optional[_Union[K8sPod, _Mapping]] = ..., sql: _Optional[_Union[Sql, _Mapping]] = ..., task_type_version: _Optional[int] = ..., security_context: _Optional[_Union[_security_pb2.SecurityContext, _Mapping]] = ..., extended_resources: _Optional[_Union[ExtendedResources, _Mapping]] = ..., config: _Optional[_Mapping[str, str]] = ...) -> None: ... class ContainerPort(_message.Message): __slots__ = ["container_port"] diff --git a/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.py b/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.py index df7570e9b36..8487a5170e3 100644 --- a/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.py +++ b/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.py @@ -22,7 +22,7 @@ from google.protobuf import duration_pb2 as google_dot_protobuf_dot_duration__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1c\x66lyteidl/core/workflow.proto\x12\rflyteidl.core\x1a\x1d\x66lyteidl/core/condition.proto\x1a\x1d\x66lyteidl/core/execution.proto\x1a\x1e\x66lyteidl/core/identifier.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x19\x66lyteidl/core/tasks.proto\x1a\x19\x66lyteidl/core/types.proto\x1a\x1c\x66lyteidl/core/security.proto\x1a\x1egoogle/protobuf/duration.proto\"{\n\x07IfBlock\x12>\n\tcondition\x18\x01 \x01(\x0b\x32 .flyteidl.core.BooleanExpressionR\tcondition\x12\x30\n\tthen_node\x18\x02 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x08thenNode\"\xd4\x01\n\x0bIfElseBlock\x12*\n\x04\x63\x61se\x18\x01 \x01(\x0b\x32\x16.flyteidl.core.IfBlockR\x04\x63\x61se\x12,\n\x05other\x18\x02 \x03(\x0b\x32\x16.flyteidl.core.IfBlockR\x05other\x12\x32\n\telse_node\x18\x03 \x01(\x0b\x32\x13.flyteidl.core.NodeH\x00R\x08\x65lseNode\x12,\n\x05\x65rror\x18\x04 \x01(\x0b\x32\x14.flyteidl.core.ErrorH\x00R\x05\x65rrorB\t\n\x07\x64\x65\x66\x61ult\"A\n\nBranchNode\x12\x33\n\x07if_else\x18\x01 \x01(\x0b\x32\x1a.flyteidl.core.IfElseBlockR\x06ifElse\"\x97\x01\n\x08TaskNode\x12>\n\x0creference_id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\x0breferenceId\x12>\n\toverrides\x18\x02 \x01(\x0b\x32 .flyteidl.core.TaskNodeOverridesR\toverridesB\x0b\n\treference\"\xa6\x01\n\x0cWorkflowNode\x12\x42\n\x0elaunchplan_ref\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\rlaunchplanRef\x12\x45\n\x10sub_workflow_ref\x18\x02 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\x0esubWorkflowRefB\x0b\n\treference\"/\n\x10\x41pproveCondition\x12\x1b\n\tsignal_id\x18\x01 \x01(\tR\x08signalId\"\x90\x01\n\x0fSignalCondition\x12\x1b\n\tsignal_id\x18\x01 \x01(\tR\x08signalId\x12.\n\x04type\x18\x02 \x01(\x0b\x32\x1a.flyteidl.core.LiteralTypeR\x04type\x12\x30\n\x14output_variable_name\x18\x03 \x01(\tR\x12outputVariableName\"G\n\x0eSleepCondition\x12\x35\n\x08\x64uration\x18\x01 \x01(\x0b\x32\x19.google.protobuf.DurationR\x08\x64uration\"\xc5\x01\n\x08GateNode\x12;\n\x07\x61pprove\x18\x01 \x01(\x0b\x32\x1f.flyteidl.core.ApproveConditionH\x00R\x07\x61pprove\x12\x38\n\x06signal\x18\x02 \x01(\x0b\x32\x1e.flyteidl.core.SignalConditionH\x00R\x06signal\x12\x35\n\x05sleep\x18\x03 \x01(\x0b\x32\x1d.flyteidl.core.SleepConditionH\x00R\x05sleepB\x0b\n\tcondition\"\xbf\x01\n\tArrayNode\x12\'\n\x04node\x18\x01 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x04node\x12 \n\x0bparallelism\x18\x02 \x01(\rR\x0bparallelism\x12%\n\rmin_successes\x18\x03 \x01(\rH\x00R\x0cminSuccesses\x12,\n\x11min_success_ratio\x18\x04 \x01(\x02H\x00R\x0fminSuccessRatioB\x12\n\x10success_criteria\"\xce\x01\n\x0cNodeMetadata\x12\x12\n\x04name\x18\x01 \x01(\tR\x04name\x12\x33\n\x07timeout\x18\x04 \x01(\x0b\x32\x19.google.protobuf.DurationR\x07timeout\x12\x36\n\x07retries\x18\x05 \x01(\x0b\x32\x1c.flyteidl.core.RetryStrategyR\x07retries\x12&\n\rinterruptible\x18\x06 \x01(\x08H\x00R\rinterruptibleB\x15\n\x13interruptible_value\"/\n\x05\x41lias\x12\x10\n\x03var\x18\x01 \x01(\tR\x03var\x12\x14\n\x05\x61lias\x18\x02 \x01(\tR\x05\x61lias\"\x9f\x04\n\x04Node\x12\x0e\n\x02id\x18\x01 \x01(\tR\x02id\x12\x37\n\x08metadata\x18\x02 \x01(\x0b\x32\x1b.flyteidl.core.NodeMetadataR\x08metadata\x12.\n\x06inputs\x18\x03 \x03(\x0b\x32\x16.flyteidl.core.BindingR\x06inputs\x12*\n\x11upstream_node_ids\x18\x04 \x03(\tR\x0fupstreamNodeIds\x12;\n\x0eoutput_aliases\x18\x05 \x03(\x0b\x32\x14.flyteidl.core.AliasR\routputAliases\x12\x36\n\ttask_node\x18\x06 \x01(\x0b\x32\x17.flyteidl.core.TaskNodeH\x00R\x08taskNode\x12\x42\n\rworkflow_node\x18\x07 \x01(\x0b\x32\x1b.flyteidl.core.WorkflowNodeH\x00R\x0cworkflowNode\x12<\n\x0b\x62ranch_node\x18\x08 \x01(\x0b\x32\x19.flyteidl.core.BranchNodeH\x00R\nbranchNode\x12\x36\n\tgate_node\x18\t \x01(\x0b\x32\x17.flyteidl.core.GateNodeH\x00R\x08gateNode\x12\x39\n\narray_node\x18\n \x01(\x0b\x32\x18.flyteidl.core.ArrayNodeH\x00R\tarrayNodeB\x08\n\x06target\"\xfc\x02\n\x10WorkflowMetadata\x12M\n\x12quality_of_service\x18\x01 \x01(\x0b\x32\x1f.flyteidl.core.QualityOfServiceR\x10qualityOfService\x12N\n\non_failure\x18\x02 \x01(\x0e\x32/.flyteidl.core.WorkflowMetadata.OnFailurePolicyR\tonFailure\x12=\n\x04tags\x18\x03 \x03(\x0b\x32).flyteidl.core.WorkflowMetadata.TagsEntryR\x04tags\x1a\x37\n\tTagsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"Q\n\x0fOnFailurePolicy\x12\x14\n\x10\x46\x41IL_IMMEDIATELY\x10\x00\x12(\n$FAIL_AFTER_EXECUTABLE_NODES_COMPLETE\x10\x01\"@\n\x18WorkflowMetadataDefaults\x12$\n\rinterruptible\x18\x01 \x01(\x08R\rinterruptible\"\xa2\x03\n\x10WorkflowTemplate\x12)\n\x02id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierR\x02id\x12;\n\x08metadata\x18\x02 \x01(\x0b\x32\x1f.flyteidl.core.WorkflowMetadataR\x08metadata\x12;\n\tinterface\x18\x03 \x01(\x0b\x32\x1d.flyteidl.core.TypedInterfaceR\tinterface\x12)\n\x05nodes\x18\x04 \x03(\x0b\x32\x13.flyteidl.core.NodeR\x05nodes\x12\x30\n\x07outputs\x18\x05 \x03(\x0b\x32\x16.flyteidl.core.BindingR\x07outputs\x12\x36\n\x0c\x66\x61ilure_node\x18\x06 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x0b\x66\x61ilureNode\x12T\n\x11metadata_defaults\x18\x07 \x01(\x0b\x32\'.flyteidl.core.WorkflowMetadataDefaultsR\x10metadataDefaults\"K\n\x11TaskNodeOverrides\x12\x36\n\tresources\x18\x01 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresourcesB\xb3\x01\n\x11\x63om.flyteidl.coreB\rWorkflowProtoP\x01Z:github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core\xa2\x02\x03\x46\x43X\xaa\x02\rFlyteidl.Core\xca\x02\rFlyteidl\\Core\xe2\x02\x19\x46lyteidl\\Core\\GPBMetadata\xea\x02\x0e\x46lyteidl::Coreb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1c\x66lyteidl/core/workflow.proto\x12\rflyteidl.core\x1a\x1d\x66lyteidl/core/condition.proto\x1a\x1d\x66lyteidl/core/execution.proto\x1a\x1e\x66lyteidl/core/identifier.proto\x1a\x1d\x66lyteidl/core/interface.proto\x1a\x1c\x66lyteidl/core/literals.proto\x1a\x19\x66lyteidl/core/tasks.proto\x1a\x19\x66lyteidl/core/types.proto\x1a\x1c\x66lyteidl/core/security.proto\x1a\x1egoogle/protobuf/duration.proto\"{\n\x07IfBlock\x12>\n\tcondition\x18\x01 \x01(\x0b\x32 .flyteidl.core.BooleanExpressionR\tcondition\x12\x30\n\tthen_node\x18\x02 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x08thenNode\"\xd4\x01\n\x0bIfElseBlock\x12*\n\x04\x63\x61se\x18\x01 \x01(\x0b\x32\x16.flyteidl.core.IfBlockR\x04\x63\x61se\x12,\n\x05other\x18\x02 \x03(\x0b\x32\x16.flyteidl.core.IfBlockR\x05other\x12\x32\n\telse_node\x18\x03 \x01(\x0b\x32\x13.flyteidl.core.NodeH\x00R\x08\x65lseNode\x12,\n\x05\x65rror\x18\x04 \x01(\x0b\x32\x14.flyteidl.core.ErrorH\x00R\x05\x65rrorB\t\n\x07\x64\x65\x66\x61ult\"A\n\nBranchNode\x12\x33\n\x07if_else\x18\x01 \x01(\x0b\x32\x1a.flyteidl.core.IfElseBlockR\x06ifElse\"\x97\x01\n\x08TaskNode\x12>\n\x0creference_id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\x0breferenceId\x12>\n\toverrides\x18\x02 \x01(\x0b\x32 .flyteidl.core.TaskNodeOverridesR\toverridesB\x0b\n\treference\"\xa6\x01\n\x0cWorkflowNode\x12\x42\n\x0elaunchplan_ref\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\rlaunchplanRef\x12\x45\n\x10sub_workflow_ref\x18\x02 \x01(\x0b\x32\x19.flyteidl.core.IdentifierH\x00R\x0esubWorkflowRefB\x0b\n\treference\"/\n\x10\x41pproveCondition\x12\x1b\n\tsignal_id\x18\x01 \x01(\tR\x08signalId\"\x90\x01\n\x0fSignalCondition\x12\x1b\n\tsignal_id\x18\x01 \x01(\tR\x08signalId\x12.\n\x04type\x18\x02 \x01(\x0b\x32\x1a.flyteidl.core.LiteralTypeR\x04type\x12\x30\n\x14output_variable_name\x18\x03 \x01(\tR\x12outputVariableName\"G\n\x0eSleepCondition\x12\x35\n\x08\x64uration\x18\x01 \x01(\x0b\x32\x19.google.protobuf.DurationR\x08\x64uration\"\xc5\x01\n\x08GateNode\x12;\n\x07\x61pprove\x18\x01 \x01(\x0b\x32\x1f.flyteidl.core.ApproveConditionH\x00R\x07\x61pprove\x12\x38\n\x06signal\x18\x02 \x01(\x0b\x32\x1e.flyteidl.core.SignalConditionH\x00R\x06signal\x12\x35\n\x05sleep\x18\x03 \x01(\x0b\x32\x1d.flyteidl.core.SleepConditionH\x00R\x05sleepB\x0b\n\tcondition\"\xbf\x01\n\tArrayNode\x12\'\n\x04node\x18\x01 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x04node\x12 \n\x0bparallelism\x18\x02 \x01(\rR\x0bparallelism\x12%\n\rmin_successes\x18\x03 \x01(\rH\x00R\x0cminSuccesses\x12,\n\x11min_success_ratio\x18\x04 \x01(\x02H\x00R\x0fminSuccessRatioB\x12\n\x10success_criteria\"\xce\x01\n\x0cNodeMetadata\x12\x12\n\x04name\x18\x01 \x01(\tR\x04name\x12\x33\n\x07timeout\x18\x04 \x01(\x0b\x32\x19.google.protobuf.DurationR\x07timeout\x12\x36\n\x07retries\x18\x05 \x01(\x0b\x32\x1c.flyteidl.core.RetryStrategyR\x07retries\x12&\n\rinterruptible\x18\x06 \x01(\x08H\x00R\rinterruptibleB\x15\n\x13interruptible_value\"/\n\x05\x41lias\x12\x10\n\x03var\x18\x01 \x01(\tR\x03var\x12\x14\n\x05\x61lias\x18\x02 \x01(\tR\x05\x61lias\"\x9f\x04\n\x04Node\x12\x0e\n\x02id\x18\x01 \x01(\tR\x02id\x12\x37\n\x08metadata\x18\x02 \x01(\x0b\x32\x1b.flyteidl.core.NodeMetadataR\x08metadata\x12.\n\x06inputs\x18\x03 \x03(\x0b\x32\x16.flyteidl.core.BindingR\x06inputs\x12*\n\x11upstream_node_ids\x18\x04 \x03(\tR\x0fupstreamNodeIds\x12;\n\x0eoutput_aliases\x18\x05 \x03(\x0b\x32\x14.flyteidl.core.AliasR\routputAliases\x12\x36\n\ttask_node\x18\x06 \x01(\x0b\x32\x17.flyteidl.core.TaskNodeH\x00R\x08taskNode\x12\x42\n\rworkflow_node\x18\x07 \x01(\x0b\x32\x1b.flyteidl.core.WorkflowNodeH\x00R\x0cworkflowNode\x12<\n\x0b\x62ranch_node\x18\x08 \x01(\x0b\x32\x19.flyteidl.core.BranchNodeH\x00R\nbranchNode\x12\x36\n\tgate_node\x18\t \x01(\x0b\x32\x17.flyteidl.core.GateNodeH\x00R\x08gateNode\x12\x39\n\narray_node\x18\n \x01(\x0b\x32\x18.flyteidl.core.ArrayNodeH\x00R\tarrayNodeB\x08\n\x06target\"\xfc\x02\n\x10WorkflowMetadata\x12M\n\x12quality_of_service\x18\x01 \x01(\x0b\x32\x1f.flyteidl.core.QualityOfServiceR\x10qualityOfService\x12N\n\non_failure\x18\x02 \x01(\x0e\x32/.flyteidl.core.WorkflowMetadata.OnFailurePolicyR\tonFailure\x12=\n\x04tags\x18\x03 \x03(\x0b\x32).flyteidl.core.WorkflowMetadata.TagsEntryR\x04tags\x1a\x37\n\tTagsEntry\x12\x10\n\x03key\x18\x01 \x01(\tR\x03key\x12\x14\n\x05value\x18\x02 \x01(\tR\x05value:\x02\x38\x01\"Q\n\x0fOnFailurePolicy\x12\x14\n\x10\x46\x41IL_IMMEDIATELY\x10\x00\x12(\n$FAIL_AFTER_EXECUTABLE_NODES_COMPLETE\x10\x01\"@\n\x18WorkflowMetadataDefaults\x12$\n\rinterruptible\x18\x01 \x01(\x08R\rinterruptible\"\xa2\x03\n\x10WorkflowTemplate\x12)\n\x02id\x18\x01 \x01(\x0b\x32\x19.flyteidl.core.IdentifierR\x02id\x12;\n\x08metadata\x18\x02 \x01(\x0b\x32\x1f.flyteidl.core.WorkflowMetadataR\x08metadata\x12;\n\tinterface\x18\x03 \x01(\x0b\x32\x1d.flyteidl.core.TypedInterfaceR\tinterface\x12)\n\x05nodes\x18\x04 \x03(\x0b\x32\x13.flyteidl.core.NodeR\x05nodes\x12\x30\n\x07outputs\x18\x05 \x03(\x0b\x32\x16.flyteidl.core.BindingR\x07outputs\x12\x36\n\x0c\x66\x61ilure_node\x18\x06 \x01(\x0b\x32\x13.flyteidl.core.NodeR\x0b\x66\x61ilureNode\x12T\n\x11metadata_defaults\x18\x07 \x01(\x0b\x32\'.flyteidl.core.WorkflowMetadataDefaultsR\x10metadataDefaults\"\x9c\x01\n\x11TaskNodeOverrides\x12\x36\n\tresources\x18\x01 \x01(\x0b\x32\x18.flyteidl.core.ResourcesR\tresources\x12O\n\x12\x65xtended_resources\x18\x02 \x01(\x0b\x32 .flyteidl.core.ExtendedResourcesR\x11\x65xtendedResourcesB\xb3\x01\n\x11\x63om.flyteidl.coreB\rWorkflowProtoP\x01Z:github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core\xa2\x02\x03\x46\x43X\xaa\x02\rFlyteidl.Core\xca\x02\rFlyteidl\\Core\xe2\x02\x19\x46lyteidl\\Core\\GPBMetadata\xea\x02\x0e\x46lyteidl::Coreb\x06proto3') _globals = globals() _builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) @@ -69,6 +69,6 @@ _globals['_WORKFLOWMETADATADEFAULTS']._serialized_end=2962 _globals['_WORKFLOWTEMPLATE']._serialized_start=2965 _globals['_WORKFLOWTEMPLATE']._serialized_end=3383 - _globals['_TASKNODEOVERRIDES']._serialized_start=3385 - _globals['_TASKNODEOVERRIDES']._serialized_end=3460 + _globals['_TASKNODEOVERRIDES']._serialized_start=3386 + _globals['_TASKNODEOVERRIDES']._serialized_end=3542 # @@protoc_insertion_point(module_scope) diff --git a/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.pyi b/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.pyi index 95a108abcdc..daf8082c7b4 100644 --- a/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.pyi +++ b/flyteidl/gen/pb_python/flyteidl/core/workflow_pb2.pyi @@ -193,7 +193,9 @@ class WorkflowTemplate(_message.Message): def __init__(self, id: _Optional[_Union[_identifier_pb2.Identifier, _Mapping]] = ..., metadata: _Optional[_Union[WorkflowMetadata, _Mapping]] = ..., interface: _Optional[_Union[_interface_pb2.TypedInterface, _Mapping]] = ..., nodes: _Optional[_Iterable[_Union[Node, _Mapping]]] = ..., outputs: _Optional[_Iterable[_Union[_literals_pb2.Binding, _Mapping]]] = ..., failure_node: _Optional[_Union[Node, _Mapping]] = ..., metadata_defaults: _Optional[_Union[WorkflowMetadataDefaults, _Mapping]] = ...) -> None: ... class TaskNodeOverrides(_message.Message): - __slots__ = ["resources"] + __slots__ = ["resources", "extended_resources"] RESOURCES_FIELD_NUMBER: _ClassVar[int] + EXTENDED_RESOURCES_FIELD_NUMBER: _ClassVar[int] resources: _tasks_pb2.Resources - def __init__(self, resources: _Optional[_Union[_tasks_pb2.Resources, _Mapping]] = ...) -> None: ... + extended_resources: _tasks_pb2.ExtendedResources + def __init__(self, resources: _Optional[_Union[_tasks_pb2.Resources, _Mapping]] = ..., extended_resources: _Optional[_Union[_tasks_pb2.ExtendedResources, _Mapping]] = ...) -> None: ... diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/README.md b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/README.md index 3f98e5376a0..e255b6e3082 100644 --- a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/README.md +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/README.md @@ -289,6 +289,8 @@ Class | Method | HTTP request | Description - [CoreEnumType](docs/CoreEnumType.md) - [CoreError](docs/CoreError.md) - [CoreExecutionError](docs/CoreExecutionError.md) + - [CoreExtendedResources](docs/CoreExtendedResources.md) + - [CoreGPUAccelerator](docs/CoreGPUAccelerator.md) - [CoreGateNode](docs/CoreGateNode.md) - [CoreIOStrategy](docs/CoreIOStrategy.md) - [CoreIdentifier](docs/CoreIdentifier.md) diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/__init__.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/__init__.py index 5db75c705d3..18f679a7f0f 100644 --- a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/__init__.py +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/__init__.py @@ -180,6 +180,8 @@ from flyteadmin.models.core_enum_type import CoreEnumType from flyteadmin.models.core_error import CoreError from flyteadmin.models.core_execution_error import CoreExecutionError +from flyteadmin.models.core_extended_resources import CoreExtendedResources +from flyteadmin.models.core_gpu_accelerator import CoreGPUAccelerator from flyteadmin.models.core_gate_node import CoreGateNode from flyteadmin.models.core_io_strategy import CoreIOStrategy from flyteadmin.models.core_identifier import CoreIdentifier diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/__init__.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/__init__.py index 8ff7d20249f..457a507c4f9 100644 --- a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/__init__.py +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/__init__.py @@ -173,6 +173,8 @@ from flyteadmin.models.core_enum_type import CoreEnumType from flyteadmin.models.core_error import CoreError from flyteadmin.models.core_execution_error import CoreExecutionError +from flyteadmin.models.core_extended_resources import CoreExtendedResources +from flyteadmin.models.core_gpu_accelerator import CoreGPUAccelerator from flyteadmin.models.core_gate_node import CoreGateNode from flyteadmin.models.core_io_strategy import CoreIOStrategy from flyteadmin.models.core_identifier import CoreIdentifier diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_extended_resources.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_extended_resources.py new file mode 100644 index 00000000000..94a58bcd4f3 --- /dev/null +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_extended_resources.py @@ -0,0 +1,119 @@ +# coding: utf-8 + +""" + flyteidl/service/admin.proto + + No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) # noqa: E501 + + OpenAPI spec version: version not set + + Generated by: https://github.com/swagger-api/swagger-codegen.git +""" + + +import pprint +import re # noqa: F401 + +import six + +from flyteadmin.models.core_gpu_accelerator import CoreGPUAccelerator # noqa: F401,E501 + + +class CoreExtendedResources(object): + """NOTE: This class is auto generated by the swagger code generator program. + + Do not edit the class manually. + """ + + """ + Attributes: + swagger_types (dict): The key is attribute name + and the value is attribute type. + attribute_map (dict): The key is attribute name + and the value is json key in definition. + """ + swagger_types = { + 'gpu_accelerator': 'CoreGPUAccelerator' + } + + attribute_map = { + 'gpu_accelerator': 'gpu_accelerator' + } + + def __init__(self, gpu_accelerator=None): # noqa: E501 + """CoreExtendedResources - a model defined in Swagger""" # noqa: E501 + + self._gpu_accelerator = None + self.discriminator = None + + if gpu_accelerator is not None: + self.gpu_accelerator = gpu_accelerator + + @property + def gpu_accelerator(self): + """Gets the gpu_accelerator of this CoreExtendedResources. # noqa: E501 + + GPU accelerator to select for task. Contains information about device type, and for multi-instance GPUs, the partition size to use. # noqa: E501 + + :return: The gpu_accelerator of this CoreExtendedResources. # noqa: E501 + :rtype: CoreGPUAccelerator + """ + return self._gpu_accelerator + + @gpu_accelerator.setter + def gpu_accelerator(self, gpu_accelerator): + """Sets the gpu_accelerator of this CoreExtendedResources. + + GPU accelerator to select for task. Contains information about device type, and for multi-instance GPUs, the partition size to use. # noqa: E501 + + :param gpu_accelerator: The gpu_accelerator of this CoreExtendedResources. # noqa: E501 + :type: CoreGPUAccelerator + """ + + self._gpu_accelerator = gpu_accelerator + + def to_dict(self): + """Returns the model properties as a dict""" + result = {} + + for attr, _ in six.iteritems(self.swagger_types): + value = getattr(self, attr) + if isinstance(value, list): + result[attr] = list(map( + lambda x: x.to_dict() if hasattr(x, "to_dict") else x, + value + )) + elif hasattr(value, "to_dict"): + result[attr] = value.to_dict() + elif isinstance(value, dict): + result[attr] = dict(map( + lambda item: (item[0], item[1].to_dict()) + if hasattr(item[1], "to_dict") else item, + value.items() + )) + else: + result[attr] = value + if issubclass(CoreExtendedResources, dict): + for key, value in self.items(): + result[key] = value + + return result + + def to_str(self): + """Returns the string representation of the model""" + return pprint.pformat(self.to_dict()) + + def __repr__(self): + """For `print` and `pprint`""" + return self.to_str() + + def __eq__(self, other): + """Returns true if both objects are equal""" + if not isinstance(other, CoreExtendedResources): + return False + + return self.__dict__ == other.__dict__ + + def __ne__(self, other): + """Returns true if both objects are not equal""" + return not self == other diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_gpu_accelerator.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_gpu_accelerator.py new file mode 100644 index 00000000000..760ff432672 --- /dev/null +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_gpu_accelerator.py @@ -0,0 +1,171 @@ +# coding: utf-8 + +""" + flyteidl/service/admin.proto + + No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) # noqa: E501 + + OpenAPI spec version: version not set + + Generated by: https://github.com/swagger-api/swagger-codegen.git +""" + + +import pprint +import re # noqa: F401 + +import six + + +class CoreGPUAccelerator(object): + """NOTE: This class is auto generated by the swagger code generator program. + + Do not edit the class manually. + """ + + """ + Attributes: + swagger_types (dict): The key is attribute name + and the value is attribute type. + attribute_map (dict): The key is attribute name + and the value is json key in definition. + """ + swagger_types = { + 'device': 'str', + 'unpartitioned': 'bool', + 'partition_size': 'str' + } + + attribute_map = { + 'device': 'device', + 'unpartitioned': 'unpartitioned', + 'partition_size': 'partition_size' + } + + def __init__(self, device=None, unpartitioned=None, partition_size=None): # noqa: E501 + """CoreGPUAccelerator - a model defined in Swagger""" # noqa: E501 + + self._device = None + self._unpartitioned = None + self._partition_size = None + self.discriminator = None + + if device is not None: + self.device = device + if unpartitioned is not None: + self.unpartitioned = unpartitioned + if partition_size is not None: + self.partition_size = partition_size + + @property + def device(self): + """Gets the device of this CoreGPUAccelerator. # noqa: E501 + + This can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. # noqa: E501 + + :return: The device of this CoreGPUAccelerator. # noqa: E501 + :rtype: str + """ + return self._device + + @device.setter + def device(self, device): + """Sets the device of this CoreGPUAccelerator. + + This can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. # noqa: E501 + + :param device: The device of this CoreGPUAccelerator. # noqa: E501 + :type: str + """ + + self._device = device + + @property + def unpartitioned(self): + """Gets the unpartitioned of this CoreGPUAccelerator. # noqa: E501 + + + :return: The unpartitioned of this CoreGPUAccelerator. # noqa: E501 + :rtype: bool + """ + return self._unpartitioned + + @unpartitioned.setter + def unpartitioned(self, unpartitioned): + """Sets the unpartitioned of this CoreGPUAccelerator. + + + :param unpartitioned: The unpartitioned of this CoreGPUAccelerator. # noqa: E501 + :type: bool + """ + + self._unpartitioned = unpartitioned + + @property + def partition_size(self): + """Gets the partition_size of this CoreGPUAccelerator. # noqa: E501 + + Like `device`, this can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc. # noqa: E501 + + :return: The partition_size of this CoreGPUAccelerator. # noqa: E501 + :rtype: str + """ + return self._partition_size + + @partition_size.setter + def partition_size(self, partition_size): + """Sets the partition_size of this CoreGPUAccelerator. + + Like `device`, this can be any arbitrary string, and should be informed by the labels or taints associated with the nodes in question. Default cloud provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc. # noqa: E501 + + :param partition_size: The partition_size of this CoreGPUAccelerator. # noqa: E501 + :type: str + """ + + self._partition_size = partition_size + + def to_dict(self): + """Returns the model properties as a dict""" + result = {} + + for attr, _ in six.iteritems(self.swagger_types): + value = getattr(self, attr) + if isinstance(value, list): + result[attr] = list(map( + lambda x: x.to_dict() if hasattr(x, "to_dict") else x, + value + )) + elif hasattr(value, "to_dict"): + result[attr] = value.to_dict() + elif isinstance(value, dict): + result[attr] = dict(map( + lambda item: (item[0], item[1].to_dict()) + if hasattr(item[1], "to_dict") else item, + value.items() + )) + else: + result[attr] = value + if issubclass(CoreGPUAccelerator, dict): + for key, value in self.items(): + result[key] = value + + return result + + def to_str(self): + """Returns the string representation of the model""" + return pprint.pformat(self.to_dict()) + + def __repr__(self): + """For `print` and `pprint`""" + return self.to_str() + + def __eq__(self, other): + """Returns true if both objects are equal""" + if not isinstance(other, CoreGPUAccelerator): + return False + + return self.__dict__ == other.__dict__ + + def __ne__(self, other): + """Returns true if both objects are not equal""" + return not self == other diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_node_overrides.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_node_overrides.py index 7320a6d6e8a..54617984db8 100644 --- a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_node_overrides.py +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_node_overrides.py @@ -16,6 +16,7 @@ import six +from flyteadmin.models.core_extended_resources import CoreExtendedResources # noqa: F401,E501 from flyteadmin.models.core_resources import CoreResources # noqa: F401,E501 @@ -33,21 +34,26 @@ class CoreTaskNodeOverrides(object): and the value is json key in definition. """ swagger_types = { - 'resources': 'CoreResources' + 'resources': 'CoreResources', + 'extended_resources': 'CoreExtendedResources' } attribute_map = { - 'resources': 'resources' + 'resources': 'resources', + 'extended_resources': 'extended_resources' } - def __init__(self, resources=None): # noqa: E501 + def __init__(self, resources=None, extended_resources=None): # noqa: E501 """CoreTaskNodeOverrides - a model defined in Swagger""" # noqa: E501 self._resources = None + self._extended_resources = None self.discriminator = None if resources is not None: self.resources = resources + if extended_resources is not None: + self.extended_resources = extended_resources @property def resources(self): @@ -72,6 +78,29 @@ def resources(self, resources): self._resources = resources + @property + def extended_resources(self): + """Gets the extended_resources of this CoreTaskNodeOverrides. # noqa: E501 + + Overrides for all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. # noqa: E501 + + :return: The extended_resources of this CoreTaskNodeOverrides. # noqa: E501 + :rtype: CoreExtendedResources + """ + return self._extended_resources + + @extended_resources.setter + def extended_resources(self, extended_resources): + """Sets the extended_resources of this CoreTaskNodeOverrides. + + Overrides for all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. # noqa: E501 + + :param extended_resources: The extended_resources of this CoreTaskNodeOverrides. # noqa: E501 + :type: CoreExtendedResources + """ + + self._extended_resources = extended_resources + def to_dict(self): """Returns the model properties as a dict""" result = {} diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_template.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_template.py index 870777f995a..df06a8db04b 100644 --- a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_template.py +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/flyteadmin/models/core_task_template.py @@ -17,6 +17,7 @@ import six from flyteadmin.models.core_container import CoreContainer # noqa: F401,E501 +from flyteadmin.models.core_extended_resources import CoreExtendedResources # noqa: F401,E501 from flyteadmin.models.core_identifier import CoreIdentifier # noqa: F401,E501 from flyteadmin.models.core_k8s_pod import CoreK8sPod # noqa: F401,E501 from flyteadmin.models.core_security_context import CoreSecurityContext # noqa: F401,E501 @@ -50,6 +51,7 @@ class CoreTaskTemplate(object): 'sql': 'CoreSql', 'task_type_version': 'int', 'security_context': 'CoreSecurityContext', + 'extended_resources': 'CoreExtendedResources', 'config': 'dict(str, str)' } @@ -64,10 +66,11 @@ class CoreTaskTemplate(object): 'sql': 'sql', 'task_type_version': 'task_type_version', 'security_context': 'security_context', + 'extended_resources': 'extended_resources', 'config': 'config' } - def __init__(self, id=None, type=None, metadata=None, interface=None, custom=None, container=None, k8s_pod=None, sql=None, task_type_version=None, security_context=None, config=None): # noqa: E501 + def __init__(self, id=None, type=None, metadata=None, interface=None, custom=None, container=None, k8s_pod=None, sql=None, task_type_version=None, security_context=None, extended_resources=None, config=None): # noqa: E501 """CoreTaskTemplate - a model defined in Swagger""" # noqa: E501 self._id = None @@ -80,6 +83,7 @@ def __init__(self, id=None, type=None, metadata=None, interface=None, custom=Non self._sql = None self._task_type_version = None self._security_context = None + self._extended_resources = None self._config = None self.discriminator = None @@ -103,6 +107,8 @@ def __init__(self, id=None, type=None, metadata=None, interface=None, custom=Non self.task_type_version = task_type_version if security_context is not None: self.security_context = security_context + if extended_resources is not None: + self.extended_resources = extended_resources if config is not None: self.config = config @@ -330,6 +336,29 @@ def security_context(self, security_context): self._security_context = security_context + @property + def extended_resources(self): + """Gets the extended_resources of this CoreTaskTemplate. # noqa: E501 + + Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. # noqa: E501 + + :return: The extended_resources of this CoreTaskTemplate. # noqa: E501 + :rtype: CoreExtendedResources + """ + return self._extended_resources + + @extended_resources.setter + def extended_resources(self, extended_resources): + """Sets the extended_resources of this CoreTaskTemplate. + + Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to allocate to a task. # noqa: E501 + + :param extended_resources: The extended_resources of this CoreTaskTemplate. # noqa: E501 + :type: CoreExtendedResources + """ + + self._extended_resources = extended_resources + @property def config(self): """Gets the config of this CoreTaskTemplate. # noqa: E501 diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_extended_resources.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_extended_resources.py new file mode 100644 index 00000000000..ec6edecb757 --- /dev/null +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_extended_resources.py @@ -0,0 +1,40 @@ +# coding: utf-8 + +""" + flyteidl/service/admin.proto + + No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) # noqa: E501 + + OpenAPI spec version: version not set + + Generated by: https://github.com/swagger-api/swagger-codegen.git +""" + + +from __future__ import absolute_import + +import unittest + +import flyteadmin +from flyteadmin.models.core_extended_resources import CoreExtendedResources # noqa: E501 +from flyteadmin.rest import ApiException + + +class TestCoreExtendedResources(unittest.TestCase): + """CoreExtendedResources unit test stubs""" + + def setUp(self): + pass + + def tearDown(self): + pass + + def testCoreExtendedResources(self): + """Test CoreExtendedResources""" + # FIXME: construct object with mandatory attributes with example values + # model = flyteadmin.models.core_extended_resources.CoreExtendedResources() # noqa: E501 + pass + + +if __name__ == '__main__': + unittest.main() diff --git a/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_gpu_accelerator.py b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_gpu_accelerator.py new file mode 100644 index 00000000000..aa99885e5f8 --- /dev/null +++ b/flyteidl/gen/pb_python/flyteidl/service/flyteadmin/test/test_core_gpu_accelerator.py @@ -0,0 +1,40 @@ +# coding: utf-8 + +""" + flyteidl/service/admin.proto + + No description provided (generated by Swagger Codegen https://github.com/swagger-api/swagger-codegen) # noqa: E501 + + OpenAPI spec version: version not set + + Generated by: https://github.com/swagger-api/swagger-codegen.git +""" + + +from __future__ import absolute_import + +import unittest + +import flyteadmin +from flyteadmin.models.core_gpu_accelerator import CoreGPUAccelerator # noqa: E501 +from flyteadmin.rest import ApiException + + +class TestCoreGPUAccelerator(unittest.TestCase): + """CoreGPUAccelerator unit test stubs""" + + def setUp(self): + pass + + def tearDown(self): + pass + + def testCoreGPUAccelerator(self): + """Test CoreGPUAccelerator""" + # FIXME: construct object with mandatory attributes with example values + # model = flyteadmin.models.core_gpu_accelerator.CoreGPUAccelerator() # noqa: E501 + pass + + +if __name__ == '__main__': + unittest.main() diff --git a/flyteidl/gen/pb_rust/flyteidl.core.rs b/flyteidl/gen/pb_rust/flyteidl.core.rs index e3328f81d3c..b0c99fbfcbe 100644 --- a/flyteidl/gen/pb_rust/flyteidl.core.rs +++ b/flyteidl/gen/pb_rust/flyteidl.core.rs @@ -1008,6 +1008,44 @@ pub mod resources { } } } +/// Metadata associated with the GPU accelerator to allocate to a task. Contains +/// information about device type, and for multi-instance GPUs, the partition size to +/// use. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct GpuAccelerator { + /// This can be any arbitrary string, and should be informed by the labels or taints + /// associated with the nodes in question. Default cloud provider labels typically + /// use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. + #[prost(string, tag="1")] + pub device: ::prost::alloc::string::String, + #[prost(oneof="gpu_accelerator::PartitionSizeValue", tags="2, 3")] + pub partition_size_value: ::core::option::Option, +} +/// Nested message and enum types in `GPUAccelerator`. +pub mod gpu_accelerator { + #[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Oneof)] + pub enum PartitionSizeValue { + #[prost(bool, tag="2")] + Unpartitioned(bool), + /// Like `device`, this can be any arbitrary string, and should be informed by + /// the labels or taints associated with the nodes in question. Default cloud + /// provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc. + #[prost(string, tag="3")] + PartitionSize(::prost::alloc::string::String), + } +} +/// Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to +/// allocate to a task. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ExtendedResources { + /// GPU accelerator to select for task. Contains information about device type, and + /// for multi-instance GPUs, the partition size to use. + #[prost(message, optional, tag="1")] + pub gpu_accelerator: ::core::option::Option, +} /// Runtime information. This is loosely defined to allow for extensibility. #[allow(clippy::derive_partial_eq_without_eq)] #[derive(Clone, PartialEq, ::prost::Message)] @@ -1144,6 +1182,10 @@ pub struct TaskTemplate { /// security_context encapsulates security attributes requested to run this task. #[prost(message, optional, tag="8")] pub security_context: ::core::option::Option, + /// Encapsulates all non-standard resources, not captured by + /// v1.ResourceRequirements, to allocate to a task. + #[prost(message, optional, tag="9")] + pub extended_resources: ::core::option::Option, /// Metadata about the custom defined for this task. This is extensible to allow various plugins in the system /// to use as required. /// reserve the field numbers 1 through 15 for very frequently occurring message elements @@ -2345,6 +2387,10 @@ pub struct TaskNodeOverrides { /// A customizable interface to convey resources requested for a task container. #[prost(message, optional, tag="1")] pub resources: ::core::option::Option, + /// Overrides for all non-standard resources, not captured by + /// v1.ResourceRequirements, to allocate to a task. + #[prost(message, optional, tag="2")] + pub extended_resources: ::core::option::Option, } /// Adjacency list for the workflow. This is created as part of the compilation process. Every process after the compilation /// step uses this created ConnectionSet diff --git a/flyteidl/protos/flyteidl/core/tasks.proto b/flyteidl/protos/flyteidl/core/tasks.proto index f648a2c93fe..3d3f4b6abfe 100644 --- a/flyteidl/protos/flyteidl/core/tasks.proto +++ b/flyteidl/protos/flyteidl/core/tasks.proto @@ -43,6 +43,31 @@ message Resources { repeated ResourceEntry limits = 2; } +// Metadata associated with the GPU accelerator to allocate to a task. Contains +// information about device type, and for multi-instance GPUs, the partition size to +// use. +message GPUAccelerator { + // This can be any arbitrary string, and should be informed by the labels or taints + // associated with the nodes in question. Default cloud provider labels typically + // use the following values: `nvidia-tesla-t4`, `nvidia-tesla-a100`, etc. + string device = 1; + oneof partition_size_value { + bool unpartitioned = 2; + // Like `device`, this can be any arbitrary string, and should be informed by + // the labels or taints associated with the nodes in question. Default cloud + // provider labels typically use the following values: `1g.5gb`, `2g.10gb`, etc. + string partition_size = 3; + } +} + +// Encapsulates all non-standard resources, not captured by v1.ResourceRequirements, to +// allocate to a task. +message ExtendedResources { + // GPU accelerator to select for task. Contains information about device type, and + // for multi-instance GPUs, the partition size to use. + GPUAccelerator gpu_accelerator = 1; +} + // Runtime information. This is loosely defined to allow for extensibility. message RuntimeMetadata { enum RuntimeType { @@ -144,6 +169,10 @@ message TaskTemplate { // security_context encapsulates security attributes requested to run this task. SecurityContext security_context = 8; + // Encapsulates all non-standard resources, not captured by + // v1.ResourceRequirements, to allocate to a task. + ExtendedResources extended_resources = 9; + // Metadata about the custom defined for this task. This is extensible to allow various plugins in the system // to use as required. // reserve the field numbers 1 through 15 for very frequently occurring message elements diff --git a/flyteidl/protos/flyteidl/core/workflow.proto b/flyteidl/protos/flyteidl/core/workflow.proto index 7f5a2fb15d6..8c74094b5b1 100644 --- a/flyteidl/protos/flyteidl/core/workflow.proto +++ b/flyteidl/protos/flyteidl/core/workflow.proto @@ -279,4 +279,8 @@ message WorkflowTemplate { message TaskNodeOverrides { // A customizable interface to convey resources requested for a task container. Resources resources = 1; + + // Overrides for all non-standard resources, not captured by + // v1.ResourceRequirements, to allocate to a task. + ExtendedResources extended_resources = 2; } diff --git a/flyteplugins/go/tasks/pluginmachinery/core/exec_metadata.go b/flyteplugins/go/tasks/pluginmachinery/core/exec_metadata.go index 18138006cea..7dc5ece5df3 100644 --- a/flyteplugins/go/tasks/pluginmachinery/core/exec_metadata.go +++ b/flyteplugins/go/tasks/pluginmachinery/core/exec_metadata.go @@ -10,6 +10,7 @@ import ( // TaskOverrides interface to expose any overrides that have been set for this task (like resource overrides etc) type TaskOverrides interface { GetResources() *v1.ResourceRequirements + GetExtendedResources() *core.ExtendedResources GetConfig() *v1.ConfigMap } diff --git a/flyteplugins/go/tasks/pluginmachinery/core/mocks/task_overrides.go b/flyteplugins/go/tasks/pluginmachinery/core/mocks/task_overrides.go index d40f1461d71..d00973c1145 100644 --- a/flyteplugins/go/tasks/pluginmachinery/core/mocks/task_overrides.go +++ b/flyteplugins/go/tasks/pluginmachinery/core/mocks/task_overrides.go @@ -3,7 +3,9 @@ package mocks import ( + flyteidlcore "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" mock "github.com/stretchr/testify/mock" + v1 "k8s.io/api/core/v1" ) @@ -46,6 +48,40 @@ func (_m *TaskOverrides) GetConfig() *v1.ConfigMap { return r0 } +type TaskOverrides_GetExtendedResources struct { + *mock.Call +} + +func (_m TaskOverrides_GetExtendedResources) Return(_a0 *flyteidlcore.ExtendedResources) *TaskOverrides_GetExtendedResources { + return &TaskOverrides_GetExtendedResources{Call: _m.Call.Return(_a0)} +} + +func (_m *TaskOverrides) OnGetExtendedResources() *TaskOverrides_GetExtendedResources { + c_call := _m.On("GetExtendedResources") + return &TaskOverrides_GetExtendedResources{Call: c_call} +} + +func (_m *TaskOverrides) OnGetExtendedResourcesMatch(matchers ...interface{}) *TaskOverrides_GetExtendedResources { + c_call := _m.On("GetExtendedResources", matchers...) + return &TaskOverrides_GetExtendedResources{Call: c_call} +} + +// GetExtendedResources provides a mock function with given fields: +func (_m *TaskOverrides) GetExtendedResources() *flyteidlcore.ExtendedResources { + ret := _m.Called() + + var r0 *flyteidlcore.ExtendedResources + if rf, ok := ret.Get(0).(func() *flyteidlcore.ExtendedResources); ok { + r0 = rf() + } else { + if ret.Get(0) != nil { + r0 = ret.Get(0).(*flyteidlcore.ExtendedResources) + } + } + + return r0 +} + type TaskOverrides_GetResources struct { *mock.Call } diff --git a/flyteplugins/go/tasks/pluginmachinery/flytek8s/config/config.go b/flyteplugins/go/tasks/pluginmachinery/flytek8s/config/config.go index 67547f82569..a58fbfe55c9 100644 --- a/flyteplugins/go/tasks/pluginmachinery/flytek8s/config/config.go +++ b/flyteplugins/go/tasks/pluginmachinery/flytek8s/config/config.go @@ -53,7 +53,9 @@ var ( ImagePullBackoffGracePeriod: config2.Duration{ Duration: time.Minute * 3, }, - GpuResourceName: ResourceNvidiaGPU, + GpuDeviceNodeLabel: "k8s.amazonaws.com/accelerator", + GpuPartitionSizeNodeLabel: "k8s.amazonaws.com/gpu-partition-size", + GpuResourceName: ResourceNvidiaGPU, DefaultPodTemplateResync: config2.Duration{ Duration: 30 * time.Second, }, @@ -140,6 +142,18 @@ type K8sPluginConfig struct { // one, and the corresponding task marked as failed ImagePullBackoffGracePeriod config2.Duration `json:"image-pull-backoff-grace-period" pflag:"-,Time to wait for transient ImagePullBackoff errors to be resolved."` + // The node label that specifies the attached GPU device. + GpuDeviceNodeLabel string `json:"gpu-device-node-label" pflag:"-,The node label that specifies the attached GPU device."` + + // The node label that specifies the attached GPU partition size. + GpuPartitionSizeNodeLabel string `json:"gpu-partition-size-node-label" pflag:"-,The node label that specifies the attached GPU partition size."` + + // Override for node selector requirement added to pods intended for unpartitioned GPU nodes. + GpuUnpartitionedNodeSelectorRequirement *v1.NodeSelectorRequirement `json:"gpu-unpartitioned-node-selector-requirement" pflag:"-,Override for node selector requirement added to pods intended for unpartitioned GPU nodes."` + + // Toleration added to pods intended for unpartitioned GPU nodes. + GpuUnpartitionedToleration *v1.Toleration `json:"gpu-unpartitioned-toleration" pflag:"-,Toleration added to pods intended for unpartitioned GPU nodes."` + // The name of the GPU resource to use when the task resource requests GPUs. GpuResourceName v1.ResourceName `json:"gpu-resource-name" pflag:"-,The name of the GPU resource to use when the task resource requests GPUs."` diff --git a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go index 93de6a0f390..6c80fbd09d8 100644 --- a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go +++ b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper.go @@ -14,6 +14,7 @@ import ( "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/template" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" + "github.com/golang/protobuf/proto" "github.com/flyteorg/flyte/flytestdlib/logger" @@ -32,6 +33,51 @@ const defaultContainerTemplateName = "default" const primaryContainerTemplateName = "primary" const PrimaryContainerKey = "primary_container_name" +// AddRequiredNodeSelectorRequirements adds the provided v1.NodeSelectorRequirement +// objects to an existing v1.Affinity object. If there are no existing required +// node selectors, the new v1.NodeSelectorRequirement will be added as-is. +// However, if there are existing required node selectors, we iterate over all existing +// node selector terms and append the node selector requirement. Note that multiple node +// selector terms are OR'd, and match expressions within a single node selector term +// are AND'd during scheduling. +// See: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity +func AddRequiredNodeSelectorRequirements(base *v1.Affinity, new ...v1.NodeSelectorRequirement) { + if base.NodeAffinity == nil { + base.NodeAffinity = &v1.NodeAffinity{} + } + if base.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution == nil { + base.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution = &v1.NodeSelector{} + } + if len(base.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms) > 0 { + nodeSelectorTerms := base.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms + for i := range nodeSelectorTerms { + nst := &nodeSelectorTerms[i] + nst.MatchExpressions = append(nst.MatchExpressions, new...) + } + } else { + base.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms = []v1.NodeSelectorTerm{v1.NodeSelectorTerm{MatchExpressions: new}} + } +} + +// AddPreferredNodeSelectorRequirements appends the provided v1.NodeSelectorRequirement +// objects to an existing v1.Affinity object's list of preferred scheduling terms. +// See: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity-weight +// for how weights are used during scheduling. +func AddPreferredNodeSelectorRequirements(base *v1.Affinity, weight int32, new ...v1.NodeSelectorRequirement) { + if base.NodeAffinity == nil { + base.NodeAffinity = &v1.NodeAffinity{} + } + base.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution = append( + base.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution, + v1.PreferredSchedulingTerm{ + Weight: weight, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: new, + }, + }, + ) +} + // ApplyInterruptibleNodeSelectorRequirement configures the node selector requirement of the node-affinity using the configuration specified. func ApplyInterruptibleNodeSelectorRequirement(interruptible bool, affinity *v1.Affinity) { // Determine node selector terms to add to node affinity @@ -48,31 +94,125 @@ func ApplyInterruptibleNodeSelectorRequirement(interruptible bool, affinity *v1. nodeSelectorRequirement = *config.GetK8sPluginConfig().NonInterruptibleNodeSelectorRequirement } - if affinity.NodeAffinity == nil { - affinity.NodeAffinity = &v1.NodeAffinity{} - } - if affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution == nil { - affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution = &v1.NodeSelector{} + AddRequiredNodeSelectorRequirements(affinity, nodeSelectorRequirement) +} + +// ApplyInterruptibleNodeAffinity configures the node-affinity for the pod using the configuration specified. +func ApplyInterruptibleNodeAffinity(interruptible bool, podSpec *v1.PodSpec) { + if podSpec.Affinity == nil { + podSpec.Affinity = &v1.Affinity{} } - if len(affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms) > 0 { - nodeSelectorTerms := affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms - for i := range nodeSelectorTerms { - nst := &nodeSelectorTerms[i] - nst.MatchExpressions = append(nst.MatchExpressions, nodeSelectorRequirement) - } + + ApplyInterruptibleNodeSelectorRequirement(interruptible, podSpec.Affinity) +} + +// Specialized merging of overrides into a base *core.ExtendedResources object. Note +// that doing a nested merge may not be the intended behavior all the time, so we +// handle each field separately here. +func applyExtendedResourcesOverrides(base, overrides *core.ExtendedResources) *core.ExtendedResources { + // Handle case where base might be nil + var new *core.ExtendedResources + if base == nil { + new = &core.ExtendedResources{} } else { - affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms = []v1.NodeSelectorTerm{{MatchExpressions: []v1.NodeSelectorRequirement{nodeSelectorRequirement}}} + new = proto.Clone(base).(*core.ExtendedResources) } + // No overrides found + if overrides == nil { + return new + } + + // GPU Accelerator + if overrides.GetGpuAccelerator() != nil { + new.GpuAccelerator = overrides.GetGpuAccelerator() + } + + return new } -// ApplyInterruptibleNodeAffinity configures the node-affinity for the pod using the configuration specified. -func ApplyInterruptibleNodeAffinity(interruptible bool, podSpec *v1.PodSpec) { +func ApplyGPUNodeSelectors(podSpec *v1.PodSpec, gpuAccelerator *core.GPUAccelerator) { + // Short circuit if pod spec does not contain any containers that use GPUs + gpuResourceName := config.GetK8sPluginConfig().GpuResourceName + requiresGPUs := false + for _, cnt := range podSpec.Containers { + if _, ok := cnt.Resources.Limits[gpuResourceName]; ok { + requiresGPUs = true + break + } + } + if !requiresGPUs { + return + } + if podSpec.Affinity == nil { podSpec.Affinity = &v1.Affinity{} } - ApplyInterruptibleNodeSelectorRequirement(interruptible, podSpec.Affinity) + // Apply changes for GPU device + device := gpuAccelerator.GetDevice() + if len(device) > 0 { + // Add node selector requirement for GPU device + deviceNsr := v1.NodeSelectorRequirement{ + Key: config.GetK8sPluginConfig().GpuDeviceNodeLabel, + Operator: v1.NodeSelectorOpIn, + Values: []string{device}, + } + AddRequiredNodeSelectorRequirements(podSpec.Affinity, deviceNsr) + // Add toleration for GPU device + deviceTol := v1.Toleration{ + Key: config.GetK8sPluginConfig().GpuDeviceNodeLabel, + Value: device, + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + } + podSpec.Tolerations = append(podSpec.Tolerations, deviceTol) + } + + // Short circuit if a partition size preference is not specified + partitionSizeValue := gpuAccelerator.GetPartitionSizeValue() + if partitionSizeValue == nil { + return + } + + // Apply changes for GPU partition size, if applicable + var partitionSizeNsr *v1.NodeSelectorRequirement + var partitionSizeTol *v1.Toleration + switch p := partitionSizeValue.(type) { + case *core.GPUAccelerator_Unpartitioned: + if !p.Unpartitioned { + break + } + if config.GetK8sPluginConfig().GpuUnpartitionedNodeSelectorRequirement != nil { + partitionSizeNsr = config.GetK8sPluginConfig().GpuUnpartitionedNodeSelectorRequirement + } else { + partitionSizeNsr = &v1.NodeSelectorRequirement{ + Key: config.GetK8sPluginConfig().GpuPartitionSizeNodeLabel, + Operator: v1.NodeSelectorOpDoesNotExist, + } + } + if config.GetK8sPluginConfig().GpuUnpartitionedToleration != nil { + partitionSizeTol = config.GetK8sPluginConfig().GpuUnpartitionedToleration + } + case *core.GPUAccelerator_PartitionSize: + partitionSizeNsr = &v1.NodeSelectorRequirement{ + Key: config.GetK8sPluginConfig().GpuPartitionSizeNodeLabel, + Operator: v1.NodeSelectorOpIn, + Values: []string{p.PartitionSize}, + } + partitionSizeTol = &v1.Toleration{ + Key: config.GetK8sPluginConfig().GpuPartitionSizeNodeLabel, + Value: p.PartitionSize, + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + } + } + if partitionSizeNsr != nil { + AddRequiredNodeSelectorRequirements(podSpec.Affinity, *partitionSizeNsr) + } + if partitionSizeTol != nil { + podSpec.Tolerations = append(podSpec.Tolerations, *partitionSizeTol) + } } // UpdatePod updates the base pod spec used to execute tasks. This is configured with plugins and task metadata-specific options @@ -250,6 +390,18 @@ func ApplyFlytePodConfiguration(ctx context.Context, tCtx pluginsCore.TaskExecut return nil, nil, err } + // handling for extended resources + // Merge overrides with base extended resources + extendedResources := applyExtendedResourcesOverrides( + taskTemplate.GetExtendedResources(), + tCtx.TaskExecutionMetadata().GetOverrides().GetExtendedResources(), + ) + + // GPU accelerator + if extendedResources.GetGpuAccelerator() != nil { + ApplyGPUNodeSelectors(podSpec, extendedResources.GetGpuAccelerator()) + } + return podSpec, objectMeta, nil } diff --git a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper_test.go b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper_test.go index 63d3b767878..5c503239d0f 100644 --- a/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper_test.go +++ b/flyteplugins/go/tasks/pluginmachinery/flytek8s/pod_helper_test.go @@ -29,7 +29,7 @@ import ( metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" ) -func dummyTaskExecutionMetadata(resources *v1.ResourceRequirements) pluginsCore.TaskExecutionMetadata { +func dummyTaskExecutionMetadata(resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionMetadata { taskExecutionMetadata := &pluginsCoreMock.TaskExecutionMetadata{} taskExecutionMetadata.On("GetNamespace").Return("test-namespace") taskExecutionMetadata.On("GetAnnotations").Return(map[string]string{"annotation-1": "val1"}) @@ -54,6 +54,7 @@ func dummyTaskExecutionMetadata(resources *v1.ResourceRequirements) pluginsCore. to := &pluginsCoreMock.TaskOverrides{} to.On("GetResources").Return(resources) + to.On("GetExtendedResources").Return(extendedResources) taskExecutionMetadata.On("GetOverrides").Return(to) taskExecutionMetadata.On("IsInterruptible").Return(true) taskExecutionMetadata.OnGetPlatformResources().Return(&v1.ResourceRequirements{}) @@ -61,9 +62,8 @@ func dummyTaskExecutionMetadata(resources *v1.ResourceRequirements) pluginsCore. return taskExecutionMetadata } -func dummyTaskReader() pluginsCore.TaskReader { - taskReader := &pluginsCoreMock.TaskReader{} - task := &core.TaskTemplate{ +func dummyTaskTemplate() *core.TaskTemplate { + return &core.TaskTemplate{ Type: "test", Target: &core.TaskTemplate_Container{ Container: &core.Container{ @@ -72,8 +72,6 @@ func dummyTaskReader() pluginsCore.TaskReader { }, }, } - taskReader.On("Read", mock.Anything).Return(task, nil) - return taskReader } func dummyInputReader() io.InputReader { @@ -84,7 +82,7 @@ func dummyInputReader() io.InputReader { return inputReader } -func dummyExecContext(r *v1.ResourceRequirements) pluginsCore.TaskExecutionContext { +func dummyExecContext(taskTemplate *core.TaskTemplate, r *v1.ResourceRequirements, rm *core.ExtendedResources) pluginsCore.TaskExecutionContext { ow := &pluginsIOMock.OutputWriter{} ow.OnGetOutputPrefixPath().Return("") ow.OnGetRawOutputPrefix().Return("") @@ -92,10 +90,13 @@ func dummyExecContext(r *v1.ResourceRequirements) pluginsCore.TaskExecutionConte ow.OnGetPreviousCheckpointsPrefix().Return("/prev") tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(r)) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(r, rm)) tCtx.OnInputReader().Return(dummyInputReader()) - tCtx.OnTaskReader().Return(dummyTaskReader()) tCtx.OnOutputWriter().Return(ow) + + taskReader := &pluginsCoreMock.TaskReader{} + taskReader.On("Read", mock.Anything).Return(taskTemplate, nil) + tCtx.OnTaskReader().Return(taskReader) return tCtx } @@ -112,6 +113,226 @@ func TestPodSetup(t *testing.T) { t.Run("ToK8sPodInterruptible", toK8sPodInterruptible) } +func TestAddRequiredNodeSelectorRequirements(t *testing.T) { + t.Run("with empty node affinity", func(t *testing.T) { + affinity := v1.Affinity{} + nst := v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + } + AddRequiredNodeSelectorRequirements(&affinity, nst) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + }, + }, + }, + }, + affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + }) + + t.Run("with existing node affinity", func(t *testing.T) { + affinity := v1.Affinity{ + NodeAffinity: &v1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &v1.NodeSelector{ + NodeSelectorTerms: []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "required", + Operator: v1.NodeSelectorOpIn, + Values: []string{"required"}, + }, + }, + }, + }, + }, + PreferredDuringSchedulingIgnoredDuringExecution: []v1.PreferredSchedulingTerm{ + v1.PreferredSchedulingTerm{ + Weight: 1, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "preferred", + Operator: v1.NodeSelectorOpIn, + Values: []string{"preferred"}, + }, + }, + }, + }, + }, + }, + } + nst := v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + } + AddRequiredNodeSelectorRequirements(&affinity, nst) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "required", + Operator: v1.NodeSelectorOpIn, + Values: []string{"required"}, + }, + v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + }, + }, + }, + }, + affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.PreferredSchedulingTerm{ + v1.PreferredSchedulingTerm{ + Weight: 1, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "preferred", + Operator: v1.NodeSelectorOpIn, + Values: []string{"preferred"}, + }, + }, + }, + }, + }, + affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution, + ) + }) +} + +func TestAddPreferredNodeSelectorRequirements(t *testing.T) { + t.Run("with empty node affinity", func(t *testing.T) { + affinity := v1.Affinity{} + nst := v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + } + AddPreferredNodeSelectorRequirements(&affinity, 10, nst) + assert.EqualValues( + t, + []v1.PreferredSchedulingTerm{ + v1.PreferredSchedulingTerm{ + Weight: 10, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + }, + }, + }, + }, + }, + affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution, + ) + }) + + t.Run("with existing node affinity", func(t *testing.T) { + affinity := v1.Affinity{ + NodeAffinity: &v1.NodeAffinity{ + RequiredDuringSchedulingIgnoredDuringExecution: &v1.NodeSelector{ + NodeSelectorTerms: []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "required", + Operator: v1.NodeSelectorOpIn, + Values: []string{"required"}, + }, + }, + }, + }, + }, + PreferredDuringSchedulingIgnoredDuringExecution: []v1.PreferredSchedulingTerm{ + v1.PreferredSchedulingTerm{ + Weight: 1, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "preferred", + Operator: v1.NodeSelectorOpIn, + Values: []string{"preferred"}, + }, + }, + }, + }, + }, + }, + } + nst := v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + } + AddPreferredNodeSelectorRequirements(&affinity, 10, nst) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "required", + Operator: v1.NodeSelectorOpIn, + Values: []string{"required"}, + }, + }, + }, + }, + affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.PreferredSchedulingTerm{ + v1.PreferredSchedulingTerm{ + Weight: 1, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "preferred", + Operator: v1.NodeSelectorOpIn, + Values: []string{"preferred"}, + }, + }, + }, + }, + v1.PreferredSchedulingTerm{ + Weight: 10, + Preference: v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "new", + Operator: v1.NodeSelectorOpIn, + Values: []string{"new"}, + }, + }, + }, + }, + }, + affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution, + ) + }) +} + func TestApplyInterruptibleNodeAffinity(t *testing.T) { t.Run("WithInterruptibleNodeSelectorRequirement", func(t *testing.T) { podSpec := v1.PodSpec{} @@ -196,6 +417,282 @@ func TestApplyInterruptibleNodeAffinity(t *testing.T) { }) } +func TestApplyExtendedResourcesOverrides(t *testing.T) { + t4 := &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + } + partitionedA100 := &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + } + unpartitionedA100 := &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_Unpartitioned{ + Unpartitioned: true, + }, + }, + } + + t.Run("base and overrides are nil", func(t *testing.T) { + final := applyExtendedResourcesOverrides(nil, nil) + assert.NotNil(t, final) + }) + + t.Run("base is nil", func(t *testing.T) { + final := applyExtendedResourcesOverrides(nil, t4) + assert.EqualValues( + t, + t4.GetGpuAccelerator(), + final.GetGpuAccelerator(), + ) + }) + + t.Run("overrides is nil", func(t *testing.T) { + final := applyExtendedResourcesOverrides(t4, nil) + assert.EqualValues( + t, + t4.GetGpuAccelerator(), + final.GetGpuAccelerator(), + ) + }) + + t.Run("merging", func(t *testing.T) { + final := applyExtendedResourcesOverrides(partitionedA100, unpartitionedA100) + assert.EqualValues( + t, + unpartitionedA100.GetGpuAccelerator(), + final.GetGpuAccelerator(), + ) + }) +} + +func TestApplyGPUNodeSelectors(t *testing.T) { + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuResourceName: "nvidia.com/gpu", + GpuDeviceNodeLabel: "gpu-device", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + })) + + basePodSpec := &v1.PodSpec{ + Containers: []v1.Container{ + { + Resources: v1.ResourceRequirements{ + Limits: v1.ResourceList{ + "nvidia.com/gpu": resource.MustParse("1"), + }, + }, + }, + }, + } + + t.Run("without gpu resource", func(t *testing.T) { + podSpec := &v1.PodSpec{} + ApplyGPUNodeSelectors( + podSpec, + &core.GPUAccelerator{Device: "nvidia-tesla-a100"}, + ) + assert.Nil(t, podSpec.Affinity) + }) + + t.Run("with gpu device spec only", func(t *testing.T) { + podSpec := basePodSpec.DeepCopy() + ApplyGPUNodeSelectors( + podSpec, + &core.GPUAccelerator{Device: "nvidia-tesla-a100"}, + ) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-device", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + }, + }, + }, + podSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.Toleration{ + { + Key: "gpu-device", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + podSpec.Tolerations, + ) + }) + + t.Run("with gpu device and partition size spec", func(t *testing.T) { + podSpec := basePodSpec.DeepCopy() + ApplyGPUNodeSelectors( + podSpec, + &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + ) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-device", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + podSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.Toleration{ + { + Key: "gpu-device", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + podSpec.Tolerations, + ) + }) + + t.Run("with unpartitioned gpu device spec", func(t *testing.T) { + podSpec := basePodSpec.DeepCopy() + ApplyGPUNodeSelectors( + podSpec, + &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_Unpartitioned{ + Unpartitioned: true, + }, + }, + ) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-device", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpDoesNotExist, + }, + }, + }, + }, + podSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.Toleration{ + { + Key: "gpu-device", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + podSpec.Tolerations, + ) + }) + + t.Run("with unpartitioned gpu device spec with custom node selector and toleration", func(t *testing.T) { + gpuUnpartitionedNodeSelectorRequirement := v1.NodeSelectorRequirement{ + Key: "gpu-unpartitioned", + Operator: v1.NodeSelectorOpIn, + Values: []string{"true"}, + } + gpuUnpartitionedToleration := v1.Toleration{ + Key: "gpu-unpartitioned", + Value: "true", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + } + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuResourceName: "nvidia.com/gpu", + GpuDeviceNodeLabel: "gpu-device", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuUnpartitionedNodeSelectorRequirement: &gpuUnpartitionedNodeSelectorRequirement, + GpuUnpartitionedToleration: &gpuUnpartitionedToleration, + })) + + podSpec := basePodSpec.DeepCopy() + ApplyGPUNodeSelectors( + podSpec, + &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_Unpartitioned{ + Unpartitioned: true, + }, + }, + ) + assert.EqualValues( + t, + []v1.NodeSelectorTerm{ + v1.NodeSelectorTerm{ + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-device", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + gpuUnpartitionedNodeSelectorRequirement, + }, + }, + }, + podSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + []v1.Toleration{ + { + Key: "gpu-device", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + gpuUnpartitionedToleration, + }, + podSpec.Tolerations, + ) + }) +} + func updatePod(t *testing.T) { taskExecutionMetadata := dummyTaskExecutionMetadata(&v1.ResourceRequirements{ Limits: v1.ResourceList{ @@ -206,7 +703,7 @@ func updatePod(t *testing.T) { v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), }, - }) + }, nil) pod := &v1.Pod{ Spec: v1.PodSpec{ @@ -259,7 +756,7 @@ func updatePod(t *testing.T) { } func TestUpdatePodWithDefaultAffinityAndInterruptibleNodeSelectorRequirement(t *testing.T) { - taskExecutionMetadata := dummyTaskExecutionMetadata(&v1.ResourceRequirements{}) + taskExecutionMetadata := dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil) assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ DefaultAffinity: &v1.Affinity{ NodeAffinity: &v1.NodeAffinity{ @@ -313,7 +810,7 @@ func TestUpdatePodWithDefaultAffinityAndInterruptibleNodeSelectorRequirement(t * func toK8sPodInterruptible(t *testing.T) { ctx := context.TODO() - x := dummyExecContext(&v1.ResourceRequirements{ + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{ Limits: v1.ResourceList{ v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), @@ -323,7 +820,7 @@ func toK8sPodInterruptible(t *testing.T) { v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), }, - }) + }, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) @@ -380,7 +877,7 @@ func TestToK8sPod(t *testing.T) { op.On("GetRawOutputPrefix").Return(storage.DataReference("")) t.Run("WithGPU", func(t *testing.T) { - x := dummyExecContext(&v1.ResourceRequirements{ + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{ Limits: v1.ResourceList{ v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), @@ -390,7 +887,7 @@ func TestToK8sPod(t *testing.T) { v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), }, - }) + }, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) @@ -398,7 +895,7 @@ func TestToK8sPod(t *testing.T) { }) t.Run("NoGPU", func(t *testing.T) { - x := dummyExecContext(&v1.ResourceRequirements{ + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{ Limits: v1.ResourceList{ v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), @@ -407,7 +904,7 @@ func TestToK8sPod(t *testing.T) { v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), }, - }) + }, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) @@ -416,7 +913,7 @@ func TestToK8sPod(t *testing.T) { }) t.Run("Default toleration, selector, scheduler", func(t *testing.T) { - x := dummyExecContext(&v1.ResourceRequirements{ + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{ Limits: v1.ResourceList{ v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), @@ -425,7 +922,7 @@ func TestToK8sPod(t *testing.T) { v1.ResourceCPU: resource.MustParse("1024m"), v1.ResourceStorage: resource.MustParse("100M"), }, - }) + }, nil) assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ DefaultNodeSelector: map[string]string{ @@ -452,7 +949,7 @@ func TestToK8sPod(t *testing.T) { }, })) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) assert.NotNil(t, p.SecurityContext) @@ -464,7 +961,7 @@ func TestToK8sPod(t *testing.T) { assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ EnableHostNetworkingPod: &enabled, })) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) assert.True(t, p.HostNetwork) @@ -475,7 +972,7 @@ func TestToK8sPod(t *testing.T) { assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ EnableHostNetworkingPod: &enabled, })) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) assert.False(t, p.HostNetwork) @@ -483,7 +980,7 @@ func TestToK8sPod(t *testing.T) { t.Run("skipSettingHostNetwork", func(t *testing.T) { assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{})) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) assert.False(t, p.HostNetwork) @@ -517,7 +1014,7 @@ func TestToK8sPod(t *testing.T) { }, })) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) assert.NotNil(t, p.DNSConfig) @@ -538,7 +1035,7 @@ func TestToK8sPod(t *testing.T) { "foo": "bar", }, })) - x := dummyExecContext(&v1.ResourceRequirements{}) + x := dummyExecContext(dummyTaskTemplate(), &v1.ResourceRequirements{}, nil) p, _, _, err := ToK8sPodSpec(ctx, x) assert.NoError(t, err) for _, c := range p.Containers { @@ -553,6 +1050,129 @@ func TestToK8sPod(t *testing.T) { }) } +func TestToK8sPodExtendedResources(t *testing.T) { + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *v1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []v1.NodeSelectorTerm + expectedTol []v1.Toleration + }{ + { + "without overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + } + + for _, f := range fixtures { + t.Run(f.name, func(t *testing.T) { + taskTemplate := dummyTaskTemplate() + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyExecContext(taskTemplate, f.resources, f.extendedResourcesOverride) + p, _, _, err := ToK8sPodSpec(context.TODO(), taskContext) + assert.NoError(t, err) + + assert.EqualValues( + t, + f.expectedNsr, + p.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + p.Tolerations, + ) + }) + } +} + func TestDemystifyPending(t *testing.T) { assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ CreateContainerErrorGracePeriod: config1.Duration{ @@ -1089,7 +1709,7 @@ func TestGetPodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) // initialize PodTemplateStore @@ -1115,7 +1735,7 @@ func TestGetPodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) // initialize PodTemplateStore @@ -1142,7 +1762,7 @@ func TestGetPodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) // initialize PodTemplateStore @@ -1170,7 +1790,7 @@ func TestGetPodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) // initialize PodTemplateStore @@ -1212,7 +1832,7 @@ func TestMergeWithBasePodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) resultPodSpec, resultObjectMeta, err := MergeWithBasePodTemplate(context.TODO(), tCtx, &podSpec, &objectMeta, "foo") @@ -1266,7 +1886,7 @@ func TestMergeWithBasePodTemplate(t *testing.T) { taskReader.On("Read", mock.Anything).Return(task, nil) tCtx := &pluginsCoreMock.TaskExecutionContext{} - tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{})) + tCtx.OnTaskExecutionMetadata().Return(dummyTaskExecutionMetadata(&v1.ResourceRequirements{}, nil)) tCtx.OnTaskReader().Return(taskReader) resultPodSpec, resultObjectMeta, err := MergeWithBasePodTemplate(context.TODO(), tCtx, &podSpec, &objectMeta, "foo") diff --git a/flyteplugins/go/tasks/plugins/array/k8s/management_test.go b/flyteplugins/go/tasks/plugins/array/k8s/management_test.go index 6eafc26bbe5..5e1d37e0666 100644 --- a/flyteplugins/go/tasks/plugins/array/k8s/management_test.go +++ b/flyteplugins/go/tasks/plugins/array/k8s/management_test.go @@ -97,6 +97,7 @@ func getMockTaskExecutionContext(ctx context.Context, parallelism int) *mocks.Ta v1.ResourceCPU: resource.MustParse("10"), }, }) + overrides.OnGetExtendedResources().Return(nil) tMeta := &mocks.TaskExecutionMetadata{} tMeta.OnGetTaskExecutionID().Return(tID) diff --git a/flyteplugins/go/tasks/plugins/k8s/dask/dask_test.go b/flyteplugins/go/tasks/plugins/k8s/dask/dask_test.go index 4eff02279af..cfec0d3af45 100644 --- a/flyteplugins/go/tasks/plugins/k8s/dask/dask_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/dask/dask_test.go @@ -147,7 +147,7 @@ func dummyDaskTaskTemplate(customImage string, resources *core.Resources, podTem } } -func dummyDaskTaskContext(taskTemplate *core.TaskTemplate, resources *v1.ResourceRequirements, isInterruptible bool) pluginsCore.TaskExecutionContext { +func dummyDaskTaskContext(taskTemplate *core.TaskTemplate, resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources, isInterruptible bool) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} @@ -192,6 +192,7 @@ func dummyDaskTaskContext(taskTemplate *core.TaskTemplate, resources *v1.Resourc taskExecutionMetadata.OnGetNamespace().Return(defaultNamespace) overrides := &mocks.TaskOverrides{} overrides.OnGetResources().Return(resources) + overrides.OnGetExtendedResources().Return(extendedResources) taskExecutionMetadata.OnGetOverrides().Return(overrides) taskCtx.On("TaskExecutionMetadata").Return(taskExecutionMetadata) return taskCtx @@ -201,7 +202,7 @@ func TestBuildResourceDaskHappyPath(t *testing.T) { daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate("", nil, "") - taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, false) + taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, nil, false) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -312,7 +313,7 @@ func TestBuildResourceDaskCustomImages(t *testing.T) { daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate(customImage, nil, "") - taskContext := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, false) + taskContext := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, nil, false) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -345,7 +346,7 @@ func TestBuildResourceDaskDefaultResoureRequirements(t *testing.T) { daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate("", nil, "") - taskContext := dummyDaskTaskContext(taskTemplate, &flyteWorkflowResources, false) + taskContext := dummyDaskTaskContext(taskTemplate, &flyteWorkflowResources, nil, false) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -402,7 +403,7 @@ func TestBuildResourcesDaskCustomResoureRequirements(t *testing.T) { daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate("", &protobufResources, "") - taskContext := dummyDaskTaskContext(taskTemplate, &flyteWorkflowResources, false) + taskContext := dummyDaskTaskContext(taskTemplate, &flyteWorkflowResources, nil, false) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -457,7 +458,7 @@ func TestBuildResourceDaskInterruptible(t *testing.T) { daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate("", nil, "") - taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, true) + taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, nil, true) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -491,7 +492,7 @@ func TestBuildResouceDaskUsePodTemplate(t *testing.T) { flytek8s.DefaultPodTemplateStore.Store(podTemplate) daskResourceHandler := daskResourceHandler{} taskTemplate := dummyDaskTaskTemplate("", nil, podTemplateName) - taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, false) + taskContext := dummyDaskTaskContext(taskTemplate, &defaultResources, nil, false) r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) assert.Nil(t, err) assert.NotNil(t, r) @@ -506,6 +507,161 @@ func TestBuildResouceDaskUsePodTemplate(t *testing.T) { flytek8s.DefaultPodTemplateStore.Delete(podTemplate) } +func TestBuildResourceDaskExtendedResources(t *testing.T) { + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *v1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []v1.NodeSelectorTerm + expectedTol []v1.Toleration + }{ + { + "without overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + } + + for _, f := range fixtures { + t.Run(f.name, func(t *testing.T) { + taskTemplate := dummyDaskTaskTemplate("", nil, "") + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyDaskTaskContext(taskTemplate, f.resources, f.extendedResourcesOverride, false) + daskResourceHandler := daskResourceHandler{} + r, err := daskResourceHandler.BuildResource(context.TODO(), taskContext) + assert.Nil(t, err) + assert.NotNil(t, r) + daskJob, ok := r.(*daskAPI.DaskJob) + assert.True(t, ok) + + // Job pod + jobSpec := daskJob.Spec.Job.Spec + assert.EqualValues( + t, + f.expectedNsr, + jobSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + jobSpec.Tolerations, + ) + + // Scheduler + schedulerSpec := daskJob.Spec.Cluster.Spec.Scheduler.Spec + assert.EqualValues( + t, + f.expectedNsr, + schedulerSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + schedulerSpec.Tolerations, + ) + + // Default Workers + workerSpec := daskJob.Spec.Cluster.Spec.Worker.Spec + assert.EqualValues( + t, + f.expectedNsr, + workerSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + workerSpec.Tolerations, + ) + }) + } +} + func TestGetPropertiesDask(t *testing.T) { daskResourceHandler := daskResourceHandler{} expected := k8s.PluginProperties{} @@ -522,7 +678,7 @@ func TestBuildIdentityResourceDask(t *testing.T) { } taskTemplate := dummyDaskTaskTemplate("", nil, "") - taskContext := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, false) + taskContext := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, nil, false) identityResources, err := daskResourceHandler.BuildIdentityResource(context.TODO(), taskContext.TaskExecutionMetadata()) if err != nil { panic(err) @@ -535,7 +691,7 @@ func TestGetTaskPhaseDask(t *testing.T) { ctx := context.TODO() taskTemplate := dummyDaskTaskTemplate("", nil, "") - taskCtx := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, false) + taskCtx := dummyDaskTaskContext(taskTemplate, &v1.ResourceRequirements{}, nil, false) taskPhase, err := daskResourceHandler.GetTaskPhase(ctx, taskCtx, dummyDaskJob("")) assert.NoError(t, err) diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go index b7c66b08f60..bda0faff75e 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/mpi/mpi_test.go @@ -13,6 +13,8 @@ import ( "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" + "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" + flytek8sConfig "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" pluginIOMocks "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/io/mocks" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" @@ -119,7 +121,7 @@ func dummyMPITaskTemplate(id string, args ...interface{}) *core.TaskTemplate { } } -func dummyMPITaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecutionContext { +func dummyMPITaskContext(taskTemplate *core.TaskTemplate, resources *corev1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} inputReader.OnGetInputPrefixPath().Return("/input/prefix") @@ -151,8 +153,9 @@ func dummyMPITaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecut }) tID.OnGetGeneratedName().Return("some-acceptable-name") - resources := &mocks.TaskOverrides{} - resources.OnGetResources().Return(resourceRequirements) + overrides := &mocks.TaskOverrides{} + overrides.OnGetResources().Return(resources) + overrides.OnGetExtendedResources().Return(extendedResources) taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.OnGetTaskExecutionID().Return(tID) @@ -164,7 +167,7 @@ func dummyMPITaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecut Name: "blah", }) taskExecutionMetadata.OnIsInterruptible().Return(true) - taskExecutionMetadata.OnGetOverrides().Return(resources) + taskExecutionMetadata.OnGetOverrides().Return(overrides) taskExecutionMetadata.OnGetK8sServiceAccount().Return(serviceAccount) taskExecutionMetadata.OnGetPlatformResources().Return(&corev1.ResourceRequirements{}) taskExecutionMetadata.OnGetEnvironmentVariables().Return(nil) @@ -274,7 +277,7 @@ func dummyMPIJobResource(mpiResourceHandler mpiOperatorResourceHandler, mpiObj := dummyMPICustomObj(workers, launcher, slots) taskTemplate := dummyMPITaskTemplate(mpiID, mpiObj) - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) if err != nil { panic(err) } @@ -301,7 +304,7 @@ func TestBuildResourceMPI(t *testing.T) { mpiObj := dummyMPICustomObj(100, 50, 1) taskTemplate := dummyMPITaskTemplate(mpiID2, mpiObj) - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -337,19 +340,19 @@ func TestBuildResourceMPIForWrongInput(t *testing.T) { mpiObj := dummyMPICustomObj(0, 0, 1) taskTemplate := dummyMPITaskTemplate(mpiID, mpiObj) - _, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + _, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) assert.Error(t, err) mpiObj = dummyMPICustomObj(1, 0, 1) taskTemplate = dummyMPITaskTemplate(mpiID2, mpiObj) - _, err = mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + _, err = mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) assert.Error(t, err) mpiObj = dummyMPICustomObj(1, 1, 1) taskTemplate = dummyMPITaskTemplate(mpiID2, mpiObj) - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) app, ok := resource.(*kubeflowv1.MPIJob) assert.Nil(t, err) assert.Equal(t, true, ok) @@ -357,6 +360,136 @@ func TestBuildResourceMPIForWrongInput(t *testing.T) { assert.Equal(t, []string{}, app.Spec.MPIReplicaSpecs[kubeflowv1.MPIJobReplicaTypeWorker].Template.Spec.Containers[0].Args) } +func TestBuildResourceMPIExtendedResources(t *testing.T) { + assert.NoError(t, flytek8sConfig.SetK8sPluginConfig(&flytek8sConfig.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *corev1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []corev1.NodeSelectorTerm + expectedTol []corev1.Toleration + }{ + { + "without overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + corev1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + } + + for _, f := range fixtures { + t.Run(f.name, func(t *testing.T) { + mpiObj := dummyMPICustomObj(100, 50, 1) + taskTemplate := dummyMPITaskTemplate(mpiID2, mpiObj) + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyMPITaskContext(taskTemplate, f.resources, f.extendedResourcesOverride) + mpiResourceHandler := mpiOperatorResourceHandler{} + r, err := mpiResourceHandler.BuildResource(context.TODO(), taskContext) + assert.Nil(t, err) + assert.NotNil(t, r) + mpiJob, ok := r.(*kubeflowv1.MPIJob) + assert.True(t, ok) + + for _, replicaSpec := range mpiJob.Spec.MPIReplicaSpecs { + assert.EqualValues( + t, + f.expectedNsr, + replicaSpec.Template.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + replicaSpec.Template.Spec.Tolerations, + ) + } + }) + } +} + func TestGetTaskPhase(t *testing.T) { mpiResourceHandler := mpiOperatorResourceHandler{} ctx := context.TODO() @@ -365,7 +498,7 @@ func TestGetTaskPhase(t *testing.T) { return dummyMPIJobResource(mpiResourceHandler, 2, 1, 1, conditionType) } - taskCtx := dummyMPITaskContext(dummyMPITaskTemplate("", dummyMPICustomObj(2, 1, 1))) + taskCtx := dummyMPITaskContext(dummyMPITaskTemplate("", dummyMPICustomObj(2, 1, 1)), resourceRequirements, nil) taskPhase, err := mpiResourceHandler.GetTaskPhase(ctx, taskCtx, dummyMPIJobResourceCreator(mpiOp.JobCreated)) assert.NoError(t, err) assert.Equal(t, pluginsCore.PhaseQueued, taskPhase.Phase()) @@ -409,7 +542,7 @@ func TestGetLogs(t *testing.T) { mpiResourceHandler := mpiOperatorResourceHandler{} mpiJob := dummyMPIJobResource(mpiResourceHandler, workers, launcher, slots, mpiOp.JobRunning) - taskCtx := dummyMPITaskContext(dummyMPITaskTemplate("", dummyMPICustomObj(workers, launcher, slots))) + taskCtx := dummyMPITaskContext(dummyMPITaskTemplate("", dummyMPICustomObj(workers, launcher, slots)), resourceRequirements, nil) jobLogs, err := common.GetLogs(taskCtx, common.MPITaskType, mpiJob.ObjectMeta, false, workers, launcher, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) @@ -442,7 +575,7 @@ func TestReplicaCounts(t *testing.T) { mpiObj := dummyMPICustomObj(test.workerReplicaCount, test.launcherReplicaCount, 1) taskTemplate := dummyMPITaskTemplate(mpiID2, mpiObj) - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) if test.expectError { assert.Error(t, err) assert.Nil(t, resource) @@ -520,7 +653,7 @@ func TestBuildResourceMPIV1(t *testing.T) { taskTemplate := dummyMPITaskTemplate(mpiID2, taskConfig) taskTemplate.TaskTypeVersion = 1 - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -568,7 +701,7 @@ func TestBuildResourceMPIV1WithOnlyWorkerReplica(t *testing.T) { taskTemplate := dummyMPITaskTemplate(mpiID2, taskConfig) taskTemplate.TaskTypeVersion = 1 - resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate)) + resource, err := mpiResourceHandler.BuildResource(context.TODO(), dummyMPITaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go index 9407294971d..f980c897417 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/pytorch/pytorch_test.go @@ -10,6 +10,7 @@ import ( "github.com/flyteorg/flyte/flyteplugins/go/tasks/logs" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" + flytek8sConfig "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" commonOp "github.com/kubeflow/common/pkg/apis/common/v1" corev1 "k8s.io/api/core/v1" @@ -128,7 +129,7 @@ func dummyPytorchTaskTemplate(id string, args ...interface{}) *core.TaskTemplate } } -func dummyPytorchTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecutionContext { +func dummyPytorchTaskContext(taskTemplate *core.TaskTemplate, resources *corev1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} inputReader.OnGetInputPrefixPath().Return("/input/prefix") @@ -160,19 +161,9 @@ func dummyPytorchTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskEx }) tID.OnGetGeneratedName().Return("some-acceptable-name") - resources := &mocks.TaskOverrides{} - resources.OnGetResources().Return(&corev1.ResourceRequirements{ - Limits: corev1.ResourceList{ - corev1.ResourceCPU: resource.MustParse("1000m"), - corev1.ResourceMemory: resource.MustParse("1Gi"), - flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), - }, - Requests: corev1.ResourceList{ - corev1.ResourceCPU: resource.MustParse("100m"), - corev1.ResourceMemory: resource.MustParse("512Mi"), - flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), - }, - }) + overrides := &mocks.TaskOverrides{} + overrides.OnGetResources().Return(resources) + overrides.OnGetExtendedResources().Return(extendedResources) taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.OnGetTaskExecutionID().Return(tID) @@ -184,7 +175,7 @@ func dummyPytorchTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskEx Name: "blah", }) taskExecutionMetadata.OnIsInterruptible().Return(true) - taskExecutionMetadata.OnGetOverrides().Return(resources) + taskExecutionMetadata.OnGetOverrides().Return(overrides) taskExecutionMetadata.OnGetK8sServiceAccount().Return(serviceAccount) taskExecutionMetadata.OnGetPlatformResources().Return(&corev1.ResourceRequirements{}) taskExecutionMetadata.OnGetEnvironmentVariables().Return(nil) @@ -293,7 +284,7 @@ func dummyPytorchJobResource(pytorchResourceHandler pytorchOperatorResourceHandl ptObj := dummyPytorchCustomObj(workers) taskTemplate := dummyPytorchTaskTemplate("job1", ptObj) - resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) if err != nil { panic(err) } @@ -321,7 +312,7 @@ func TestBuildResourcePytorchElastic(t *testing.T) { ptObj := dummyElasticPytorchCustomObj(2, plugins.ElasticConfig{MinReplicas: 1, MaxReplicas: 2, NprocPerNode: 4, RdzvBackend: "c10d"}) taskTemplate := dummyPytorchTaskTemplate("job2", ptObj) - resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -337,24 +328,22 @@ func TestBuildResourcePytorchElastic(t *testing.T) { assert.Equal(t, 1, len(pytorchJob.Spec.PyTorchReplicaSpecs)) assert.Contains(t, pytorchJob.Spec.PyTorchReplicaSpecs, kubeflowv1.PyTorchJobReplicaTypeWorker) - var hasContainerWithDefaultPytorchName = false - - for _, container := range pytorchJob.Spec.PyTorchReplicaSpecs[kubeflowv1.PyTorchJobReplicaTypeWorker].Template.Spec.Containers { - if container.Name == kubeflowv1.PytorchJobDefaultContainerName { - hasContainerWithDefaultPytorchName = true + for _, replicaSpec := range pytorchJob.Spec.PyTorchReplicaSpecs { + var hasContainerWithDefaultPytorchName = false + podSpec := replicaSpec.Template.Spec + for _, container := range podSpec.Containers { + if container.Name == kubeflowv1.PytorchJobDefaultContainerName { + hasContainerWithDefaultPytorchName = true + } } - } - assert.True(t, hasContainerWithDefaultPytorchName) + assert.True(t, hasContainerWithDefaultPytorchName) - // verify TaskExecutionMetadata labels and annotations are copied to the PyTorchJob - for k, v := range dummyAnnotations { - for _, replicaSpec := range pytorchJob.Spec.PyTorchReplicaSpecs { + // verify TaskExecutionMetadata labels and annotations are copied to the PyTorchJob + for k, v := range dummyAnnotations { assert.Equal(t, v, replicaSpec.Template.ObjectMeta.Annotations[k]) } - } - for k, v := range dummyLabels { - for _, replicaSpec := range pytorchJob.Spec.PyTorchReplicaSpecs { + for k, v := range dummyLabels { assert.Equal(t, v, replicaSpec.Template.ObjectMeta.Labels[k]) } } @@ -366,7 +355,7 @@ func TestBuildResourcePytorch(t *testing.T) { ptObj := dummyPytorchCustomObj(100) taskTemplate := dummyPytorchTaskTemplate("job3", ptObj) - res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, res) @@ -389,7 +378,6 @@ func TestBuildResourcePytorch(t *testing.T) { for _, replicaSpec := range pytorchJob.Spec.PyTorchReplicaSpecs { var hasContainerWithDefaultPytorchName = false - for _, container := range replicaSpec.Template.Spec.Containers { if container.Name == kubeflowv1.PytorchJobDefaultContainerName { hasContainerWithDefaultPytorchName = true @@ -403,6 +391,151 @@ func TestBuildResourcePytorch(t *testing.T) { } } +func TestBuildResourcePytorchExtendedResources(t *testing.T) { + assert.NoError(t, flytek8sConfig.SetK8sPluginConfig(&flytek8sConfig.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *corev1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []corev1.NodeSelectorTerm + expectedTol []corev1.Toleration + }{ + { + "without overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + corev1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + } + + testConfigs := []struct { + name string + plugin *plugins.DistributedPyTorchTrainingTask + }{ + { + "pytorch", + dummyPytorchCustomObj(100), + }, + { + "elastic pytorch", + dummyElasticPytorchCustomObj(2, plugins.ElasticConfig{MinReplicas: 1, MaxReplicas: 2, NprocPerNode: 4, RdzvBackend: "c10d"}), + }, + } + + for _, tCfg := range testConfigs { + for _, f := range fixtures { + t.Run(tCfg.name+" "+f.name, func(t *testing.T) { + taskTemplate := dummyPytorchTaskTemplate("job", tCfg.plugin) + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyPytorchTaskContext(taskTemplate, f.resources, f.extendedResourcesOverride) + pytorchResourceHandler := pytorchOperatorResourceHandler{} + r, err := pytorchResourceHandler.BuildResource(context.TODO(), taskContext) + assert.NoError(t, err) + assert.NotNil(t, r) + pytorchJob, ok := r.(*kubeflowv1.PyTorchJob) + assert.True(t, ok) + + for _, replicaSpec := range pytorchJob.Spec.PyTorchReplicaSpecs { + assert.EqualValues( + t, + f.expectedNsr, + replicaSpec.Template.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + replicaSpec.Template.Spec.Tolerations, + ) + } + }) + } + } +} + func TestGetTaskPhase(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} ctx := context.TODO() @@ -411,7 +544,7 @@ func TestGetTaskPhase(t *testing.T) { return dummyPytorchJobResource(pytorchResourceHandler, 2, conditionType) } - taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(2))) + taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(2)), resourceRequirements, nil) taskPhase, err := pytorchResourceHandler.GetTaskPhase(ctx, taskCtx, dummyPytorchJobResourceCreator(commonOp.JobCreated)) assert.NoError(t, err) assert.Equal(t, pluginsCore.PhaseQueued, taskPhase.Phase()) @@ -454,7 +587,7 @@ func TestGetLogs(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} pytorchJob := dummyPytorchJobResource(pytorchResourceHandler, workers, commonOp.JobRunning) - taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers))) + taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers)), resourceRequirements, nil) jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0, 0) assert.NoError(t, err) assert.Equal(t, 3, len(jobLogs)) @@ -474,7 +607,7 @@ func TestGetLogsElastic(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} pytorchJob := dummyPytorchJobResource(pytorchResourceHandler, workers, commonOp.JobRunning) - taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers))) + taskCtx := dummyPytorchTaskContext(dummyPytorchTaskTemplate("", dummyPytorchCustomObj(workers)), resourceRequirements, nil) jobLogs, err := common.GetLogs(taskCtx, common.PytorchTaskType, pytorchJob.ObjectMeta, hasMaster, workers, 0, 0, 0) assert.NoError(t, err) assert.Equal(t, 2, len(jobLogs)) @@ -505,7 +638,7 @@ func TestReplicaCounts(t *testing.T) { ptObj := dummyPytorchCustomObj(test.workerReplicaCount) taskTemplate := dummyPytorchTaskTemplate("the job", ptObj) - res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) if test.expectError { assert.Error(t, err) assert.Nil(t, res) @@ -579,7 +712,7 @@ func TestBuildResourcePytorchV1(t *testing.T) { taskTemplate := dummyPytorchTaskTemplate("job4", taskConfig) taskTemplate.TaskTypeVersion = 1 - res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, res) @@ -623,7 +756,7 @@ func TestBuildResourcePytorchV1WithRunPolicy(t *testing.T) { taskTemplate := dummyPytorchTaskTemplate("job5", taskConfig) taskTemplate.TaskTypeVersion = 1 - res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, res) @@ -679,7 +812,7 @@ func TestBuildResourcePytorchV1WithOnlyWorkerSpec(t *testing.T) { taskTemplate := dummyPytorchTaskTemplate("job5", taskConfig) taskTemplate.TaskTypeVersion = 1 - res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + res, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, res) @@ -712,7 +845,7 @@ func TestBuildResourcePytorchV1WithElastic(t *testing.T) { taskTemplate.TaskTypeVersion = 1 pytorchResourceHandler := pytorchOperatorResourceHandler{} - resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + resource, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -748,7 +881,7 @@ func TestBuildResourcePytorchV1WithZeroWorker(t *testing.T) { pytorchResourceHandler := pytorchOperatorResourceHandler{} taskTemplate := dummyPytorchTaskTemplate("job5", taskConfig) taskTemplate.TaskTypeVersion = 1 - _, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate)) + _, err := pytorchResourceHandler.BuildResource(context.TODO(), dummyPytorchTaskContext(taskTemplate, resourceRequirements, nil)) assert.Error(t, err) } diff --git a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go index 764be381811..ee5e97db16a 100644 --- a/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/kfoperators/tensorflow/tensorflow_test.go @@ -18,6 +18,7 @@ import ( "github.com/stretchr/testify/mock" pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" + flytek8sConfig "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/utils" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" @@ -122,7 +123,7 @@ func dummyTensorFlowTaskTemplate(id string, args ...interface{}) *core.TaskTempl } } -func dummyTensorFlowTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecutionContext { +func dummyTensorFlowTaskContext(taskTemplate *core.TaskTemplate, resources *corev1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} inputReader.OnGetInputPrefixPath().Return("/input/prefix") @@ -154,8 +155,9 @@ func dummyTensorFlowTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.Tas }) tID.OnGetGeneratedName().Return("some-acceptable-name") - resources := &mocks.TaskOverrides{} - resources.OnGetResources().Return(resourceRequirements) + overrides := &mocks.TaskOverrides{} + overrides.OnGetResources().Return(resources) + overrides.OnGetExtendedResources().Return(extendedResources) taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.OnGetTaskExecutionID().Return(tID) @@ -167,7 +169,7 @@ func dummyTensorFlowTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.Tas Name: "blah", }) taskExecutionMetadata.OnIsInterruptible().Return(true) - taskExecutionMetadata.OnGetOverrides().Return(resources) + taskExecutionMetadata.OnGetOverrides().Return(overrides) taskExecutionMetadata.OnGetK8sServiceAccount().Return(serviceAccount) taskExecutionMetadata.OnGetPlatformResources().Return(&corev1.ResourceRequirements{}) taskExecutionMetadata.OnGetEnvironmentVariables().Return(nil) @@ -277,7 +279,7 @@ func dummyTensorFlowJobResource(tensorflowResourceHandler tensorflowOperatorReso tfObj := dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas, evaluatorReplicas) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) - resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) + resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate, resourceRequirements, nil)) if err != nil { panic(err) } @@ -304,7 +306,7 @@ func TestBuildResourceTensorFlow(t *testing.T) { tfObj := dummyTensorFlowCustomObj(100, 50, 1, 1) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) - resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) + resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -329,8 +331,8 @@ func TestBuildResourceTensorFlow(t *testing.T) { for _, replicaSpec := range tensorflowJob.Spec.TFReplicaSpecs { var hasContainerWithDefaultTensorFlowName = false - - for _, container := range replicaSpec.Template.Spec.Containers { + podSpec := replicaSpec.Template.Spec + for _, container := range podSpec.Containers { if container.Name == kubeflowv1.TFJobDefaultContainerName { hasContainerWithDefaultTensorFlowName = true } @@ -343,6 +345,161 @@ func TestBuildResourceTensorFlow(t *testing.T) { } } +func TestBuildResourceTensorFlowExtendedResources(t *testing.T) { + assert.NoError(t, flytek8sConfig.SetK8sPluginConfig(&flytek8sConfig.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *corev1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []corev1.NodeSelectorTerm + expectedTol []corev1.Toleration + }{ + { + "without overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + corev1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + } + + v0TaskTemplate := dummyTensorFlowTaskTemplate("v0", dummyTensorFlowCustomObj(100, 50, 1, 1)) + v1TaskTemplate := dummyTensorFlowTaskTemplate("v1", &kfplugins.DistributedTensorflowTrainingTask{ + ChiefReplicas: &kfplugins.DistributedTensorflowTrainingReplicaSpec{ + Replicas: 1, + }, + WorkerReplicas: &kfplugins.DistributedTensorflowTrainingReplicaSpec{ + Replicas: 100, + }, + PsReplicas: &kfplugins.DistributedTensorflowTrainingReplicaSpec{ + Replicas: 50, + }, + EvaluatorReplicas: &kfplugins.DistributedTensorflowTrainingReplicaSpec{ + Replicas: 1, + }, + }) + v1TaskTemplate.TaskTypeVersion = 1 + testConfigs := []struct { + name string + taskTemplate *core.TaskTemplate + }{ + {"v0", v0TaskTemplate}, + {"v1", v1TaskTemplate}, + } + + for _, tCfg := range testConfigs { + for _, f := range fixtures { + t.Run(tCfg.name+" "+f.name, func(t *testing.T) { + taskTemplate := *tCfg.taskTemplate + taskTemplate.ExtendedResources = f.extendedResourcesBase + tensorflowResourceHandler := tensorflowOperatorResourceHandler{} + taskContext := dummyTensorFlowTaskContext(&taskTemplate, f.resources, f.extendedResourcesOverride) + r, err := tensorflowResourceHandler.BuildResource(context.TODO(), taskContext) + assert.NoError(t, err) + assert.NotNil(t, r) + tensorflowJob, ok := r.(*kubeflowv1.TFJob) + assert.True(t, ok) + + for _, replicaSpec := range tensorflowJob.Spec.TFReplicaSpecs { + assert.EqualValues( + t, + f.expectedNsr, + replicaSpec.Template.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + replicaSpec.Template.Spec.Tolerations, + ) + } + }) + } + } +} + func TestGetTaskPhase(t *testing.T) { tensorflowResourceHandler := tensorflowOperatorResourceHandler{} ctx := context.TODO() @@ -351,7 +508,7 @@ func TestGetTaskPhase(t *testing.T) { return dummyTensorFlowJobResource(tensorflowResourceHandler, 2, 1, 1, 1, conditionType) } - taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(2, 1, 1, 1))) + taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(2, 1, 1, 1)), resourceRequirements, nil) taskPhase, err := tensorflowResourceHandler.GetTaskPhase(ctx, taskCtx, dummyTensorFlowJobResourceCreator(commonOp.JobCreated)) assert.NoError(t, err) assert.Equal(t, pluginsCore.PhaseQueued, taskPhase.Phase()) @@ -396,7 +553,7 @@ func TestGetLogs(t *testing.T) { tensorflowResourceHandler := tensorflowOperatorResourceHandler{} tensorFlowJob := dummyTensorFlowJobResource(tensorflowResourceHandler, workers, psReplicas, chiefReplicas, evaluatorReplicas, commonOp.JobRunning) - taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas, evaluatorReplicas))) + taskCtx := dummyTensorFlowTaskContext(dummyTensorFlowTaskTemplate("", dummyTensorFlowCustomObj(workers, psReplicas, chiefReplicas, evaluatorReplicas)), resourceRequirements, nil) jobLogs, err := common.GetLogs(taskCtx, common.TensorflowTaskType, tensorFlowJob.ObjectMeta, false, workers, psReplicas, chiefReplicas, evaluatorReplicas) assert.NoError(t, err) @@ -443,7 +600,7 @@ func TestReplicaCounts(t *testing.T) { tfObj := dummyTensorFlowCustomObj(test.workerReplicaCount, test.psReplicaCount, test.chiefReplicaCount, test.evaluatorReplicaCount) taskTemplate := dummyTensorFlowTaskTemplate("the job", tfObj) - resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) + resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate, resourceRequirements, nil)) if test.expectError { assert.Error(t, err) assert.Nil(t, resource) @@ -575,7 +732,7 @@ func TestBuildResourceTensorFlowV1(t *testing.T) { taskTemplate := dummyTensorFlowTaskTemplate("v1", taskConfig) taskTemplate.TaskTypeVersion = 1 - resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) + resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) @@ -637,7 +794,7 @@ func TestBuildResourceTensorFlowV1WithOnlyWorker(t *testing.T) { taskTemplate := dummyTensorFlowTaskTemplate("v1 with only worker replica", taskConfig) taskTemplate.TaskTypeVersion = 1 - resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate)) + resource, err := tensorflowResourceHandler.BuildResource(context.TODO(), dummyTensorFlowTaskContext(taskTemplate, resourceRequirements, nil)) assert.NoError(t, err) assert.NotNil(t, resource) diff --git a/flyteplugins/go/tasks/plugins/k8s/pod/container_test.go b/flyteplugins/go/tasks/plugins/k8s/pod/container_test.go index 0e1a2876da3..286c2673d71 100644 --- a/flyteplugins/go/tasks/plugins/k8s/pod/container_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/pod/container_test.go @@ -10,6 +10,7 @@ import ( pluginsCore "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core" pluginsCoreMock "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/core/mocks" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s" + flytek8sConfig "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/flytek8s/config" pluginsIOMock "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/io/mocks" "github.com/flyteorg/flyte/flyteplugins/go/tasks/pluginmachinery/k8s" @@ -30,7 +31,19 @@ var containerResourceRequirements = &v1.ResourceRequirements{ }, } -func dummyContainerTaskMetadata(resources *v1.ResourceRequirements) pluginsCore.TaskExecutionMetadata { +func dummyContainerTaskTemplate(command []string, args []string) *core.TaskTemplate { + return &core.TaskTemplate{ + Type: "test", + Target: &core.TaskTemplate_Container{ + Container: &core.Container{ + Command: command, + Args: args, + }, + }, + } +} + +func dummyContainerTaskMetadata(resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionMetadata { taskMetadata := &pluginsCoreMock.TaskExecutionMetadata{} taskMetadata.On("GetNamespace").Return("test-namespace") taskMetadata.On("GetAnnotations").Return(map[string]string{"annotation-1": "val1"}) @@ -64,24 +77,15 @@ func dummyContainerTaskMetadata(resources *v1.ResourceRequirements) pluginsCore. to := &pluginsCoreMock.TaskOverrides{} to.On("GetResources").Return(resources) + to.On("GetExtendedResources").Return(extendedResources) taskMetadata.On("GetOverrides").Return(to) taskMetadata.On("IsInterruptible").Return(true) taskMetadata.On("GetEnvironmentVariables").Return(nil) return taskMetadata } -func dummyContainerTaskContext(resources *v1.ResourceRequirements, command []string, args []string) pluginsCore.TaskExecutionContext { - task := &core.TaskTemplate{ - Type: "test", - Target: &core.TaskTemplate_Container{ - Container: &core.Container{ - Command: command, - Args: args, - }, - }, - } - - dummyTaskMetadata := dummyContainerTaskMetadata(resources) +func dummyContainerTaskContext(taskTemplate *core.TaskTemplate, resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { + dummyTaskMetadata := dummyContainerTaskMetadata(resources, extendedResources) taskCtx := &pluginsCoreMock.TaskExecutionContext{} inputReader := &pluginsIOMock.InputReader{} inputReader.OnGetInputPrefixPath().Return("test-data-reference") @@ -99,7 +103,7 @@ func dummyContainerTaskContext(resources *v1.ResourceRequirements, command []str taskCtx.OnOutputWriter().Return(outputReader) taskReader := &pluginsCoreMock.TaskReader{} - taskReader.OnReadMatch(mock.Anything).Return(task, nil) + taskReader.OnReadMatch(mock.Anything).Return(taskTemplate, nil) taskCtx.OnTaskReader().Return(taskReader) taskCtx.OnTaskExecutionMetadata().Return(dummyTaskMetadata) @@ -124,7 +128,8 @@ func TestContainerTaskExecutor_BuildIdentityResource(t *testing.T) { func TestContainerTaskExecutor_BuildResource(t *testing.T) { command := []string{"command"} args := []string{"{{.Input}}"} - taskCtx := dummyContainerTaskContext(containerResourceRequirements, command, args) + taskTemplate := dummyContainerTaskTemplate(command, args) + taskCtx := dummyContainerTaskContext(taskTemplate, containerResourceRequirements, nil) r, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.NoError(t, err) @@ -145,10 +150,137 @@ func TestContainerTaskExecutor_BuildResource(t *testing.T) { assert.Equal(t, "service-account", j.Spec.ServiceAccountName) } +func TestContainerTaskExecutor_BuildResource_ExtendedResources(t *testing.T) { + assert.NoError(t, flytek8sConfig.SetK8sPluginConfig(&flytek8sConfig.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *v1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []v1.NodeSelectorTerm + expectedTol []v1.Toleration + }{ + { + "without overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + } + + for _, f := range fixtures { + t.Run(f.name, func(t *testing.T) { + taskTemplate := dummyContainerTaskTemplate([]string{"command"}, []string{"{{.Input}}"}) + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyContainerTaskContext(taskTemplate, f.resources, f.extendedResourcesOverride) + r, err := DefaultPodPlugin.BuildResource(context.TODO(), taskContext) + assert.Nil(t, err) + assert.NotNil(t, r) + pod, ok := r.(*v1.Pod) + assert.True(t, ok) + + assert.EqualValues( + t, + f.expectedNsr, + pod.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + pod.Spec.Tolerations, + ) + }) + } +} + func TestContainerTaskExecutor_GetTaskStatus(t *testing.T) { command := []string{"command"} args := []string{"{{.Input}}"} - taskCtx := dummyContainerTaskContext(containerResourceRequirements, command, args) + taskTemplate := dummyContainerTaskTemplate(command, args) + taskCtx := dummyContainerTaskContext(taskTemplate, containerResourceRequirements, nil) j := &v1.Pod{ Status: v1.PodStatus{}, @@ -236,7 +368,8 @@ func TestContainerTaskExecutor_GetProperties(t *testing.T) { func TestContainerTaskExecutor_GetTaskStatus_InvalidImageName(t *testing.T) { command := []string{"command"} args := []string{"{{.Input}}"} - taskCtx := dummyContainerTaskContext(containerResourceRequirements, command, args) + taskTemplate := dummyContainerTaskTemplate(command, args) + taskCtx := dummyContainerTaskContext(taskTemplate, containerResourceRequirements, nil) ctx := context.TODO() reason := "InvalidImageName" diff --git a/flyteplugins/go/tasks/plugins/k8s/pod/sidecar_test.go b/flyteplugins/go/tasks/plugins/k8s/pod/sidecar_test.go index bf389de09a4..bbc797065ce 100644 --- a/flyteplugins/go/tasks/plugins/k8s/pod/sidecar_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/pod/sidecar_test.go @@ -58,7 +58,7 @@ func getSidecarTaskTemplateForTest(sideCarJob sidecarJob) *core.TaskTemplate { } } -func dummySidecarTaskMetadata(resources *v1.ResourceRequirements) pluginsCore.TaskExecutionMetadata { +func dummySidecarTaskMetadata(resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionMetadata { taskMetadata := &pluginsCoreMock.TaskExecutionMetadata{} taskMetadata.On("GetNamespace").Return("test-namespace") taskMetadata.On("GetAnnotations").Return(map[string]string{"annotation-1": "val1"}) @@ -92,15 +92,16 @@ func dummySidecarTaskMetadata(resources *v1.ResourceRequirements) pluginsCore.Ta to := &pluginsCoreMock.TaskOverrides{} to.On("GetResources").Return(resources) + to.On("GetExtendedResources").Return(extendedResources) taskMetadata.On("GetOverrides").Return(to) taskMetadata.On("GetEnvironmentVariables").Return(nil) return taskMetadata } -func getDummySidecarTaskContext(taskTemplate *core.TaskTemplate, resources *v1.ResourceRequirements) pluginsCore.TaskExecutionContext { +func getDummySidecarTaskContext(taskTemplate *core.TaskTemplate, resources *v1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { taskCtx := &pluginsCoreMock.TaskExecutionContext{} - dummyTaskMetadata := dummySidecarTaskMetadata(resources) + dummyTaskMetadata := dummySidecarTaskMetadata(resources, extendedResources) inputReader := &pluginsIOMock.InputReader{} inputReader.OnGetInputPrefixPath().Return("test-data-prefix") inputReader.OnGetInputPath().Return("test-data-reference") @@ -248,7 +249,7 @@ func TestBuildSidecarResource_TaskType2(t *testing.T) { DefaultMemoryRequest: resource.MustParse("1024Mi"), GpuResourceName: ResourceNvidiaGPU, })) - taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) res, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.Nil(t, err) assert.EqualValues(t, map[string]string{ @@ -309,7 +310,7 @@ func TestBuildSidecarResource_TaskType2_Invalid_Spec(t *testing.T) { }, } - taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) _, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.EqualError(t, err, "[BadTaskSpecification] Pod tasks with task type version > 1 should specify their target as a K8sPod with a defined pod spec") } @@ -357,7 +358,7 @@ func TestBuildSidecarResource_TaskType1(t *testing.T) { DefaultCPURequest: resource.MustParse("1024m"), DefaultMemoryRequest: resource.MustParse("1024Mi"), })) - taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) res, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.Nil(t, err) assert.EqualValues(t, map[string]string{ @@ -422,14 +423,14 @@ func TestBuildSideResource_TaskType1_InvalidSpec(t *testing.T) { DefaultCPURequest: resource.MustParse("1024m"), DefaultMemoryRequest: resource.MustParse("1024Mi"), })) - taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) _, err = DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.EqualError(t, err, "[BadTaskSpecification] invalid TaskSpecification, config needs to be non-empty and include missing [primary_container_name] key") task.Config = map[string]string{ "foo": "bar", } - taskCtx = getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx = getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) _, err = DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.EqualError(t, err, "[BadTaskSpecification] invalid TaskSpecification, config missing [primary_container_name] key in [map[foo:bar]]") @@ -474,7 +475,7 @@ func TestBuildSidecarResource(t *testing.T) { DefaultCPURequest: resource.MustParse("1024m"), DefaultMemoryRequest: resource.MustParse("1024Mi"), })) - taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&task, sidecarResourceRequirements, nil) res, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.Nil(t, err) assert.EqualValues(t, map[string]string{ @@ -524,7 +525,7 @@ func TestBuildSidecarReosurceMissingAnnotationsAndLabels(t *testing.T) { task := getSidecarTaskTemplateForTest(sideCarJob) - taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements, nil) resp, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.NoError(t, err) assert.EqualValues(t, map[string]string{}, resp.GetLabels()) @@ -545,11 +546,193 @@ func TestBuildSidecarResourceMissingPrimary(t *testing.T) { task := getSidecarTaskTemplateForTest(sideCarJob) - taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements, nil) _, err := DefaultPodPlugin.BuildResource(context.TODO(), taskCtx) assert.True(t, errors.Is(err, errors2.Errorf("BadTaskSpecification", ""))) } +func TestBuildSidecarResource_ExtendedResources(t *testing.T) { + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *v1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []v1.NodeSelectorTerm + expectedTol []v1.Toleration + }{ + { + "without overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &v1.ResourceRequirements{ + Limits: v1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []v1.NodeSelectorTerm{ + { + MatchExpressions: []v1.NodeSelectorRequirement{ + v1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: v1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + v1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: v1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []v1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: v1.TolerationOpEqual, + Effect: v1.TaintEffectNoSchedule, + }, + }, + }, + } + + podSpec := v1.PodSpec{ + Containers: []v1.Container{ + { + Name: "primary container", + }, + }, + } + b, err := json.Marshal(podSpec) + if err != nil { + t.Fatal(err) + } + structObj := &structpb.Struct{} + if err := json.Unmarshal(b, structObj); err != nil { + t.Fatal(err) + } + testConfigs := []struct { + name string + taskTemplate core.TaskTemplate + }{ + { + "v0", + *getSidecarTaskTemplateForTest(sidecarJob{ + PrimaryContainerName: podSpec.Containers[0].Name, + PodSpec: &podSpec, + }), + }, + { + "v1", + core.TaskTemplate{ + Type: SidecarTaskType, + Custom: structObj, + TaskTypeVersion: 1, + Config: map[string]string{ + flytek8s.PrimaryContainerKey: podSpec.Containers[0].Name, + }, + }, + }, + { + "v2", + core.TaskTemplate{ + Type: SidecarTaskType, + TaskTypeVersion: 2, + Config: map[string]string{ + flytek8s.PrimaryContainerKey: podSpec.Containers[0].Name, + }, + Target: &core.TaskTemplate_K8SPod{ + K8SPod: &core.K8SPod{ + PodSpec: structObj, + }, + }, + }, + }, + } + + for _, tCfg := range testConfigs { + for _, f := range fixtures { + t.Run(tCfg.name+" "+f.name, func(t *testing.T) { + taskTemplate := tCfg.taskTemplate + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := getDummySidecarTaskContext(&taskTemplate, f.resources, f.extendedResourcesOverride) + r, err := DefaultPodPlugin.BuildResource(context.TODO(), taskContext) + assert.Nil(t, err) + assert.NotNil(t, r) + pod, ok := r.(*v1.Pod) + assert.True(t, ok) + + assert.EqualValues( + t, + f.expectedNsr, + pod.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + pod.Spec.Tolerations, + ) + }) + } + } +} + func TestGetTaskSidecarStatus(t *testing.T) { sideCarJob := sidecarJob{ PrimaryContainerName: "PrimaryContainer", @@ -580,7 +763,7 @@ func TestGetTaskSidecarStatus(t *testing.T) { res.SetAnnotations(map[string]string{ flytek8s.PrimaryContainerKey: "PrimaryContainer", }) - taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(task, sidecarResourceRequirements, nil) phaseInfo, err := DefaultPodPlugin.GetTaskPhase(context.TODO(), taskCtx, res) assert.Nil(t, err) assert.Equal(t, expectedTaskPhase, phaseInfo.Phase(), @@ -607,7 +790,7 @@ func TestDemystifiedSidecarStatus_PrimaryFailed(t *testing.T) { res.SetAnnotations(map[string]string{ flytek8s.PrimaryContainerKey: "Primary", }) - taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements, nil) phaseInfo, err := DefaultPodPlugin.GetTaskPhase(context.TODO(), taskCtx, res) assert.Nil(t, err) assert.Equal(t, pluginsCore.PhaseRetryableFailure, phaseInfo.Phase()) @@ -632,7 +815,7 @@ func TestDemystifiedSidecarStatus_PrimarySucceeded(t *testing.T) { res.SetAnnotations(map[string]string{ flytek8s.PrimaryContainerKey: "Primary", }) - taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements, nil) phaseInfo, err := DefaultPodPlugin.GetTaskPhase(context.TODO(), taskCtx, res) assert.Nil(t, err) assert.Equal(t, pluginsCore.PhaseSuccess, phaseInfo.Phase()) @@ -657,7 +840,7 @@ func TestDemystifiedSidecarStatus_PrimaryRunning(t *testing.T) { res.SetAnnotations(map[string]string{ flytek8s.PrimaryContainerKey: "Primary", }) - taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements, nil) phaseInfo, err := DefaultPodPlugin.GetTaskPhase(context.TODO(), taskCtx, res) assert.Nil(t, err) assert.Equal(t, pluginsCore.PhaseRunning, phaseInfo.Phase()) @@ -677,7 +860,7 @@ func TestDemystifiedSidecarStatus_PrimaryMissing(t *testing.T) { res.SetAnnotations(map[string]string{ flytek8s.PrimaryContainerKey: "Primary", }) - taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements) + taskCtx := getDummySidecarTaskContext(&core.TaskTemplate{}, sidecarResourceRequirements, nil) phaseInfo, err := DefaultPodPlugin.GetTaskPhase(context.TODO(), taskCtx, res) assert.Nil(t, err) assert.Equal(t, pluginsCore.PhasePermanentFailure, phaseInfo.Phase()) diff --git a/flyteplugins/go/tasks/plugins/k8s/ray/ray_test.go b/flyteplugins/go/tasks/plugins/k8s/ray/ray_test.go index 9556b0242c1..40513e0ffaf 100644 --- a/flyteplugins/go/tasks/plugins/k8s/ray/ray_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/ray/ray_test.go @@ -69,7 +69,6 @@ func dummyRayCustomObj() *plugins.RayJob { } func dummyRayTaskTemplate(id string, rayJobObj *plugins.RayJob) *core.TaskTemplate { - ptObjJSON, err := utils.MarshalToString(rayJobObj) if err != nil { panic(err) @@ -96,7 +95,7 @@ func dummyRayTaskTemplate(id string, rayJobObj *plugins.RayJob) *core.TaskTempla } } -func dummyRayTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecutionContext { +func dummyRayTaskContext(taskTemplate *core.TaskTemplate, resources *corev1.ResourceRequirements, extendedResources *core.ExtendedResources) pluginsCore.TaskExecutionContext { taskCtx := &mocks.TaskExecutionContext{} inputReader := &pluginIOMocks.InputReader{} inputReader.OnGetInputPrefixPath().Return("/input/prefix") @@ -128,8 +127,9 @@ func dummyRayTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecut }) tID.OnGetGeneratedName().Return("some-acceptable-name") - resources := &mocks.TaskOverrides{} - resources.OnGetResources().Return(resourceRequirements) + overrides := &mocks.TaskOverrides{} + overrides.OnGetResources().Return(resources) + overrides.OnGetExtendedResources().Return(extendedResources) taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.OnGetTaskExecutionID().Return(tID) @@ -141,7 +141,7 @@ func dummyRayTaskContext(taskTemplate *core.TaskTemplate) pluginsCore.TaskExecut Name: "blah", }) taskExecutionMetadata.OnIsInterruptible().Return(true) - taskExecutionMetadata.OnGetOverrides().Return(resources) + taskExecutionMetadata.OnGetOverrides().Return(overrides) taskExecutionMetadata.OnGetK8sServiceAccount().Return(serviceAccount) taskExecutionMetadata.OnGetPlatformResources().Return(&corev1.ResourceRequirements{}) taskExecutionMetadata.OnGetSecurityContext().Return(core.SecurityContext{ @@ -164,7 +164,7 @@ func TestBuildResourceRay(t *testing.T) { err := config.SetK8sPluginConfig(&config.K8sPluginConfig{DefaultTolerations: toleration}) assert.Nil(t, err) - RayResource, err := rayJobResourceHandler.BuildResource(context.TODO(), dummyRayTaskContext(taskTemplate)) + RayResource, err := rayJobResourceHandler.BuildResource(context.TODO(), dummyRayTaskContext(taskTemplate, resourceRequirements, nil)) assert.Nil(t, err) assert.NotNil(t, RayResource) @@ -194,6 +194,148 @@ func TestBuildResourceRay(t *testing.T) { assert.Equal(t, ray.Spec.RayClusterSpec.WorkerGroupSpecs[0].Template.Spec.Tolerations, toleration) } +func TestBuildResourceRayExtendedResources(t *testing.T) { + assert.NoError(t, config.SetK8sPluginConfig(&config.K8sPluginConfig{ + GpuDeviceNodeLabel: "gpu-node-label", + GpuPartitionSizeNodeLabel: "gpu-partition-size", + GpuResourceName: flytek8s.ResourceNvidiaGPU, + })) + + fixtures := []struct { + name string + resources *corev1.ResourceRequirements + extendedResourcesBase *core.ExtendedResources + extendedResourcesOverride *core.ExtendedResources + expectedNsr []corev1.NodeSelectorTerm + expectedTol []corev1.Toleration + }{ + { + "without overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + nil, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-t4"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-t4", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + { + "with overrides", + &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + flytek8s.ResourceNvidiaGPU: resource.MustParse("1"), + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-a100", + PartitionSizeValue: &core.GPUAccelerator_PartitionSize{ + PartitionSize: "1g.5gb", + }, + }, + }, + []corev1.NodeSelectorTerm{ + { + MatchExpressions: []corev1.NodeSelectorRequirement{ + corev1.NodeSelectorRequirement{ + Key: "gpu-node-label", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"nvidia-tesla-a100"}, + }, + corev1.NodeSelectorRequirement{ + Key: "gpu-partition-size", + Operator: corev1.NodeSelectorOpIn, + Values: []string{"1g.5gb"}, + }, + }, + }, + }, + []corev1.Toleration{ + { + Key: "gpu-node-label", + Value: "nvidia-tesla-a100", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + { + Key: "gpu-partition-size", + Value: "1g.5gb", + Operator: corev1.TolerationOpEqual, + Effect: corev1.TaintEffectNoSchedule, + }, + }, + }, + } + + for _, f := range fixtures { + t.Run(f.name, func(t *testing.T) { + taskTemplate := dummyRayTaskTemplate("ray-id", dummyRayCustomObj()) + taskTemplate.ExtendedResources = f.extendedResourcesBase + taskContext := dummyRayTaskContext(taskTemplate, f.resources, f.extendedResourcesOverride) + rayJobResourceHandler := rayJobResourceHandler{} + r, err := rayJobResourceHandler.BuildResource(context.TODO(), taskContext) + assert.Nil(t, err) + assert.NotNil(t, r) + rayJob, ok := r.(*rayv1alpha1.RayJob) + assert.True(t, ok) + + // Head node + headNodeSpec := rayJob.Spec.RayClusterSpec.HeadGroupSpec.Template.Spec + assert.EqualValues( + t, + f.expectedNsr, + headNodeSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + headNodeSpec.Tolerations, + ) + + // Worker node + workerNodeSpec := rayJob.Spec.RayClusterSpec.WorkerGroupSpecs[0].Template.Spec + assert.EqualValues( + t, + f.expectedNsr, + workerNodeSpec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms, + ) + assert.EqualValues( + t, + f.expectedTol, + workerNodeSpec.Tolerations, + ) + }) + } +} + func TestDefaultStartParameters(t *testing.T) { rayJobResourceHandler := rayJobResourceHandler{} rayJob := &plugins.RayJob{ @@ -213,7 +355,7 @@ func TestDefaultStartParameters(t *testing.T) { err := config.SetK8sPluginConfig(&config.K8sPluginConfig{DefaultTolerations: toleration}) assert.Nil(t, err) - RayResource, err := rayJobResourceHandler.BuildResource(context.TODO(), dummyRayTaskContext(taskTemplate)) + RayResource, err := rayJobResourceHandler.BuildResource(context.TODO(), dummyRayTaskContext(taskTemplate, resourceRequirements, nil)) assert.Nil(t, err) assert.NotNil(t, RayResource) diff --git a/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go b/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go index 18565fc0a48..be2f9477b67 100644 --- a/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go +++ b/flyteplugins/go/tasks/plugins/k8s/spark/spark_test.go @@ -370,6 +370,8 @@ func dummySparkTaskContext(taskTemplate *core.TaskTemplate, interruptible bool) overrides := &mocks.TaskOverrides{} overrides.On("GetResources").Return(&corev1.ResourceRequirements{}) + // No support for GPUs, and consequently, ExtendedResources on Spark plugin. + overrides.On("GetExtendedResources").Return(nil) taskExecutionMetadata := &mocks.TaskExecutionMetadata{} taskExecutionMetadata.On("GetTaskExecutionID").Return(tID) diff --git a/flyteplugins/tests/end_to_end.go b/flyteplugins/tests/end_to_end.go index 6043ec88fd1..77522ab1d1b 100644 --- a/flyteplugins/tests/end_to_end.go +++ b/flyteplugins/tests/end_to_end.go @@ -156,6 +156,7 @@ func RunPluginEndToEndTest(t *testing.T, executor pluginCore.Plugin, template *i Requests: map[v1.ResourceName]resource.Quantity{}, Limits: map[v1.ResourceName]resource.Quantity{}, }) + overrides.OnGetExtendedResources().Return(&idlCore.ExtendedResources{}) tMeta := &coreMocks.TaskExecutionMetadata{} tMeta.OnGetTaskExecutionID().Return(tID) diff --git a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/iface.go b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/iface.go index 19fa0ec5edc..ab9562e4a3b 100644 --- a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/iface.go +++ b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/iface.go @@ -436,6 +436,7 @@ type ExecutableNode interface { GetOutputAlias() []Alias GetInputBindings() []*Binding GetResources() *v1.ResourceRequirements + GetExtendedResources() *core.ExtendedResources GetConfig() *v1.ConfigMap GetRetryStrategy() *RetryStrategy GetExecutionDeadline() *time.Duration diff --git a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/mocks/ExecutableNode.go b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/mocks/ExecutableNode.go index 901148e8207..ae1663bba1f 100644 --- a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/mocks/ExecutableNode.go +++ b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/mocks/ExecutableNode.go @@ -3,10 +3,11 @@ package mocks import ( - time "time" - + core "github.com/flyteorg/flyte/flyteidl/gen/pb-go/flyteidl/core" mock "github.com/stretchr/testify/mock" + time "time" + v1 "k8s.io/api/core/v1" v1alpha1 "github.com/flyteorg/flyte/flytepropeller/pkg/apis/flyteworkflow/v1alpha1" @@ -187,6 +188,40 @@ func (_m *ExecutableNode) GetExecutionDeadline() *time.Duration { return r0 } +type ExecutableNode_GetExtendedResources struct { + *mock.Call +} + +func (_m ExecutableNode_GetExtendedResources) Return(_a0 *core.ExtendedResources) *ExecutableNode_GetExtendedResources { + return &ExecutableNode_GetExtendedResources{Call: _m.Call.Return(_a0)} +} + +func (_m *ExecutableNode) OnGetExtendedResources() *ExecutableNode_GetExtendedResources { + c_call := _m.On("GetExtendedResources") + return &ExecutableNode_GetExtendedResources{Call: c_call} +} + +func (_m *ExecutableNode) OnGetExtendedResourcesMatch(matchers ...interface{}) *ExecutableNode_GetExtendedResources { + c_call := _m.On("GetExtendedResources", matchers...) + return &ExecutableNode_GetExtendedResources{Call: c_call} +} + +// GetExtendedResources provides a mock function with given fields: +func (_m *ExecutableNode) GetExtendedResources() *core.ExtendedResources { + ret := _m.Called() + + var r0 *core.ExtendedResources + if rf, ok := ret.Get(0).(func() *core.ExtendedResources); ok { + r0 = rf() + } else { + if ret.Get(0) != nil { + r0 = ret.Get(0).(*core.ExtendedResources) + } + } + + return r0 +} + type ExecutableNode_GetGateNode struct { *mock.Call } diff --git a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/nodes.go b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/nodes.go index 4f6df3f87b6..d061408d766 100644 --- a/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/nodes.go +++ b/flytepropeller/pkg/apis/flyteworkflow/v1alpha1/nodes.go @@ -92,6 +92,30 @@ func (in *NodeMetadata) DeepCopyInto(out *NodeMetadata) { // Once we figure out the autogenerate story we can replace this } +type ExtendedResources struct { + *core.ExtendedResources +} + +func (in *ExtendedResources) MarshalJSON() ([]byte, error) { + var buf bytes.Buffer + if err := marshaler.Marshal(&buf, in.ExtendedResources); err != nil { + return nil, err + } + + return buf.Bytes(), nil +} + +func (in *ExtendedResources) UnmarshalJSON(b []byte) error { + in.ExtendedResources = &core.ExtendedResources{} + return jsonpb.Unmarshal(bytes.NewReader(b), in.ExtendedResources) +} + +func (in *ExtendedResources) DeepCopyInto(out *ExtendedResources) { + *out = *in + // We do not manipulate the object, so its ok + // Once we figure out the autogenerate story we can replace this +} + type NodeSpec struct { ID NodeID `json:"id"` Name string `json:"name,omitempty"` @@ -134,6 +158,10 @@ type NodeSpec struct { // If not specified, the pod will be dispatched by default scheduler. // +optional SchedulerName string `json:"schedulerName,omitempty" protobuf:"bytes,19,opt,name=schedulerName"` + // If specified, includes overrides for extended resources to allocate to the + // node. + // +optional + ExtendedResources *ExtendedResources `json:"extendedResources,omitempty" protobuf:"bytes,20,opt,name=extendedResources"` // If specified, the pod's tolerations. // +optional Tolerations []typesv1.Toleration `json:"tolerations,omitempty" protobuf:"bytes,22,opt,name=tolerations"` @@ -182,6 +210,13 @@ func (in *NodeSpec) GetResources() *typesv1.ResourceRequirements { return in.Resources } +func (in *NodeSpec) GetExtendedResources() *core.ExtendedResources { + if in.ExtendedResources == nil { + return nil + } + return in.ExtendedResources.ExtendedResources +} + func (in *NodeSpec) GetOutputAlias() []Alias { return in.OutputAliases } diff --git a/flytepropeller/pkg/compiler/transformers/k8s/node.go b/flytepropeller/pkg/compiler/transformers/k8s/node.go index 55f45c8a9bf..7ab65029464 100644 --- a/flytepropeller/pkg/compiler/transformers/k8s/node.go +++ b/flytepropeller/pkg/compiler/transformers/k8s/node.go @@ -28,6 +28,7 @@ func buildNodeSpec(n *core.Node, tasks []*core.CompiledTask, errs errors.Compile var task *core.TaskTemplate var resources *core.Resources + var extendedResources *v1alpha1.ExtendedResources if n.GetTaskNode() != nil { taskID := n.GetTaskNode().GetReferenceId().String() // TODO: Use task index for quick lookup @@ -43,8 +44,16 @@ func buildNodeSpec(n *core.Node, tasks []*core.CompiledTask, errs errors.Compile return nil, !errs.HasErrors() } - if n.GetTaskNode().Overrides != nil && n.GetTaskNode().Overrides.Resources != nil { - resources = n.GetTaskNode().Overrides.Resources + if overrides := n.GetTaskNode().Overrides; overrides != nil { + if overrides.GetResources() != nil { + resources = overrides.GetResources() + } + + if overrides.GetExtendedResources() != nil { + extendedResources = &v1alpha1.ExtendedResources{ + ExtendedResources: overrides.GetExtendedResources(), + } + } } } @@ -81,6 +90,7 @@ func buildNodeSpec(n *core.Node, tasks []*core.CompiledTask, errs errors.Compile RetryStrategy: computeRetryStrategy(n, task), ExecutionDeadline: timeout, Resources: res, + ExtendedResources: extendedResources, OutputAliases: toAliasValueArray(n.GetOutputAliases()), InputBindings: toBindingValueArray(n.GetInputs()), ActiveDeadline: activeDeadline, diff --git a/flytepropeller/pkg/compiler/transformers/k8s/node_test.go b/flytepropeller/pkg/compiler/transformers/k8s/node_test.go index 8e312756ee6..4ba9010c96f 100644 --- a/flytepropeller/pkg/compiler/transformers/k8s/node_test.go +++ b/flytepropeller/pkg/compiler/transformers/k8s/node_test.go @@ -117,6 +117,30 @@ func TestBuildNodeSpec(t *testing.T) { assert.NotNil(t, spec.Resources) assert.NotNil(t, spec.Resources.Requests.Cpu()) assert.Equal(t, expectedCPU.Value(), spec.Resources.Requests.Cpu().Value()) + assert.Nil(t, spec.GetExtendedResources()) + }) + + t.Run("node with resource extensions overrides", func(t *testing.T) { + expectedGpuDevice := "nvidia-tesla-t4" + n.Node.Target = &core.Node_TaskNode{ + TaskNode: &core.TaskNode{ + Reference: &core.TaskNode_ReferenceId{ + ReferenceId: &core.Identifier{Name: "ref_2"}, + }, + Overrides: &core.TaskNodeOverrides{ + ExtendedResources: &core.ExtendedResources{ + GpuAccelerator: &core.GPUAccelerator{ + Device: "nvidia-tesla-t4", + }, + }, + }, + }, + } + + spec := mustBuild(t, n, 1, errs.NewScope()) + assert.NotNil(t, spec.GetExtendedResources()) + assert.NotNil(t, spec.GetExtendedResources().GetGpuAccelerator()) + assert.Equal(t, expectedGpuDevice, spec.GetExtendedResources().GetGpuAccelerator().GetDevice()) }) t.Run("LaunchPlanRef", func(t *testing.T) { From ba10ac0ae23a335dea0c289a2c313ff7950381c7 Mon Sep 17 00:00:00 2001 From: Katrina Rogan Date: Thu, 12 Oct 2023 11:01:24 -0700 Subject: [PATCH 10/18] Allow specifying default cluster pool assignments in admin config (#4208) --- charts/flyte-core/templates/admin/configmap.yaml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/charts/flyte-core/templates/admin/configmap.yaml b/charts/flyte-core/templates/admin/configmap.yaml index fb5820c8420..53af659e8b5 100644 --- a/charts/flyte-core/templates/admin/configmap.yaml +++ b/charts/flyte-core/templates/admin/configmap.yaml @@ -36,6 +36,9 @@ data: {{- end }} {{- with .Values.configmap.namespace_config }} namespace_config.yaml: | {{ toYaml . | nindent 4 }} +{{- end }} +{{- with .Values.configmap.clusterpool_config }} + clusterpool_config.yaml: | {{ toYaml . | nindent 4 }} {{- end }} storage.yaml: | {{ tpl (include "storage" .) $ | nindent 4 }} {{- with .Values.configmap.task_resource_defaults }} From db3a7ff0ec6f04ac4cd7f454bfea82ac68574c80 Mon Sep 17 00:00:00 2001 From: Iaroslav Ciupin Date: Thu, 12 Oct 2023 22:57:33 +0300 Subject: [PATCH 11/18] allow passing cluster pool to run-tests.py (#4210) Signed-off-by: Iaroslav Ciupin --- boilerplate/flyte/end2end/run-tests.py | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/boilerplate/flyte/end2end/run-tests.py b/boilerplate/flyte/end2end/run-tests.py index 5365da006e7..58b2a5dc74b 100644 --- a/boilerplate/flyte/end2end/run-tests.py +++ b/boilerplate/flyte/end2end/run-tests.py @@ -5,7 +5,7 @@ import sys import time import traceback -from typing import Dict, List, Mapping, Tuple +from typing import Dict, List, Mapping, Tuple, Optional import click import requests @@ -89,10 +89,10 @@ } -def execute_workflow(remote, version, workflow_name, inputs): +def execute_workflow(remote: FlyteRemote, version, workflow_name, inputs, cluster_pool_name: Optional[str] = None): print(f"Fetching workflow={workflow_name} and version={version}") wf = remote.fetch_workflow(name=workflow_name, version=version) - return remote.execute(wf, inputs=inputs, wait=False) + return remote.execute(wf, inputs=inputs, wait=False, cluster_pool=cluster_pool_name) def executions_finished(executions_by_wfgroup: Dict[str, List[FlyteWorkflowExecution]]) -> bool: @@ -125,6 +125,7 @@ def schedule_workflow_groups( workflow_groups: List[str], remote: FlyteRemote, terminate_workflow_on_failure: bool, + cluster_pool_name: Optional[str] = None, ) -> Dict[str, bool]: """ Schedule workflows executions for all workflow groups and return True if all executions succeed, otherwise @@ -135,7 +136,7 @@ def schedule_workflow_groups( for wf_group in workflow_groups: workflows = FLYTESNACKS_WORKFLOW_GROUPS.get(wf_group, []) executions_by_wfgroup[wf_group] = [ - execute_workflow(remote, tag, workflow[0], workflow[1]) for workflow in workflows + execute_workflow(remote, tag, workflow[0], workflow[1], cluster_pool_name) for workflow in workflows ] # Wait for all executions to finish @@ -179,6 +180,7 @@ def run( priorities: List[str], config_file_path, terminate_workflow_on_failure: bool, + cluster_pool_name: Optional[str] = None, ) -> List[Dict[str, str]]: remote = FlyteRemote( Config.auto(config_file=config_file_path), @@ -215,7 +217,7 @@ def run( valid_workgroups.append(workflow_group) results_by_wfgroup = schedule_workflow_groups( - flytesnacks_release_tag, valid_workgroups, remote, terminate_workflow_on_failure + flytesnacks_release_tag, valid_workgroups, remote, terminate_workflow_on_failure, cluster_pool_name ) for workflow_group, succeeded in results_by_wfgroup.items(): @@ -261,15 +263,17 @@ def run( @click.argument("flytesnacks_release_tag") @click.argument("priorities") @click.argument("config_file") +@click.argument("cluster_pool_name") def cli( flytesnacks_release_tag, priorities, config_file, return_non_zero_on_failure, terminate_workflow_on_failure, + cluster_pool_name, ): print(f"return_non_zero_on_failure={return_non_zero_on_failure}") - results = run(flytesnacks_release_tag, priorities, config_file, terminate_workflow_on_failure) + results = run(flytesnacks_release_tag, priorities, config_file, terminate_workflow_on_failure, cluster_pool_name) # Write a json object in its own line describing the result of this run to stdout print(f"Result of run:\n{json.dumps(results)}") From 8d7b3869ee629b1e18723589a4458256ccd0dda3 Mon Sep 17 00:00:00 2001 From: Kevin Su Date: Thu, 12 Oct 2023 13:34:25 -0700 Subject: [PATCH 12/18] Skip spell check on .pb files (#4207) Signed-off-by: Kevin Su --- .github/workflows/codespell.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/codespell.yml b/.github/workflows/codespell.yml index 7373affc383..a123509488e 100644 --- a/.github/workflows/codespell.yml +++ b/.github/workflows/codespell.yml @@ -20,3 +20,5 @@ jobs: uses: actions/checkout@v3 - name: Codespell uses: codespell-project/actions-codespell@v2 + with: + skip: "*.pb" From b932cf1179b263b7c4404d9ea30b7138b0d00862 Mon Sep 17 00:00:00 2001 From: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> Date: Thu, 12 Oct 2023 14:07:15 -0700 Subject: [PATCH 13/18] Add gh workflow to publish flyteidl release to PyPI and npm (#4211) Signed-off-by: Eduardo Apolinario Co-authored-by: Eduardo Apolinario --- .github/workflows/flyteidl-release.yml | 51 ++++++++++++++++++++++++++ script/release.sh | 2 +- 2 files changed, 52 insertions(+), 1 deletion(-) create mode 100644 .github/workflows/flyteidl-release.yml diff --git a/.github/workflows/flyteidl-release.yml b/.github/workflows/flyteidl-release.yml new file mode 100644 index 00000000000..ea8a7899fa2 --- /dev/null +++ b/.github/workflows/flyteidl-release.yml @@ -0,0 +1,51 @@ +name: Upload flyteidl to PyPI and npm + +on: + release: + types: [created] + +jobs: + deploy: + runs-on: ubuntu-latest + defaults: + run: + working-directory: flyteidl + steps: + - uses: actions/checkout@v2 + - name: Set up Python + uses: actions/setup-python@v1 + with: + python-version: "3.x" + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install setuptools wheel twine + - name: Autobump version + run: | + # from refs/tags/v1.2.3 get 1.2.3 + VERSION=$(echo $GITHUB_REF | sed 's#.*/v##') + VERSION=$VERSION make -C flyteidl update_pyversion + shell: bash + - name: Build and publish + env: + TWINE_USERNAME: ${{ secrets.PYPI_USERNAME }} + TWINE_PASSWORD: ${{ secrets.PYPI_PASSWORD }} + run: | + python setup.py sdist bdist_wheel + twine upload dist/* + - uses: actions/setup-node@v1 + with: + node-version: "12.x" + registry-url: "https://registry.npmjs.org" + - name: Autobump version + run: | + # from refs/tags/v1.2.3 get 1.2.3 + VERSION=$(echo $GITHUB_REF | sed 's#.*/v##') + VERSION=$VERSION make update_npmversion + shell: bash + - run: | + npm install + - run: | + npm publish --access=public + env: + NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} diff --git a/script/release.sh b/script/release.sh index 14bec2a54ff..190ecbc8f18 100755 --- a/script/release.sh +++ b/script/release.sh @@ -6,7 +6,7 @@ set -ex # TODO(monorepo): This only works if we have at least one component tag per release. # In other words, if we have two consecutive releases the latest tag in the second release is going to point to an invalid # tag (because there will not be images tagged with the previous release tag). -LATEST_TAG=$(git tag | sed 's#[^/]*/##' | sort | tail -n 1) +LATEST_TAG=$(git tag | sed 's#[^/]*/##' | sort --version-sort | tail -n1) FLYTEKIT_TAG=$(curl --silent "https://api.github.com/repos/flyteorg/flytekit/releases/latest" | jq -r .tag_name | sed 's/^v//') FLYTECONSOLE_TAG=$(curl --silent "https://api.github.com/repos/flyteorg/flyteconsole/releases/latest" | jq -r .tag_name) From da77476ecbb8b183612405e9f3e3f7117d67d2a8 Mon Sep 17 00:00:00 2001 From: Kevin Su Date: Thu, 12 Oct 2023 15:22:51 -0700 Subject: [PATCH 14/18] Update the item only if it exists in the cache (#4117) * Add item ID to the workqueue instead Signed-off-by: Kevin Su * nit Signed-off-by: Kevin Su * Update the item only if it exists in the cache Signed-off-by: Kevin Su * Update tests Signed-off-by: Kevin Su * nit Signed-off-by: Kevin Su * address comment Signed-off-by: Kevin Su * fixed tests Signed-off-by: Kevin Su * address comment Signed-off-by: Kevin Su --------- Signed-off-by: Kevin Su --- .../internal/webapi/monitor_test.go | 2 +- flytestdlib/cache/auto_refresh.go | 62 +++++++++++++++---- flytestdlib/cache/auto_refresh_test.go | 30 ++++++++- 3 files changed, 80 insertions(+), 14 deletions(-) diff --git a/flyteplugins/go/tasks/pluginmachinery/internal/webapi/monitor_test.go b/flyteplugins/go/tasks/pluginmachinery/internal/webapi/monitor_test.go index f916a938006..bb6f0b089c6 100644 --- a/flyteplugins/go/tasks/pluginmachinery/internal/webapi/monitor_test.go +++ b/flyteplugins/go/tasks/pluginmachinery/internal/webapi/monitor_test.go @@ -37,7 +37,7 @@ func Test_monitor(t *testing.T) { client.OnStatusMatch(ctx, mock.Anything).Return(core2.PhaseInfoSuccess(nil), nil) wg := sync.WaitGroup{} - wg.Add(4) + wg.Add(8) cacheObj, err := cache.NewAutoRefreshCache(rand.String(5), func(ctx context.Context, batch cache.Batch) (updatedBatch []cache.ItemSyncResponse, err error) { wg.Done() t.Logf("Syncing Item [%+v]", batch[0]) diff --git a/flytestdlib/cache/auto_refresh.go b/flytestdlib/cache/auto_refresh.go index 13a787b03f1..d0a609dc3b1 100644 --- a/flytestdlib/cache/auto_refresh.go +++ b/flytestdlib/cache/auto_refresh.go @@ -3,6 +3,7 @@ package cache import ( "context" "fmt" + "sync" "time" "github.com/flyteorg/flyte/flytestdlib/contextutils" @@ -122,6 +123,7 @@ type autoRefresh struct { syncPeriod time.Duration workqueue workqueue.RateLimitingInterface parallelizm int + lock sync.RWMutex } func getEvictionFunction(counter prometheus.Counter) func(key interface{}, value interface{}) { @@ -173,6 +175,25 @@ func (w *autoRefresh) Start(ctx context.Context) error { return nil } +// Update updates the item only if it exists in the cache, return true if we updated the item. +func (w *autoRefresh) Update(id ItemID, item Item) (ok bool) { + w.lock.Lock() + defer w.lock.Unlock() + ok = w.lruMap.Contains(id) + if ok { + w.lruMap.Add(id, item) + } + return ok +} + +// Delete deletes the item from the cache if it exists. +func (w *autoRefresh) Delete(key interface{}) { + w.lock.Lock() + defer w.lock.Unlock() + w.toDelete.Remove(key) + w.lruMap.Remove(key) +} + func (w *autoRefresh) Get(id ItemID) (Item, error) { if val, ok := w.lruMap.Get(id); ok { w.metrics.CacheHit.Inc() @@ -212,8 +233,7 @@ func (w *autoRefresh) enqueueBatches(ctx context.Context) error { snapshot := make([]ItemWrapper, 0, len(keys)) for _, k := range keys { if w.toDelete.Contains(k) { - w.lruMap.Remove(k) - w.toDelete.Remove(k) + w.Delete(k) continue } // If not ok, it means evicted between the item was evicted between getting the keys and this update loop @@ -273,18 +293,37 @@ func (w *autoRefresh) sync(ctx context.Context) (err error) { case <-ctx.Done(): return nil default: - item, shutdown := w.workqueue.Get() + batch, shutdown := w.workqueue.Get() if shutdown { + logger.Debugf(ctx, "Shutting down worker") return nil } - t := w.metrics.SyncLatency.Start() - updatedBatch, err := w.syncCb(ctx, *item.(*Batch)) - // Since we create batches every time we sync, we will just remove the item from the queue here // regardless of whether it succeeded the sync or not. - w.workqueue.Forget(item) - w.workqueue.Done(item) + w.workqueue.Forget(batch) + w.workqueue.Done(batch) + + newBatch := make(Batch, 0, len(*batch.(*Batch))) + for _, b := range *batch.(*Batch) { + itemID := b.GetID() + item, ok := w.lruMap.Get(itemID) + if !ok { + logger.Debugf(ctx, "item with id [%v] not found in cache", itemID) + continue + } + if item.(Item).IsTerminal() { + logger.Debugf(ctx, "item with id [%v] is terminal", itemID) + continue + } + newBatch = append(newBatch, b) + } + if len(newBatch) == 0 { + continue + } + + t := w.metrics.SyncLatency.Start() + updatedBatch, err := w.syncCb(ctx, newBatch) if err != nil { w.metrics.SyncErrors.Inc() @@ -295,14 +334,13 @@ func (w *autoRefresh) sync(ctx context.Context) (err error) { for _, item := range updatedBatch { if item.Action == Update { - // Add adds the item if it has been evicted or updates an existing one. - w.lruMap.Add(item.ID, item.Item) + // Updates an existing item. + w.Update(item.ID, item.Item) } } w.toDelete.Range(func(key interface{}) bool { - w.lruMap.Remove(key) - w.toDelete.Remove(key) + w.Delete(key) return true }) diff --git a/flytestdlib/cache/auto_refresh_test.go b/flytestdlib/cache/auto_refresh_test.go index e1250afd68b..ccd2da6a191 100644 --- a/flytestdlib/cache/auto_refresh_test.go +++ b/flytestdlib/cache/auto_refresh_test.go @@ -62,7 +62,7 @@ func syncTerminalItem(_ context.Context, batch Batch) ([]ItemSyncResponse, error panic("This should never be called") } -func TestCacheThree(t *testing.T) { +func TestCacheFour(t *testing.T) { testResyncPeriod := time.Millisecond rateLimiter := workqueue.DefaultControllerRateLimiter() @@ -142,6 +142,34 @@ func TestCacheThree(t *testing.T) { cancel() }) + + t.Run("Test update and delete cache", func(t *testing.T) { + cache, err := NewAutoRefreshCache("fake3", syncTerminalItem, rateLimiter, testResyncPeriod, 10, 2, promutils.NewTestScope()) + assert.NoError(t, err) + + ctx, cancel := context.WithCancel(context.Background()) + assert.NoError(t, cache.Start(ctx)) + + itemID := "dummy_id" + _, err = cache.GetOrCreate(itemID, terminalCacheItem{ + val: 0, + }) + assert.NoError(t, err) + + // Wait half a second for all resync periods to complete + // If the cache tries to enqueue the item, a panic will be thrown. + time.Sleep(500 * time.Millisecond) + + err = cache.DeleteDelayed(itemID) + assert.NoError(t, err) + + time.Sleep(500 * time.Millisecond) + item, err := cache.Get(itemID) + assert.Nil(t, item) + assert.Error(t, err) + + cancel() + }) } func TestQueueBuildUp(t *testing.T) { From 68c055b7b51eebc4cf009b5caf488026328cfcc0 Mon Sep 17 00:00:00 2001 From: Eduardo Apolinario <653394+eapolinario@users.noreply.github.com> Date: Thu, 12 Oct 2023 17:07:13 -0700 Subject: [PATCH 15/18] Revert "allow passing cluster pool to run-tests.py (#4210)" (#4212) This reverts commit db3a7ff0ec6f04ac4cd7f454bfea82ac68574c80. Signed-off-by: Eduardo Apolinario Co-authored-by: Eduardo Apolinario --- boilerplate/flyte/end2end/run-tests.py | 16 ++++++---------- 1 file changed, 6 insertions(+), 10 deletions(-) diff --git a/boilerplate/flyte/end2end/run-tests.py b/boilerplate/flyte/end2end/run-tests.py index 58b2a5dc74b..5365da006e7 100644 --- a/boilerplate/flyte/end2end/run-tests.py +++ b/boilerplate/flyte/end2end/run-tests.py @@ -5,7 +5,7 @@ import sys import time import traceback -from typing import Dict, List, Mapping, Tuple, Optional +from typing import Dict, List, Mapping, Tuple import click import requests @@ -89,10 +89,10 @@ } -def execute_workflow(remote: FlyteRemote, version, workflow_name, inputs, cluster_pool_name: Optional[str] = None): +def execute_workflow(remote, version, workflow_name, inputs): print(f"Fetching workflow={workflow_name} and version={version}") wf = remote.fetch_workflow(name=workflow_name, version=version) - return remote.execute(wf, inputs=inputs, wait=False, cluster_pool=cluster_pool_name) + return remote.execute(wf, inputs=inputs, wait=False) def executions_finished(executions_by_wfgroup: Dict[str, List[FlyteWorkflowExecution]]) -> bool: @@ -125,7 +125,6 @@ def schedule_workflow_groups( workflow_groups: List[str], remote: FlyteRemote, terminate_workflow_on_failure: bool, - cluster_pool_name: Optional[str] = None, ) -> Dict[str, bool]: """ Schedule workflows executions for all workflow groups and return True if all executions succeed, otherwise @@ -136,7 +135,7 @@ def schedule_workflow_groups( for wf_group in workflow_groups: workflows = FLYTESNACKS_WORKFLOW_GROUPS.get(wf_group, []) executions_by_wfgroup[wf_group] = [ - execute_workflow(remote, tag, workflow[0], workflow[1], cluster_pool_name) for workflow in workflows + execute_workflow(remote, tag, workflow[0], workflow[1]) for workflow in workflows ] # Wait for all executions to finish @@ -180,7 +179,6 @@ def run( priorities: List[str], config_file_path, terminate_workflow_on_failure: bool, - cluster_pool_name: Optional[str] = None, ) -> List[Dict[str, str]]: remote = FlyteRemote( Config.auto(config_file=config_file_path), @@ -217,7 +215,7 @@ def run( valid_workgroups.append(workflow_group) results_by_wfgroup = schedule_workflow_groups( - flytesnacks_release_tag, valid_workgroups, remote, terminate_workflow_on_failure, cluster_pool_name + flytesnacks_release_tag, valid_workgroups, remote, terminate_workflow_on_failure ) for workflow_group, succeeded in results_by_wfgroup.items(): @@ -263,17 +261,15 @@ def run( @click.argument("flytesnacks_release_tag") @click.argument("priorities") @click.argument("config_file") -@click.argument("cluster_pool_name") def cli( flytesnacks_release_tag, priorities, config_file, return_non_zero_on_failure, terminate_workflow_on_failure, - cluster_pool_name, ): print(f"return_non_zero_on_failure={return_non_zero_on_failure}") - results = run(flytesnacks_release_tag, priorities, config_file, terminate_workflow_on_failure, cluster_pool_name) + results = run(flytesnacks_release_tag, priorities, config_file, terminate_workflow_on_failure) # Write a json object in its own line describing the result of this run to stdout print(f"Result of run:\n{json.dumps(results)}") From 2edc3bd552b559d250bfff77987506da586a1759 Mon Sep 17 00:00:00 2001 From: "Fabio M. Graetz, Ph.D" Date: Thu, 12 Oct 2023 23:52:44 -0500 Subject: [PATCH 16/18] Fix: Prevent unintended use of proxy authorization when no command is provided (#4217) * Don't check for cfg.ProxyCommand != nil but for empty slice Signed-off-by: Fabio Graetz * Check for proxy command != nil and len > 0 Signed-off-by: Fabio Graetz --------- Signed-off-by: Fabio Graetz --- flyteidl/clients/go/admin/auth_interceptor.go | 2 +- flyteidl/clients/go/admin/client.go | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/flyteidl/clients/go/admin/auth_interceptor.go b/flyteidl/clients/go/admin/auth_interceptor.go index ef94d857567..09391df67ae 100644 --- a/flyteidl/clients/go/admin/auth_interceptor.go +++ b/flyteidl/clients/go/admin/auth_interceptor.go @@ -109,7 +109,7 @@ func setHTTPClientContext(ctx context.Context, cfg *Config, proxyCredentialsFutu transport.Proxy = http.ProxyURL(&cfg.HTTPProxyURL.URL) } - if cfg.ProxyCommand != nil { + if cfg.ProxyCommand != nil && len(cfg.ProxyCommand) > 0 { httpClient.Transport = &proxyAuthTransport{ transport: transport, proxyCredentialsFuture: proxyCredentialsFuture, diff --git a/flyteidl/clients/go/admin/client.go b/flyteidl/clients/go/admin/client.go index 7555f64045a..69c3542367b 100644 --- a/flyteidl/clients/go/admin/client.go +++ b/flyteidl/clients/go/admin/client.go @@ -153,7 +153,7 @@ func NewAdminConnection(ctx context.Context, cfg *Config, proxyCredentialsFuture opts = append(opts, GetAdditionalAdminClientConfigOptions(cfg)...) - if cfg.ProxyCommand != nil { + if cfg.ProxyCommand != nil && len(cfg.ProxyCommand) > 0 { opts = append(opts, grpc.WithChainUnaryInterceptor(NewProxyAuthInterceptor(cfg, proxyCredentialsFuture))) opts = append(opts, grpc.WithPerRPCCredentials(proxyCredentialsFuture)) } From abfbb24c85bd10303a1263781240195c817274f0 Mon Sep 17 00:00:00 2001 From: Surav Shrestha Date: Fri, 13 Oct 2023 23:26:58 +0545 Subject: [PATCH 17/18] docs: fix typo in rfc/system/3749-sane-overridable-defaults.md (#4219) Signed-off-by: Surav Shrestha --- rfc/system/3749-sane-overridable-defaults.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/rfc/system/3749-sane-overridable-defaults.md b/rfc/system/3749-sane-overridable-defaults.md index eff22063242..f7288763ab9 100644 --- a/rfc/system/3749-sane-overridable-defaults.md +++ b/rfc/system/3749-sane-overridable-defaults.md @@ -174,7 +174,7 @@ metadata: ``` This can work the same way cluster resource templates are mounted as ConfigMaps in the FlyteAdmin deployment but are [fully customizable](https://github.com/flyteorg/flyte/blob/5ab8cb887a5a7051070cb93fca603ed1f22f2f74/charts/flyte-core/values.yaml#L752-L810). -This does require restarting the FlyteAdmin deployment for individual overrides changes but allows for declaratively defining the complete set of overrides in a reproducible manner. See more discusison on the trade-offs [here](https://github.com/flyteorg/flyte/discussions/3731#discussioncomment-6053743). +This does require restarting the FlyteAdmin deployment for individual overrides changes but allows for declaratively defining the complete set of overrides in a reproducible manner. See more discussion on the trade-offs [here](https://github.com/flyteorg/flyte/discussions/3731#discussioncomment-6053743). #### FlyteAdmin changes From c6476cc178e576a4b7853d5d2adcb54081848974 Mon Sep 17 00:00:00 2001 From: Surav Shrestha Date: Sat, 14 Oct 2023 03:46:20 +0545 Subject: [PATCH 18/18] docs: fix typo in ./kustomize/README.md (#4218) Signed-off-by: Surav Shrestha --- kustomize/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kustomize/README.md b/kustomize/README.md index 94f0f6ec568..69a9c10d0ce 100644 --- a/kustomize/README.md +++ b/kustomize/README.md @@ -6,7 +6,7 @@ In brief, Kustomize allows composing a deployment yaml using multiple components a composed set of components that can be deployed to a [Single Kubernetes cluster](./base/single_cluster). This deployment configures various components using [Flyte Configuration system](todo). -The *Single Cluster* configuration on its own is not deployable. But indiviudal [overlays](./overlays) are deployable. +The *Single Cluster* configuration on its own is not deployable. But individual [overlays](./overlays) are deployable. Refer to 1. [Base Components](./base): If you want to build your own overlay start here