primitive 0.2.52__py3-none-any.whl → 0.2.54__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
primitive/__about__.py CHANGED
@@ -1,4 +1,4 @@
1
1
  # SPDX-FileCopyrightText: 2025-present Dylan Stein <dylan@primitive.tech>
2
2
  #
3
3
  # SPDX-License-Identifier: MIT
4
- __version__ = "0.2.52"
4
+ __version__ = "0.2.54"
@@ -118,56 +118,55 @@ class Agent(BaseAction):
118
118
  logger.error(
119
119
  f"JobRun {job_run_data.get('name')} is already completed with conclusion {job_run_data.get('jobRun', {}).get('conclusion', 'unknown')}. Exiting."
120
120
  )
121
- if RUNNING_IN_CONTAINER:
122
- logger.info("Running in container, exiting after job run")
123
- break
121
+ break
124
122
 
125
- logger.info(
126
- f"Waiting for JobRun {job_run_data.get('id')} to be in_progress. Current status: {job_run_status}"
127
- )
128
123
  job_run_result = self.primitive.jobs.get_job_run(id=job_run_id)
129
124
  if job_run_result.data is not None:
130
125
  job_run_data = job_run_result.data.get("jobRun", {})
131
126
  job_run_status = job_run_data.get("status", None)
132
-
133
- sleep(1)
134
-
135
- try:
136
- runner = Runner(
137
- primitive=self.primitive,
138
- job_run=job_run_data,
139
- )
140
- runner.setup()
141
- except Exception as exception:
142
- logger.exception(
143
- f"Exception while initializing runner: {exception}"
144
- )
145
- self.primitive.jobs.job_run_update(
146
- id=job_run_id,
147
- status="request_completed",
148
- conclusion="failure",
127
+ logger.info(
128
+ f"Waiting for JobRun {job_run_data.get('id')} to be in_progress. Current status: {job_run_status}"
149
129
  )
150
- continue
130
+ sleep(1)
151
131
 
152
- try:
153
- runner.execute_job_run()
154
- except Exception as exception:
155
- logger.exception(f"Exception while executing job: {exception}")
156
- self.primitive.jobs.job_run_update(
157
- id=job_run_id,
158
- status="request_completed",
159
- conclusion="failure",
160
- )
161
- finally:
162
- runner.cleanup()
132
+ # the backend has said it's okay for the agent to run this job
133
+ if job_run_status == "in_progress":
134
+ try:
135
+ runner = Runner(
136
+ primitive=self.primitive,
137
+ job_run=job_run_data,
138
+ )
139
+ runner.setup()
140
+ except Exception as exception:
141
+ logger.exception(
142
+ f"Exception while initializing runner: {exception}"
143
+ )
144
+ self.primitive.jobs.job_run_update(
145
+ id=job_run_id,
146
+ status="request_completed",
147
+ conclusion="failure",
148
+ )
149
+ continue
150
+
151
+ try:
152
+ runner.execute_job_run()
153
+ except Exception as exception:
154
+ logger.exception(f"Exception while executing job: {exception}")
155
+ self.primitive.jobs.job_run_update(
156
+ id=job_run_id,
157
+ status="request_completed",
158
+ conclusion="failure",
159
+ )
160
+ finally:
161
+ runner.cleanup()
163
162
 
164
- # NOTE: also run scan here to force upload of artifacts
165
- # This should probably eventually be another daemon?
166
- uploader.scan()
163
+ # NOTE: also run scan here to force upload of artifacts
164
+ # This should probably eventually be another daemon?
165
+ uploader.scan()
167
166
 
168
- if RUNNING_IN_CONTAINER:
169
- logger.info("Running in container, exiting after job run")
170
- break
167
+ if RUNNING_IN_CONTAINER:
168
+ logger.info("Running in container, exiting after job run")
169
+ break
171
170
 
172
171
  sleep(5)
173
172
  except KeyboardInterrupt:
primitive/agent/runner.py CHANGED
@@ -242,8 +242,14 @@ class Runner:
242
242
  )
243
243
 
244
244
  try:
