diff --git a/src/DIRAC/Core/DISET/private/Transports/test/README.md b/src/DIRAC/Core/DISET/private/Transports/test/README.md index bd6b25eb94c..a0b70dec73a 100644 --- a/src/DIRAC/Core/DISET/private/Transports/test/README.md +++ b/src/DIRAC/Core/DISET/private/Transports/test/README.md @@ -1,5 +1,5 @@ Generated with ``` -dirac-proxy-init -C src/DIRAC/Core/Security/test/certs/user/usercert.pem -K src/DIRAC/Core/Security/test/certs/user/userkey.pem -u src/DIRAC/Core/DISET/private/Transports/test/proxy.pem +dirac-proxy-init -C src/DIRAC/Core/Security/test/certs/user/usercert.pem -K src/DIRAC/Core/Security/test/certs/user/userkey.pem -u src/DIRAC/Core/DISET/private/Transports/test/proxy.pem --valid 87600:00 ``` diff --git a/src/DIRAC/Core/DISET/private/Transports/test/proxy.pem b/src/DIRAC/Core/DISET/private/Transports/test/proxy.pem index b1558d908a3..c93e7ccc94b 100644 --- a/src/DIRAC/Core/DISET/private/Transports/test/proxy.pem +++ b/src/DIRAC/Core/DISET/private/Transports/test/proxy.pem @@ -1,55 +1,55 @@ -----BEGIN CERTIFICATE----- -MIIEOTCCAiGgAwIBAgIFASuiMw0wDQYJKoZIhvcNAQELBQAwOjEYMBYGA1UECgwP +MIIEOTCCAiGgAwIBAgIFAUXu3+AwDQYJKoZIhvcNAQELBQAwOjEYMBYGA1UECgwP RGlyYWMgQ29tcHV0aW5nMQ0wCwYDVQQKDARDRVJOMQ8wDQYDVQQDDAZNclVzZXIw -HhcNMjMxMTI3MTIyMjU3WhcNMjMxMTI4MTIzNzU3WjBPMRgwFgYDVQQKDA9EaXJh +HhcNMjMxMTI4MjExMDIzWhcNMzMxMTI1MjEyNTIzWjBPMRgwFgYDVQQKDA9EaXJh YyBDb21wdXRpbmcxDTALBgNVBAoMBENFUk4xDzANBgNVBAMMBk1yVXNlcjETMBEG -A1UEAwwKNTAyNzAxNzQ4NTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEB -AMMJhgkvJQLX82UTSYawiWqFQ/Hq8N9WxSXKv6mCkL8tuDwD4ojAkj8/n8R7Hx0k -Q10y/k3O7lSXQeh6JJRTatdWL4jFV3WQf0sfulcvkIzMgRb/y9rjpWnKiZRbR5dh -aeWRDkP8CR93bdY6yqx6bdvVUyhRDIJURCzedXjnebHYCzDmW4puo0aA5+uSHzC6 -uRDH1rGf8lihwAWWCLRmYommc9XD4uoH1ejjlYvuJdfHSZHjHpqj21op6oNnC/We -DmnqkeyJSIR2oGeZYuBRXhE9Nf3mYs/rHim6X/5/WXmWsx7+K4NKf9aFPjxoTPRN -n+O4AY8Musj9j8sKzg41eOcCAwEAAaMxMC8wDgYDVR0PAQH/BAQDAgSwMB0GCCsG -AQUFBwEOAQH/BA4wDDAKBggrBgEFBQcVATANBgkqhkiG9w0BAQsFAAOCAgEASc7n -+732nOr1stq0qIy9NZi+zbhILIU4UR1ZO8LNhvQ1JPpyGfPEFqWFmFyuwaiwZRkS -MWQwY7NzJIq5i0FWNNWmuNT76M5YY1hwVZFL4F4knDWwgW08AtxXCN1A7SUNTXG5 -bP9E8h6Rh1c8lOlPxPEkEbrccxoOVUG0xP4BvDR6oU9yHdh2CZiZNvteh160HYz0 -mPciFcK9ee0Sb2ylwBYY0T9gIabU/F3jUf663TMFevhxhVw/rcKoNOPHHK9Gd/QN -RH74/1uV4JqJEb51xcE0GJAduyCc7MzBK9mpf0ejfA8+Lm+oYtaroDo/5vqPmPX4 -FcXsP153kBObRYvZv550hyO5KhB370F2QCr1jU6/1g4FbcoONdRJQc+Hgz0UWyY3 -hVXYm3WsfpyFCRRR7aIf/nOA2X/fAgs7RCoFqItJOVC9dtBuhHbdcGTdWJneAb1k -1Zy3DL7y0j3JFun1WeooHmSJcIfNpLwztmFl9LcgNRf5DjJdItXEIWVICqTJht7i -i4t88P9Eof63dF0LqTjR1HU+cOmy7Mc8Kqk36vK/y6Suw5nQf0knmCRJcAMc4+vm -5ADrMoCGuNeZNOzTr0GW52iS+6jOefe3e6pbCh4zSMwHDjhkYsYVI92Pc0ifBX0I -OP9NOPUGs/ZPOzQTvhSytpoYkjtE06atEe/xwUI= +A1UEAwwKNTQ2ODI1MDA4MDCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEB +AP3qqOreMUBzNxGFCuqGb1Oh8H0oHRvkxptVixOw7+ckg36Lxn8l0tpHBCuAJeDQ +w8lKCix5SDVBK/FsKBhCCln8uMlv9dM+X8kez2rPFc0Hp6L6W5zEzW/0CbPs4Zwd +xvWysGcC3dVlxhXw/UAsUC4aJtD74VQGXUI03y7ozh/UmjUDTt5m33UIlIZEisPI +YZpA+6RMgNLJXx8EOhrdCJI3oD45+mBMqZUvRGWRAsIWYRr1jAdSmIEGLvLx/7dt +Dmi7L9Y4ZLcz7Wal83OWcUp1uX1y+k5yTss3SWzN2GLqQZIHL1XmgfOwrdqe50qa +Q3ii2qdWbH6ldS6ap2mGrrsCAwEAAaMxMC8wDgYDVR0PAQH/BAQDAgSwMB0GCCsG +AQUFBwEOAQH/BA4wDDAKBggrBgEFBQcVATANBgkqhkiG9w0BAQsFAAOCAgEABskd +Tjksd8cMhz7Ar8waASgIj9mjf/Jhg8L1BIMYKsCMi8rVTGX5Wl2F2WLfiMrp0RqO +MxSUk9sJ6b5JjrXNpvTQzFm4AKemmMC40k1N4FjkQngVemE0xiHxanxLtrggO2/9 +zxgTobEqaXV94gBXcRLrhIvHTcEDPekc9hBY6RRdH7PA7jpyZblF+RuwIbSCC+vM +JvL3PY5PiYRgzxeC1qDKXjo/NHO0cQ/LhjKDZoguFz8lYctLLezlCyocj0Wl5zwm +kbzejs9mA39UOYC7HYGJbfXXqEYjnNBtfcPE6naW5J0C26J6WPMrs+hm0jshIli5 +c1aQraEm9IYzY2s4ovmx0ZGfGUdnSOfbamJpYQLVsY0BNh4yzp5Qn3jEJm3ktoQ5 +0U54biUYbbUfg1WdthBqi+Zogk9+AbJ8b+C2ZzDwI/QsF5r9jPbh/J+dAshVUquz +UbCmERbq5dfNNO3qp1DAfCSAu20mDnO6ig105aU+58beL8Y6FphmZWm9JaIDnln4 +TXq1YGv+IuaYftuoBF/I2oa5k3WfteNVki7widk/bx3fW1luYitSROjsVDiU1ZkW +CWjclXEmsXKL/44mqx9iPzVAy3zhfkJ84gLr7gdhlnCP2KHJpyJWb9JOWFYi+fwf +2jTePuiDTA63Qb6/o4kLx3NR9fRO6G8/AZw8lvo= -----END CERTIFICATE----- -----BEGIN PRIVATE KEY----- -MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQDDCYYJLyUC1/Nl -E0mGsIlqhUPx6vDfVsUlyr+pgpC/Lbg8A+KIwJI/P5/Eex8dJENdMv5Nzu5Ul0Ho -eiSUU2rXVi+IxVd1kH9LH7pXL5CMzIEW/8va46VpyomUW0eXYWnlkQ5D/Akfd23W -Osqsem3b1VMoUQyCVEQs3nV453mx2Asw5luKbqNGgOfrkh8wurkQx9axn/JYocAF -lgi0ZmKJpnPVw+LqB9Xo45WL7iXXx0mR4x6ao9taKeqDZwv1ng5p6pHsiUiEdqBn -mWLgUV4RPTX95mLP6x4pul/+f1l5lrMe/iuDSn/WhT48aEz0TZ/juAGPDLrI/Y/L -Cs4ONXjnAgMBAAECggEAHbTPo4p4uqwluFlts4QkCgNuS+Sa94mB0bXF542pelT+ -YLW+xVF/dP692PFf1dJ2KDXiIfvItT2snjosZCMofM/1wYE9RvzRRqH4gdsOy0NC -zjv70sUlpq1gsyYMC3Sc6ohjgdtbkF1ib2TZBdhH71nYpqVhoGAxWhwwDFBFat7W -yWq6xGU5EpOnnCzYnlkctGUEKqmY0NH2B2tDdtHac8675POlXk+sG7SlI4nlMl3x -MjpqUPkag9KbxvlMTszMuI4uheVgI/nzGvwFnNJHuMJqBlCCzX3WjsnHDD9loTZz -kxg/VieqngLDOtqBG6y92u8FUla9jTEZ6AiSLJnzoQKBgQD1GTleUV2di4d4afCP -ZCnqZNupw/cq4bvEsjXPGq7BZvVdi1i0djC1qGxFGoZyKNOVqAjFfLlzicmCGBYf -88txzlJ33Ff27NzA2qs8J3wXj2NB64/2ck+6IYY1Kq5donxrztUmlseldCzWh75B -TB4jpBjf/KOxi6G1ZNrvs+MsKQKBgQDLtki1Ox5SHBp2iawbikBnvZMD2fgXKXpz -hP5jTIoz/oAPK0vIR39AYHQChnWvksNliNFbpVeeuWuMb+ejHkL0SG0XV7+AB0Mc -NJNzfCgzgJ3r2J63gkO7HEAfY3mIkhzFjiA8uDfyS27+Qi/D/zt/8yxvWPYXwdSE -C4qMTmsejwKBgFap3ZwY/xYG0Rbltras/xkvInCrpwjVmnQZY2dgkuw1wIqcC/h7 -VgvVirsAfwt8nhvrt3ZE/Ln6vXKvm5setBAAmS5ijgP0irXyW9vARmN6Qul7JoNI -tiIlWnQQLGlVW9Mu5jVn2R4KrxQIGWCzTnDhxB3nEeZkbgBasgNRwwDJAoGBAJn2 -fyVaNSRh0S5eBxIZf2vov8BdcUAwPTgqAAoUwy9g3nCYjIImZ2rEe2mdLj0wczW+ -Hl1hrcNs6HgP68ZBDOGMC8dkrnuZAYPDyN2zrL6gKrsh0XNopMA6LcyLZkNuCwjU -zlCV+tLJaG4KoAdGVDX0GFVfqz9xY6kY//HjItrJAoGBANlMVS2Kpymnbm6a9P4/ -jyT+dpPKh0aph5RH5kFSHEXoL49ArgYcTqKW3IeWS4drxENcARxpKbogxGhkTlBX -HSSOjXR7iu34lRSuYAec/cj1Do+DsskjjBdcRLl1PTgaHVzsaeY9csxR6tmxw54X -nIw3dI+7A4h4ylx7jP0NYPqW +MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQD96qjq3jFAczcR +hQrqhm9TofB9KB0b5MabVYsTsO/nJIN+i8Z/JdLaRwQrgCXg0MPJSgoseUg1QSvx +bCgYQgpZ/LjJb/XTPl/JHs9qzxXNB6ei+lucxM1v9Amz7OGcHcb1srBnAt3VZcYV +8P1ALFAuGibQ++FUBl1CNN8u6M4f1Jo1A07eZt91CJSGRIrDyGGaQPukTIDSyV8f +BDoa3QiSN6A+OfpgTKmVL0RlkQLCFmEa9YwHUpiBBi7y8f+3bQ5ouy/WOGS3M+1m +pfNzlnFKdbl9cvpOck7LN0lszdhi6kGSBy9V5oHzsK3anudKmkN4otqnVmx+pXUu +mqdphq67AgMBAAECggEATc7DHVmiCQhlRxKYQj6Yza1xhsCsjtgffgkMGageE5vu +SXBmjp7WCno5jmTx9n6yiDOOg4tUs7D5WL0WWjyedG3LaDrNPwK9kmFFGQtFOHNQ +jNIgEZ2DAHvtHzwG9HJxfefYQ3Cu7o4F0cJrsGcD2OS9oUuWBEwA9uFBxNulEj5V +E4h6KMD8TMbvTIuRiJYoPl0vIARqeD2nY+/C3gw1n2376PZuC0omeIiMJY0EegfC +5MAoFx5DGwjwnMldFeY0iN+qY6wnrZK/5tV4K2TrW34JJa/QrA6SF+Pqt+mVaMD4 +c0hOvecj3EJCSEPv4msYSNBzY/ZAbwaj0PR2jxmcbQKBgQD/P5GFp5zwUEGBcfzR +2WVtQt3u8Y1hMK6N7uXOx+4l5VPMr7wExVCE8G4jOmwrFJDo3firxroVKItk5MMA +8/ka1LdW3X6Scr5iVtXeUXstBcoi8Qt0fsGkz2COlLXY/W8+ihlJPIeqMaYUoiB1 +81nQORuUvqhj59gUYm0njyZ0hQKBgQD+qhZicqE0mdpxeV5uS+1mYscUJySJKsZM +0vIui/UuNVmE2tAXRmC6zKdR857VvVZ/FVyftWo7EGEybAfF4zWcb5LwuNq3zQXt +WYN++69GEriLRuMHIQwhlJwt+X6XVB0a+urR7HM4+jlf6HGXaA2ZINYEhAmCpf8J +pg2SxpiaPwKBgEQLb0DhKQ5LZtsaRxquSMKy47UyQc1aC/6cZDkWxV7m3ssfQhFH +hKqb6dCMX4+wgN0DZ6prZOoFD/wKnA2h/JNxh5qpm3dxDV3r5kHJGPwsofFkrvgU +Xo0QF56K+FtrXH+gkxMaBtSRPcQcYGjxQc0nnDmwBfX0NX9hqdW07Lx9AoGAN4Y8 +JTDbBw34e787oI67bxRgVXuHUsTZwYxIs29egLmvD/FpZ3m3w2K1pH+ahP2oK0Ms +E8JJLCGRH55AP5wfZ0FIZ2XWgjaYcTyQGBKmD4ArbmqBO1+wNm4hc0Cvoiz7v5Mv +uZ91K9oawld61MkiFd3767YiILMynRbwZK0aPp8CgYBOVPxXPIPPTcE+Y66ARhQb +up0YsDn8YlvGIrSzF+ujGxXLEyJBSzr8eytmASpT3C+6xRkmdLMFzJHxeutl1VaI +IK02GbFZz0NWPBc7o4uwpVkvKWayvsoojfzAV6UwETRxCYTyMx2OaejeZRx3Z68z +QtFezp67ih0dS7gBa7eh7A== -----END PRIVATE KEY----- -----BEGIN CERTIFICATE----- MIIFszCCA5ugAwIBAgICEAEwDQYJKoZIhvcNAQELBQAwVDEYMBYGA1UECgwPRElS diff --git a/src/DIRAC/Interfaces/API/Dirac.py b/src/DIRAC/Interfaces/API/Dirac.py index a0b2d061665..3d5a37d4444 100755 --- a/src/DIRAC/Interfaces/API/Dirac.py +++ b/src/DIRAC/Interfaces/API/Dirac.py @@ -1652,14 +1652,15 @@ def deleteJob(self, jobID): return ret jobIDs = ret["Value"] - jobIDsToDelete = [] - for jobID in jobIDs: - can_kill = JobStatus.checkJobStateTransition(jobID, JobStatus.KILLED)["OK"] - can_del = JobStatus.checkJobStateTransition(jobID, JobStatus.DELETED)["OK"] - if can_kill or can_del: - jobIDsToDelete.append(jobID) - - result = WMSClient(useCertificates=self.useCertificates).deleteJob(jobIDsToDelete) + # Remove any job IDs that can't change to the Killed or Deleted states + filteredJobs = set() + for filterState in (JobStatus.KILLED, JobStatus.DELETED): + filterRes = JobStatus.filterJobStateTransition(jobIDs, filterState) + if not filterRes["OK"]: + return filterRes + filteredJobs.update(filterRes["Value"]) + + result = WMSClient(useCertificates=self.useCertificates).deleteJob(list(filteredJobs)) if result["OK"]: if self.jobRepo: for jID in result["Value"]: @@ -1689,11 +1690,11 @@ def rescheduleJob(self, jobID): return ret jobIDs = ret["Value"] - jobIDsToReschedule = [] - for jobID in jobIDs: - res = JobStatus.checkJobStateTransition(jobID, JobStatus.RESCHEDULED) - if res["OK"]: - jobIDsToReschedule.append(jobID) + # Remove any job IDs that can't change to the rescheduled state + filterRes = JobStatus.filterJobStateTransition(jobIDs, JobStatus.RESCHEDULED) + if not filterRes["OK"]: + return filterRes + jobIDsToReschedule = filterRes["Value"] result = WMSClient(useCertificates=self.useCertificates).rescheduleJob(jobIDsToReschedule) if result["OK"]: @@ -1724,14 +1725,15 @@ def killJob(self, jobID): return ret jobIDs = ret["Value"] - jobIDsToKill = [] - for jobID in jobIDs: - can_kill = JobStatus.checkJobStateTransition(jobID, JobStatus.KILLED)["OK"] - can_del = JobStatus.checkJobStateTransition(jobID, JobStatus.DELETED)["OK"] - if can_kill or can_del: - jobIDsToKill.append(jobID) + # Remove any job IDs that can't change to the Killed or Deleted states + filteredJobs = set() + for filterState in (JobStatus.KILLED, JobStatus.DELETED): + filterRes = JobStatus.filterJobStateTransition(jobIDs, filterState) + if not filterRes["OK"]: + return filterRes + filteredJobs.update(filterRes["Value"]) - result = WMSClient(useCertificates=self.useCertificates).killJob(jobIDsToKill) + result = WMSClient(useCertificates=self.useCertificates).killJob(list(filteredJobs)) if result["OK"]: if self.jobRepo: for jID in result["Value"]: diff --git a/src/DIRAC/WorkloadManagementSystem/Client/JobStatus.py b/src/DIRAC/WorkloadManagementSystem/Client/JobStatus.py index f8c912cda2a..2fac939997b 100644 --- a/src/DIRAC/WorkloadManagementSystem/Client/JobStatus.py +++ b/src/DIRAC/WorkloadManagementSystem/Client/JobStatus.py @@ -4,6 +4,7 @@ from DIRAC import gLogger, S_OK, S_ERROR from DIRAC.Core.Utilities.StateMachine import State, StateMachine +from DIRAC.Core.Utilities.Decorators import deprecated #: @@ -97,6 +98,7 @@ def __init__(self, state): } +@deprecated("Use filterJobStateTransition instead") def checkJobStateTransition(jobID, candidateState, currentStatus=None, jobMonitoringClient=None): """Utility to check if a job state transition is allowed""" if not currentStatus: @@ -125,3 +127,31 @@ def checkJobStateTransition(jobID, candidateState, currentStatus=None, jobMonito ) return S_ERROR("Job state transition not allowed") return S_OK() + + +def filterJobStateTransition(jobIDs, candidateState, jobMonitoringClient=None): + """Given a list of jobIDs, return a list that are allowed to transition + to the given candidate state. + """ + allowedJobs = [] + + if not isinstance(jobIDs, list): + jobIDs = [jobIDs] + + if not jobMonitoringClient: + from DIRAC.WorkloadManagementSystem.Client.JobMonitoringClient import JobMonitoringClient + + jobMonitoringClient = JobMonitoringClient() + + res = jobMonitoringClient.getJobsStatus(jobIDs) + if not res["OK"]: + return res + + for jobID in jobIDs: + if jobID in res["Value"]: + curState = res["Value"][jobID]["Status"] + stateRes = JobsStateMachine(curState).getNextState(candidateState) + if stateRes["OK"]: + if stateRes["Value"] == candidateState: + allowedJobs.append(jobID) + return S_OK(allowedJobs)