diracx-db 0.0.1a19__py3-none-any.whl → 0.0.1a21__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,574 @@
1
+ import asyncio
2
+ from collections import defaultdict
3
+ from copy import deepcopy
4
+ from datetime import datetime, timezone
5
+ from typing import Any
6
+ from unittest.mock import MagicMock
7
+
8
+ from fastapi import BackgroundTasks
9
+ from pydantic import BaseModel
10
+
11
+ from diracx.core.config.schema import Config
12
+ from diracx.core.models import (
13
+ JobMinorStatus,
14
+ JobStatus,
15
+ JobStatusUpdate,
16
+ SetJobStatusReturn,
17
+ VectorSearchOperator,
18
+ VectorSearchSpec,
19
+ )
20
+ from diracx.db.sql.job_logging.db import JobLoggingRecord
21
+
22
+ from .. import JobDB, JobLoggingDB, SandboxMetadataDB, TaskQueueDB
23
+
24
+
25
+ class JobSubmissionSpec(BaseModel):
26
+ jdl: str
27
+ owner: str
28
+ owner_group: str
29
+ initial_status: str
30
+ initial_minor_status: str
31
+ vo: str
32
+
33
+
34
+ async def submit_jobs_jdl(jobs: list[JobSubmissionSpec], job_db: JobDB):
35
+ from DIRAC.Core.Utilities.ClassAd.ClassAdLight import ClassAd
36
+ from DIRAC.Core.Utilities.ReturnValues import returnValueOrRaise
37
+ from DIRAC.WorkloadManagementSystem.DB.JobDBUtils import (
38
+ checkAndAddOwner,
39
+ createJDLWithInitialStatus,
40
+ )
41
+
42
+ jobs_to_insert = {}
43
+ jdls_to_update = {}
44
+ inputdata_to_insert = {}
45
+ original_jdls = []
46
+
47
+ # generate the jobIDs first
48
+ # TODO: should ForgivingTaskGroup be used?
49
+ async with asyncio.TaskGroup() as tg:
50
+ for job in jobs:
51
+ original_jdl = deepcopy(job.jdl)
52
+ jobManifest = returnValueOrRaise(
53
+ checkAndAddOwner(original_jdl, job.owner, job.owner_group)
54
+ )
55
+
56
+ # Fix possible lack of brackets
57
+ if original_jdl.strip()[0] != "[":
58
+ original_jdl = f"[{original_jdl}]"
59
+
60
+ original_jdls.append(
61
+ (
62
+ original_jdl,
63
+ jobManifest,
64
+ tg.create_task(job_db.create_job(original_jdl)),
65
+ )
66
+ )
67
+
68
+ async with asyncio.TaskGroup() as tg:
69
+ for job, (original_jdl, jobManifest_, job_id_task) in zip(jobs, original_jdls):
70
+ job_id = job_id_task.result()
71
+ job_attrs = {
72
+ "JobID": job_id,
73
+ "LastUpdateTime": datetime.now(tz=timezone.utc),
74
+ "SubmissionTime": datetime.now(tz=timezone.utc),
75
+ "Owner": job.owner,
76
+ "OwnerGroup": job.owner_group,
77
+ "VO": job.vo,
78
+ }
79
+
80
+ jobManifest_.setOption("JobID", job_id)
81
+
82
+ # 2.- Check JDL and Prepare DIRAC JDL
83
+ jobJDL = jobManifest_.dumpAsJDL()
84
+
85
+ # Replace the JobID placeholder if any
86
+ if jobJDL.find("%j") != -1:
87
+ jobJDL = jobJDL.replace("%j", str(job_id))
88
+
89
+ class_ad_job = ClassAd(jobJDL)
90
+
91
+ class_ad_req = ClassAd("[]")
92
+ if not class_ad_job.isOK():
93
+ # Rollback the entire transaction
94
+ raise ValueError(f"Error in JDL syntax for job JDL: {original_jdl}")
95
+ # TODO: check if that is actually true
96
+ if class_ad_job.lookupAttribute("Parameters"):
97
+ raise NotImplementedError("Parameters in the JDL are not supported")
98
+
99
+ # TODO is this even needed?
100
+ class_ad_job.insertAttributeInt("JobID", job_id)
101
+
102
+ await job_db.checkAndPrepareJob(
103
+ job_id,
104
+ class_ad_job,
105
+ class_ad_req,
106
+ job.owner,
107
+ job.owner_group,
108
+ job_attrs,
109
+ job.vo,
110
+ )
111
+ jobJDL = createJDLWithInitialStatus(
112
+ class_ad_job,
113
+ class_ad_req,
114
+ job_db.jdl2DBParameters,
115
+ job_attrs,
116
+ job.initial_status,
117
+ job.initial_minor_status,
118
+ modern=True,
119
+ )
120
+
121
+ jobs_to_insert[job_id] = job_attrs
122
+ jdls_to_update[job_id] = jobJDL
123
+
124
+ if class_ad_job.lookupAttribute("InputData"):
125
+ inputData = class_ad_job.getListFromExpression("InputData")
126
+ inputdata_to_insert[job_id] = [lfn for lfn in inputData if lfn]
127
+
128
+ tg.create_task(job_db.update_job_jdls(jdls_to_update))
129
+ tg.create_task(job_db.insert_job_attributes(jobs_to_insert))
130
+
131
+ if inputdata_to_insert:
132
+ tg.create_task(job_db.insert_input_data(inputdata_to_insert))
133
+
134
+ return list(jobs_to_insert.keys())
135
+
136
+
137
+ async def reschedule_jobs_bulk(
138
+ job_ids: list[int],
139
+ config: Config,
140
+ job_db: JobDB,
141
+ job_logging_db: JobLoggingDB,
142
+ task_queue_db: TaskQueueDB,
143
+ background_task: BackgroundTasks,
144
+ *,
145
+ reset_counter=False,
146
+ ) -> dict[str, Any]:
147
+ """Reschedule given job."""
148
+ from DIRAC.Core.Utilities.ClassAd.ClassAdLight import ClassAd
149
+ from DIRAC.Core.Utilities.ReturnValues import SErrorException
150
+
151
+ failed = {}
152
+ reschedule_max = config.Operations[
153
+ "Defaults"
154
+ ].Services.JobScheduling.MaxRescheduling # type: ignore
155
+
156
+ status_changes = {}
157
+ attribute_changes: dict[int, dict[str, str]] = defaultdict(dict)
158
+ jdl_changes = {}
159
+
160
+ _, results = await job_db.search(
161
+ parameters=[
162
+ "Status",
163
+ "MinorStatus",
164
+ "VerifiedFlag",
165
+ "RescheduleCounter",
166
+ "Owner",
167
+ "OwnerGroup",
168
+ "JobID",
169
+ ],
170
+ search=[
171
+ VectorSearchSpec(
172
+ parameter="JobID", operator=VectorSearchOperator.IN, values=job_ids
173
+ )
174
+ ],
175
+ sorts=[],
176
+ )
177
+ if not results:
178
+ for job_id in job_ids:
179
+ failed[job_id] = {"detail": "Not found"}
180
+
181
+ jobs_to_resched = {}
182
+
183
+ for job_attrs in results or []:
184
+ job_id = int(job_attrs["JobID"])
185
+
186
+ if "VerifiedFlag" not in job_attrs:
187
+ failed[job_id] = {"detail": "Not found: No verified flag"}
188
+ # Noop
189
+ continue
190
+
191
+ if not job_attrs["VerifiedFlag"]:
192
+ failed[job_id] = {
193
+ "detail": (
194
+ f"VerifiedFlag is False: Status {job_attrs['Status']}, "
195
+ f"Minor Status: {job_attrs['MinorStatus']}"
196
+ )
197
+ }
198
+ # Noop
199
+ continue
200
+
201
+ if reset_counter:
202
+ job_attrs["RescheduleCounter"] = 0
203
+ else:
204
+ job_attrs["RescheduleCounter"] = int(job_attrs["RescheduleCounter"]) + 1
205
+
206
+ if job_attrs["RescheduleCounter"] > reschedule_max:
207
+ status_changes[job_id] = {
208
+ datetime.now(tz=timezone.utc): JobStatusUpdate(
209
+ Status=JobStatus.FAILED,
210
+ MinorStatus=JobMinorStatus.MAX_RESCHEDULING,
211
+ ApplicationStatus="Unknown",
212
+ )
213
+ }
214
+ failed[job_id] = {
215
+ "detail": f"Maximum number of reschedules exceeded ({reschedule_max})"
216
+ }
217
+ # DATABASE OPERATION (status change)
218
+ continue
219
+ jobs_to_resched[job_id] = job_attrs
220
+
221
+ surviving_job_ids = set(jobs_to_resched.keys())
222
+
223
+ # TODO: get the job parameters from JobMonitoringClient
224
+ # result = JobMonitoringClient().getJobParameters(jobID)
225
+ # if result["OK"]:
226
+ # parDict = result["Value"]
227
+ # for key, value in parDict.get(jobID, {}).items():
228
+ # result = self.setAtticJobParameter(jobID, key, value, rescheduleCounter - 1)
229
+ # if not result["OK"]:
230
+ # break
231
+
232
+ # TODO: IF we keep JobParameters and OptimizerParameters: Delete job in those tables.
233
+ # await self.delete_job_parameters(job_id)
234
+ # await self.delete_job_optimizer_parameters(job_id)
235
+
236
+ def parse_jdl(job_id, job_jdl):
237
+ if not job_jdl.strip().startswith("["):
238
+ job_jdl = f"[{job_jdl}]"
239
+ class_ad_job = ClassAd(job_jdl)
240
+ class_ad_job.insertAttributeInt("JobID", job_id)
241
+ return class_ad_job
242
+
243
+ job_jdls = {
244
+ jobid: parse_jdl(jobid, jdl)
245
+ for jobid, jdl in (
246
+ (await job_db.getJobJDLs(surviving_job_ids, original=True)).items()
247
+ )
248
+ }
249
+
250
+ for job_id in surviving_job_ids:
251
+ class_ad_job = job_jdls[job_id]
252
+ class_ad_req = ClassAd("[]")
253
+ try:
254
+ await job_db.checkAndPrepareJob(
255
+ job_id,
256
+ class_ad_job,
257
+ class_ad_req,
258
+ jobs_to_resched[job_id]["Owner"],
259
+ jobs_to_resched[job_id]["OwnerGroup"],
260
+ {"RescheduleCounter": jobs_to_resched[job_id]["RescheduleCounter"]},
261
+ class_ad_job.getAttributeString("VirtualOrganization"),
262
+ )
263
+ except SErrorException as e:
264
+ failed[job_id] = {"detail": str(e)}
265
+ # surviving_job_ids.remove(job_id)
266
+ continue
267
+
268
+ priority = class_ad_job.getAttributeInt("Priority")
269
+ if priority is None:
270
+ priority = 0
271
+
272
+ site_list = class_ad_job.getListFromExpression("Site")
273
+ if not site_list:
274
+ site = "ANY"
275
+ elif len(site_list) > 1:
276
+ site = "Multiple"
277
+ else:
278
+ site = site_list[0]
279
+
280
+ reqJDL = class_ad_req.asJDL()
281
+ class_ad_job.insertAttributeInt("JobRequirements", reqJDL)
282
+ jobJDL = class_ad_job.asJDL()
283
+ # Replace the JobID placeholder if any
284
+ jobJDL = jobJDL.replace("%j", str(job_id))
285
+
286
+ additional_attrs = {
287
+ "Site": site,
288
+ "UserPriority": priority,
289
+ "RescheduleTime": datetime.now(tz=timezone.utc),
290
+ "RescheduleCounter": jobs_to_resched[job_id]["RescheduleCounter"],
291
+ }
292
+
293
+ # set new JDL
294
+ jdl_changes[job_id] = jobJDL
295
+
296
+ # set new status
297
+ status_changes[job_id] = {
298
+ datetime.now(tz=timezone.utc): JobStatusUpdate(
299
+ Status=JobStatus.RECEIVED,
300
+ MinorStatus=JobMinorStatus.RESCHEDULED,
301
+ ApplicationStatus="Unknown",
302
+ )
303
+ }
304
+ # set new attributes
305
+ attribute_changes[job_id].update(additional_attrs)
306
+
307
+ if surviving_job_ids:
308
+ # BULK STATUS UPDATE
309
+ # DATABASE OPERATION
310
+ set_job_status_result = await set_job_status_bulk(
311
+ status_changes,
312
+ config,
313
+ job_db,
314
+ job_logging_db,
315
+ task_queue_db,
316
+ background_task,
317
+ additional_attributes=attribute_changes,
318
+ )
319
+
320
+ # BULK JDL UPDATE
321
+ # DATABASE OPERATION
322
+ await job_db.setJobJDLsBulk(jdl_changes)
323
+
324
+ return {
325
+ "failed": failed,
326
+ "success": {
327
+ job_id: {
328
+ "InputData": job_jdls[job_id],
329
+ **attribute_changes[job_id],
330
+ **set_status_result.model_dump(),
331
+ }
332
+ for job_id, set_status_result in set_job_status_result.success.items()
333
+ },
334
+ }
335
+
336
+ return {
337
+ "success": [],
338
+ "failed": failed,
339
+ }
340
+
341
+
342
+ async def set_job_status_bulk(
343
+ status_changes: dict[int, dict[datetime, JobStatusUpdate]],
344
+ config: Config,
345
+ job_db: JobDB,
346
+ job_logging_db: JobLoggingDB,
347
+ task_queue_db: TaskQueueDB,
348
+ background_task: BackgroundTasks,
349
+ *,
350
+ force: bool = False,
351
+ additional_attributes: dict[int, dict[str, str]] = {},
352
+ ) -> SetJobStatusReturn:
353
+ """Set various status fields for job specified by its jobId.
354
+ Set only the last status in the JobDB, updating all the status
355
+ logging information in the JobLoggingDB. The status dict has datetime
356
+ as a key and status information dictionary as values.
357
+
358
+ :raises: JobNotFound if the job is not found in one of the DBs
359
+ """
360
+ from DIRAC.Core.Utilities import TimeUtilities
361
+ from DIRAC.Core.Utilities.ReturnValues import returnValueOrRaise
362
+ from DIRAC.WorkloadManagementSystem.Utilities.JobStatusUtility import (
363
+ getNewStatus,
364
+ getStartAndEndTime,
365
+ )
366
+
367
+ failed: dict[int, Any] = {}
368
+ deletable_killable_jobs = set()
369
+ job_attribute_updates: dict[int, dict[str, str]] = {}
370
+ job_logging_updates: list[JobLoggingRecord] = []
371
+ status_dicts: dict[int, dict[datetime, dict[str, str]]] = defaultdict(dict)
372
+
373
+ # transform JobStateUpdate objects into dicts
374
+ status_dicts = {
375
+ job_id: {
376
+ key: {k: v for k, v in value.model_dump().items() if v is not None}
377
+ for key, value in status.items()
378
+ }
379
+ for job_id, status in status_changes.items()
380
+ }
381
+
382
+ # search all jobs at once
383
+ _, results = await job_db.search(
384
+ parameters=["Status", "StartExecTime", "EndExecTime", "JobID"],
385
+ search=[
386
+ {
387
+ "parameter": "JobID",
388
+ "operator": VectorSearchOperator.IN,
389
+ "values": list(set(status_changes.keys())),
390
+ }
391
+ ],
392
+ sorts=[],
393
+ )
394
+ if not results:
395
+ return SetJobStatusReturn(
396
+ success={},
397
+ failed={
398
+ int(job_id): {"detail": "Not found"} for job_id in status_changes.keys()
399
+ },
400
+ )
401
+
402
+ found_jobs = set(int(res["JobID"]) for res in results)
403
+ failed.update(
404
+ {
405
+ int(nf_job_id): {"detail": "Not found"}
406
+ for nf_job_id in set(status_changes.keys()) - found_jobs
407
+ }
408
+ )
409
+ # Get the latest time stamps of major status updates
410
+ wms_time_stamps = await job_logging_db.get_wms_time_stamps_bulk(found_jobs)
411
+
412
+ for res in results:
413
+ job_id = int(res["JobID"])
414
+ currentStatus = res["Status"]
415
+ startTime = res["StartExecTime"]
416
+ endTime = res["EndExecTime"]
417
+
418
+ # If the current status is Stalled and we get an update, it should probably be "Running"
419
+ if currentStatus == JobStatus.STALLED:
420
+ currentStatus = JobStatus.RUNNING
421
+
422
+ #####################################################################################################
423
+ statusDict = status_dicts[job_id]
424
+ # This is more precise than "LastTime". timeStamps is a sorted list of tuples...
425
+ timeStamps = sorted((float(t), s) for s, t in wms_time_stamps[job_id].items())
426
+ lastTime = TimeUtilities.fromEpoch(timeStamps[-1][0]).replace(
427
+ tzinfo=timezone.utc
428
+ )
429
+
430
+ # Get chronological order of new updates
431
+ updateTimes = sorted(statusDict)
432
+
433
+ newStartTime, newEndTime = getStartAndEndTime(
434
+ startTime, endTime, updateTimes, timeStamps, statusDict
435
+ )
436
+
437
+ job_data: dict[str, str] = {}
438
+ if updateTimes[-1] >= lastTime:
439
+ new_status, new_minor, new_application = (
440
+ returnValueOrRaise( # TODO: Catch this
441
+ getNewStatus(
442
+ job_id,
443
+ updateTimes,
444
+ lastTime,
445
+ statusDict,
446
+ currentStatus,
447
+ force,
448
+ MagicMock(), # FIXME
449
+ )
450
+ )
451
+ )
452
+
453
+ if new_status:
454
+ job_data.update(additional_attributes.get(job_id, {}))
455
+ job_data["Status"] = new_status
456
+ job_data["LastUpdateTime"] = str(datetime.now(timezone.utc))
457
+ if new_minor:
458
+ job_data["MinorStatus"] = new_minor
459
+ if new_application:
460
+ job_data["ApplicationStatus"] = new_application
461
+
462
+ # TODO: implement elasticJobParametersDB ?
463
+ # if cls.elasticJobParametersDB:
464
+ # result = cls.elasticJobParametersDB.setJobParameter(int(jobID), "Status", status)
465
+ # if not result["OK"]:
466
+ # return result
467
+
468
+ for updTime in updateTimes:
469
+ if statusDict[updTime]["Source"].startswith("Job"):
470
+ job_data["HeartBeatTime"] = str(updTime)
471
+
472
+ if not startTime and newStartTime:
473
+ job_data["StartExecTime"] = newStartTime
474
+
475
+ if not endTime and newEndTime:
476
+ job_data["EndExecTime"] = newEndTime
477
+
478
+ #####################################################################################################
479
+ # delete or kill job, if we transition to DELETED or KILLED state
480
+ if new_status in [JobStatus.DELETED, JobStatus.KILLED]:
481
+ deletable_killable_jobs.add(job_id)
482
+
483
+ # Update database tables
484
+ if job_data:
485
+ job_attribute_updates[job_id] = job_data
486
+
487
+ for updTime in updateTimes:
488
+ sDict = statusDict[updTime]
489
+ job_logging_updates.append(
490
+ JobLoggingRecord(
491
+ job_id=job_id,
492
+ status=sDict.get("Status", "idem"),
493
+ minor_status=sDict.get("MinorStatus", "idem"),
494
+ application_status=sDict.get("ApplicationStatus", "idem"),
495
+ date=updTime,
496
+ source=sDict.get("Source", "Unknown"),
497
+ )
498
+ )
499
+
500
+ await job_db.setJobAttributesBulk(job_attribute_updates)
501
+
502
+ await remove_jobs_from_task_queue(
503
+ list(deletable_killable_jobs), config, task_queue_db, background_task
504
+ )
505
+
506
+ # TODO: implement StorageManagerClient
507
+ # returnValueOrRaise(StorageManagerClient().killTasksBySourceTaskID(job_ids))
508
+
509
+ if deletable_killable_jobs:
510
+ await job_db.set_job_command_bulk(
511
+ [(job_id, "Kill", "") for job_id in deletable_killable_jobs]
512
+ )
513
+
514
+ await job_logging_db.bulk_insert_record(job_logging_updates)
515
+
516
+ return SetJobStatusReturn(
517
+ success=job_attribute_updates,
518
+ failed=failed,
519
+ )
520
+
521
+
522
+ async def remove_jobs(
523
+ job_ids: list[int],
524
+ config: Config,
525
+ job_db: JobDB,
526
+ job_logging_db: JobLoggingDB,
527
+ sandbox_metadata_db: SandboxMetadataDB,
528
+ task_queue_db: TaskQueueDB,
529
+ background_task: BackgroundTasks,
530
+ ):
531
+ """Fully remove a job from the WMS databases.
532
+ :raises: nothing.
533
+ """
534
+ # Remove the staging task from the StorageManager
535
+ # TODO: this was not done in the JobManagerHandler, but it was done in the kill method
536
+ # I think it should be done here too
537
+ # TODO: implement StorageManagerClient
538
+ # returnValueOrRaise(StorageManagerClient().killTasksBySourceTaskID([job_id]))
539
+
540
+ # TODO: this was also not done in the JobManagerHandler, but it was done in the JobCleaningAgent
541
+ # I think it should be done here as well
542
+ await sandbox_metadata_db.unassign_sandboxes_to_jobs(job_ids)
543
+
544
+ # Remove the job from TaskQueueDB
545
+ await remove_jobs_from_task_queue(job_ids, config, task_queue_db, background_task)
546
+
547
+ # Remove the job from JobLoggingDB
548
+ await job_logging_db.delete_records(job_ids)
549
+
550
+ # Remove the job from JobDB
551
+ await job_db.delete_jobs(job_ids)
552
+
553
+
554
+ async def remove_jobs_from_task_queue(
555
+ job_ids: list[int],
556
+ config: Config,
557
+ task_queue_db: TaskQueueDB,
558
+ background_task: BackgroundTasks,
559
+ ):
560
+ """Remove the job from TaskQueueDB."""
561
+ tq_infos = await task_queue_db.get_tq_infos_for_jobs(job_ids)
562
+ await task_queue_db.remove_jobs(job_ids)
563
+ for tq_id, owner, owner_group, vo in tq_infos:
564
+ # TODO: move to Celery
565
+ background_task.add_task(
566
+ task_queue_db.delete_task_queue_if_empty,
567
+ tq_id,
568
+ owner,
569
+ owner_group,
570
+ config.Registry[vo].Groups[owner_group].JobShare,
571
+ config.Registry[vo].Groups[owner_group].Properties,
572
+ config.Operations[vo].Services.JobScheduling.EnableSharesCorrection,
573
+ config.Registry[vo].Groups[owner_group].AllowBackgroundTQs,
574
+ )
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: diracx-db
3
- Version: 0.0.1a19
3
+ Version: 0.0.1a21
4
4
  Summary: TODO