245
+ while process.pid is None:
246
+ logger.debug(
247
+ f"Waiting for process PID to be set for command {i + 1}/{len(commands)}: {cmd}"
248
+ )
249
+ await asyncio.sleep(1)
250
+ logger.debug(f"Process started with PID {process.pid} for command")
245
251
  await self.primitive.jobs.ajob_run_update(
246
- self.job_run["id"],
252
+ id=self.job_run["id"],
247
253
  parent_pid=process.pid,
248
254
  )
249
255
  except ValueError:
@@ -46,22 +46,24 @@ class Uploader:
46
46
 
47
47
  for file in files:
48
48
  try:
49
- result = self.primitive.files.upload_file_direct(
49
+ upload_file_result = self.primitive.files.upload_file_direct(
50
50
  path=file,
51
51
  key_prefix=str(PurePath(file).relative_to(cache.parent).parent),
52
52
  )
53
- except Exception as exception:
54
- if "is empty" in str(exception):
55
- logger.warning(f"{file} is empty, skipping upload")
56
- continue
57
53
 
58
- upload_id = result.data["fileUpdate"]["id"]
54
+ if upload_file_result and upload_file_result.data is not None:
55
+ upload_file_data = upload_file_result.data
56
+ upload_id = upload_file_data.get("fileUpdate", {}).get("id")
59
57
 
60
- if upload_id:
61
- file_ids.append(upload_id)
62
- continue
58
+ if upload_id:
59
+ file_ids.append(upload_id)
60
+ continue
63
61
 
64
- logger.error(f"Unable to upload file {file}")
62
+ logger.error(f"Unable to upload file {file}")
63
+ except Exception as exception:
64
+ if "is empty" in str(exception):
65
+ logger.warning(f"{file} is empty, skipping upload")
66
+ continue
65
67
 
66
68
  # Clean up job cache
67
69
  shutil.rmtree(path=cache)
@@ -67,7 +67,6 @@ class Monitor(BaseAction):
67
67
  # handles cleanup of old reservations
68
68
  # obtains an active JobRun's ID
69
69
  if not RUNNING_IN_CONTAINER:
70
- self.primitive.hardware.check_in_http(is_online=True)
71
70
  hardware = self.primitive.hardware.get_own_hardware_details()
72
71
  # fetch the latest hardware and activeReservation details
73
72
  if active_reservation_data := hardware["activeReservation"]:
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: primitive
3
- Version: 0.2.52
3
+ Version: 0.2.54
4
4
  Project-URL: Documentation, https://github.com//primitivecorp/primitive-cli#readme
5
5
  Project-URL: Issues, https://github.com//primitivecorp/primitive-cli/issues
6
6
  Project-URL: Source, https://github.com//primitivecorp/primitive-cli
@@ -1,12 +1,12 @@
1
- primitive/__about__.py,sha256=UtAiRLmZsc3f1bwn5K2VjhCbBzl-o3wVd6tEGPtrw_M,130
1
+ primitive/__about__.py,sha256=3GVdNm566EatiamrqY12JpaMf-SbFWKLhamV3dQjvPE,130
2
2
  primitive/__init__.py,sha256=bwKdgggKNVssJFVPfKSxqFMz4IxSr54WWbmiZqTMPNI,106
3
3
  primitive/cli.py,sha256=g7EtHI9MATAB0qQu5w-WzbXtxz_8zu8z5E7sETmMkKU,2509
4
4
  primitive/client.py,sha256=RMF46F89oK82gfZH6Bf0WZrhXPUu01pbieSO_Vcuoc4,3624
5
5
  primitive/agent/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
6
- primitive/agent/actions.py,sha256=5lWnhLX8pH972u9WBbgmx3zIUrHhB5KXON4ofPvh84Q,7871
6
+ primitive/agent/actions.py,sha256=-vHuwZZv8tWfq5SSljr18tUJz3BbyodrfdXKZvvrWEM,7993
7
7
  primitive/agent/commands.py,sha256=o847pK7v7EWQGG67tky6a33qtwoutX6LZrP2FIS_NOk,388
8
- primitive/agent/runner.py,sha256=deMLa4l758k-BeK7Up4LVeD9hXfmAf6o3jJh8OmURys,11315
9
- primitive/agent/uploader.py,sha256=ZzrzsajNBogwEC7mT6Ejy0h2Jd9axMYGzt9pbCvVMlk,3171
8
+ primitive/agent/runner.py,sha256=1ceYsEY1mYTLzvsm3vIT8UgohxyvEQZVrErFci5rcM0,11648
9
+ primitive/agent/uploader.py,sha256=DT_Nzt5eOTm_uRcYKm1sjBBaQZzp5iNZ_uN5XktfQ30,3382
10
10
  primitive/auth/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
11
11
  primitive/auth/actions.py,sha256=9NIEXJ1BNJutJs6AMMSjMN_ziONUAUhY_xHwojYJCLA,942
12
12
  primitive/auth/commands.py,sha256=Krm38ioduDJZw0OIrIcb6eR2X6iECiWX0JPAI-2HNxY,2352
@@ -54,7 +54,7 @@ primitive/jobs/graphql/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3
54
54
  primitive/jobs/graphql/fragments.py,sha256=Ldhsc-NFHZGY8_1FjO2uZIdOL5UUGZjsldHea_GLJ_g,667
55
55
  primitive/jobs/graphql/mutations.py,sha256=8ASvCmwQh7cMeeiykOdYaYVryG8FRIuVF6v_J8JJZuw,219
56
56
  primitive/jobs/graphql/queries.py,sha256=ZxNmm-WovytbggNuKRnwa0kc26T34_0yhqkoqx-2uj0,1736
57
- primitive/monitor/actions.py,sha256=vXj6JQC17qNjoVUkOhnvOWMjVIuNUH2jKys3k599Bj8,9632
57
+ primitive/monitor/actions.py,sha256=nXa8UZT0dXVZma39KDnIpxHt3147xWpnflLIHGOQHpg,9558
58
58
  primitive/monitor/commands.py,sha256=VDlEL_Qpm_ysHxug7VpI0cVAZ0ny6AS91Y58D7F1zkU,409
59
59
  primitive/organizations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
60
60
  primitive/organizations/actions.py,sha256=kVHOhG1oS2sI5p8uldSo5L-RUZsnG36eaulVuKLyZ-M,1863
@@ -95,8 +95,8 @@ primitive/utils/printer.py,sha256=f1XUpqi5dkTL3GWvYRUGlSwtj2IxU1q745T4Fxo7Tn4,37
95
95
  primitive/utils/psutil.py,sha256=xa7ef435UL37jyjmUPbEqCO2ayQMpCs0HCrxVEvLcuM,763
96
96
  primitive/utils/shell.py,sha256=Z4zxmOaSyGCrS0D6I436iQci-ewHLt4UxVg1CD9Serc,2171
97
97
  primitive/utils/text.py,sha256=XiESMnlhjQ534xE2hMNf08WehE1SKaYFRNih0MmnK0k,829
98
- primitive-0.2.52.dist-info/METADATA,sha256=mDQ41sAqkHcxC1iAP4_9lXskUln2QrHVZllTW1HXoII,3513
99
- primitive-0.2.52.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
100
- primitive-0.2.52.dist-info/entry_points.txt,sha256=p1K8DMCWka5FqLlqP1sPek5Uovy9jq8u51gUsP-z334,48
101
- primitive-0.2.52.dist-info/licenses/LICENSE.txt,sha256=B8kmQMJ2sxYygjCLBk770uacaMci4mPSoJJ8WoDBY_c,1098
102
- primitive-0.2.52.dist-info/RECORD,,
98
+ primitive-0.2.54.dist-info/METADATA,sha256=TCd6axGqx7iJZeVaDAVzXrBrSzRRtaRgtqRDmcLjc6A,3513
99
+ primitive-0.2.54.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
100
+ primitive-0.2.54.dist-info/entry_points.txt,sha256=p1K8DMCWka5FqLlqP1sPek5Uovy9jq8u51gUsP-z334,48
101
+ primitive-0.2.54.dist-info/licenses/LICENSE.txt,sha256=B8kmQMJ2sxYygjCLBk770uacaMci4mPSoJJ8WoDBY_c,1098
102
+ primitive-0.2.54.dist-info/RECORD,,