5
5
  License: GPL-3.0-only
6
6
  Classifier: Intended Audience :: Science/Research
@@ -14,8 +14,7 @@ Requires-Dist: dirac
14
14
  Requires-Dist: diracx-core
15
15
  Requires-Dist: fastapi
16
16
  Requires-Dist: opensearch-py[async]
17
- Requires-Dist: pydantic>=2.4
17
+ Requires-Dist: pydantic>=2.10
18
18
  Requires-Dist: sqlalchemy[aiomysql,aiosqlite]>=2
19
19
  Provides-Extra: testing
20
20
  Requires-Dist: diracx-testing; extra == "testing"
21
-
@@ -13,10 +13,10 @@ diracx/db/sql/dummy/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSu
13
13
  diracx/db/sql/dummy/db.py,sha256=4Xyo7gUh_5b6Q2a_ggJG6e7fCtc9HrP_BRXfKGfqZIs,1642
14
14
  diracx/db/sql/dummy/schema.py,sha256=uEkGDNVZbmJecytkHY1CO-M1MiKxe5w1_h0joJMPC9E,680
15
15
  diracx/db/sql/job/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
16
- diracx/db/sql/job/db.py,sha256=DaU1SGeXl7TqX1QxT1RldCeMIOYGnzNwkqBrwgGE90A,16248
16
+ diracx/db/sql/job/db.py,sha256=54TjRzWBgD2xhM3ljdel399xq8ro4Z0k3c7zWlXXCUI,11616
17
17
  diracx/db/sql/job/schema.py,sha256=w9Ht9LyVK-fB5T9-hYGsqifzneeG2YP123j1-Mx8Xio,4283
18
18
  diracx/db/sql/job_logging/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
19
- diracx/db/sql/job_logging/db.py,sha256=OGVHYIDcWhGYVfHacsz9DEPSoJ7aRbKVoQOyCCj8XvU,5036
19
+ diracx/db/sql/job_logging/db.py,sha256=AQCjtdDajYmPNpAuLsLr24BILteRYFEs9oSlH5a3wng,7509
20
20
  diracx/db/sql/job_logging/schema.py,sha256=dD2arl-6bffeK8INT6tZ1HWEpJuYTx2iNiVzswVXXF8,812
21
21
  diracx/db/sql/pilot_agents/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
22
22
  diracx/db/sql/pilot_agents/db.py,sha256=7-cuCbh_KhM0jlybsHMWV-W66bHsPHIVBpbuqwjncj0,1232
@@ -27,10 +27,10 @@ diracx/db/sql/sandbox_metadata/schema.py,sha256=rngYYkJxBhjETBHGLD1CTipDGe44mRYR
27
27
  diracx/db/sql/task_queue/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
28
28
  diracx/db/sql/task_queue/db.py,sha256=e6yauZO0nWaUVqjqQycH8iPO4wXLXaC82eaIq1K_KI8,9102
29
29
  diracx/db/sql/task_queue/schema.py,sha256=fvzQyCw_xWAOWTLW6Qrp1m-WzEKb0tlYmafoLTbCy1I,3222
30
- diracx/db/sql/utils/__init__.py,sha256=8YcDra_P_FL3QIDzl53HvN1tHUE396F0tGKB-TPQxeM,15541
31
- diracx/db/sql/utils/job_status.py,sha256=GNQTKiyguhnB348mLIB7BT-PEOEKpKljR4JzvOd_h8M,10414
32
- diracx_db-0.0.1a19.dist-info/METADATA,sha256=_e0zgiSjl92T83BUpBXkEHzMeCcLcr4L_sE9h9v_iwQ,688
33
- diracx_db-0.0.1a19.dist-info/WHEEL,sha256=GV9aMThwP_4oNCtvEC2ec3qUYutgWeAzklro_0m4WJQ,91
34
- diracx_db-0.0.1a19.dist-info/entry_points.txt,sha256=YLI4f6640bri8Ud6Jt9WNq79pSTVQAkfUasb9f75fR8,315
35
- diracx_db-0.0.1a19.dist-info/top_level.txt,sha256=vJx10tdRlBX3rF2Psgk5jlwVGZNcL3m_7iQWwgPXt-U,7
36
- diracx_db-0.0.1a19.dist-info/RECORD,,
30
+ diracx/db/sql/utils/__init__.py,sha256=fANPhXofb3ghvnOeLXmcK33YiViJFG8gI2C-2AMArEs,15647
31
+ diracx/db/sql/utils/job.py,sha256=tr_XvQJ5srjgXEJI2A2UBvfKT8JQwvChCTnqe6h8aAU,19380
32
+ diracx_db-0.0.1a21.dist-info/METADATA,sha256=xU59UyfIXc45hodIh3hkuk-FWOw9z29p09jcyZlmriM,688
33
+ diracx_db-0.0.1a21.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
34
+ diracx_db-0.0.1a21.dist-info/entry_points.txt,sha256=YLI4f6640bri8Ud6Jt9WNq79pSTVQAkfUasb9f75fR8,315
35
+ diracx_db-0.0.1a21.dist-info/top_level.txt,sha256=vJx10tdRlBX3rF2Psgk5jlwVGZNcL3m_7iQWwgPXt-U,7
36
+ diracx_db-0.0.1a21.dist-info/RECORD,,
@@ -1,5 +1,5 @@
1
1
  Wheel-Version: 1.0
2
- Generator: setuptools (75.1.0)
2
+ Generator: setuptools (75.6.0)
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
5
5