From acb1cd2314b7b8100aab30aba9fe33c04dd5e373 Mon Sep 17 00:00:00 2001 From: kaloyane <kaloyane> Date: Fri, 1 Jul 2011 09:08:02 +0000 Subject: [PATCH] [LMS-2332] further improvements 1) allow empty wells in the library template (no well sample will be created in openBIS) 2) handle Sanofi wellCode format (e.g. AC007 is translated to AC7) 3) bugfix: well rows start counting from 1 (not from 0) SVN: 21946 --- .../dist/etc/sanofi-dropbox/sanofi-dropbox.py | 55 ++++++++++++++----- .../dss/test/SanofiDropboxJythonTest.java | 26 ++++----- 2 files changed, 52 insertions(+), 29 deletions(-) diff --git a/sanofi/dist/etc/sanofi-dropbox/sanofi-dropbox.py b/sanofi/dist/etc/sanofi-dropbox/sanofi-dropbox.py index 037baf878d6..cd3f29d7429 100644 --- a/sanofi/dist/etc/sanofi-dropbox/sanofi-dropbox.py +++ b/sanofi/dist/etc/sanofi-dropbox/sanofi-dropbox.py @@ -1,3 +1,5 @@ +import re + from ch.systemsx.cisd.common.mail import From from ch.systemsx.cisd.openbis.generic.shared.api.v1.dto import SearchCriteria @@ -98,14 +100,25 @@ class SanofiMaterial: A data structure class holding compound materials as they exist in the Abase (Sanofi) database. """ def __init__(self, wellCode, materialCode, sanofiId, sanofiBatchId): - self.wellCode = wellCode + self.wellCode = self.normalizeWellCode(wellCode) self.materialCode = materialCode self.sanofiId = sanofiId self.sanofiBatchId = sanofiBatchId + + def normalizeWellCode(self, wellCode): + """ normalizes Sanofi wellCodes openBIS wellCodes e.g. AB007 to AB7 """ + return re.sub("(?<=\w)(0+)(?=\d)", "", wellCode) class PlateInitializer: ABASE_DATA_SOURCE = "abase-datasource" - ABASE_QUERY = "TODO: this a query provided by Matt" + ABASE_QUERY = """select + ptodwellreference WELL_CODE, + translate(objdbatchref,'{/:()+','{_____') MATERIAL_CODE, + objdbatchref ABASE_COMPOUND_BATCH_ID, + objdid ABASE_COMPOUND_ID, + olptid ABASE_PLATE_CODE + from sysadmin.plteobjd + where olptid = ?{1}""" LIBRARY_TEMPLATE_PROPNAME = "LIBRARY_TEMPLATE" @@ -113,10 +126,10 @@ class PlateInitializer: NEGATIVE_CONTROL_TYPE = "NEGATIVE_CONTROL" COMPOUND_WELL_TYPE = "COMPOUND_WELL" - COMPOUND_WELL_CONCENTRATION_PROPNAME = "CONCENTRATION" - COMPOUND_WELL_MATERIAL_PROPNAME = "COMPOUND_BATCH" + COMPOUND_WELL_CONCENTRATION_PROPNAME = "CONCENTRATION_M" + COMPOUND_WELL_MATERIAL_PROPNAME = "COMPOUND" - MATERIAL_TYPE = "COMPOUND_BATCH" + MATERIAL_TYPE = "COMPOUND" MATERIAL_ID_PROPNAME = "COMPOUND_ID" MATERIAL_BATCH_ID_PROPNAME = "COMPOUND_BATCH_ID" @@ -125,9 +138,16 @@ class PlateInitializer: self.plate = plate self.plateCode = plate.getCode() self.experimentId = plate.getExperiment().getExperimentIdentifier() + + def wellColumn(self, x): + numLetters = 26 + if x < numLetters: + return chr(ord('A') + x) + else: + return self.wellColumn((x / numLetters) - 1) + self.wellColumn( x % numLetters) def getWellCode(self, x, y): - return chr(ord('A') + x) + str(y) + return self.wellColumn(x) + str(y + 1) def getPlateDimensions(self): """ @@ -146,14 +166,15 @@ class PlateInitializer: if plateHeight != len(tsvLines) : raise RuntimeError("The geometry property of plate %(plateCode)s (height=%(plateHeight)s)" " does not agree with the value of the %(LIBRARY_TEMPLATE_PROPNAME)s" - " property in experiment %(experimentId)s (height=%(numLines)s)." % vars()) + " property in experiment %(experimentId)s (height=%(numLines)s)." % vars(self)) for i in range(0, len(tsvLines)): lineWidth = len(tsvLines[i]) if plateWidth != lineWidth: - raise RuntimeError("The geometry property of plate %(plateCode)s (width=%(plateWidth)s)" - " does not agree with the value of the %(LIBRARY_TEMPLATE_PROPNAME)s" - " property in experiment %(experimentId)s (line=%(i)s, width=%(lineWidth)s)." % vars()) + raise RuntimeError("The geometry property of plate %s (width=%s)" + " does not agree with the value of the %s" + " property in experiment %s (line=%s, width=%s)." % \ + (plateCode, plateWidth, self.LIBRARY_TEMPLATE_PROPNAME, self.experimentId, i, lineWidth)) def parseLibraryTemplate(self): template = experiment.getPropertyValue(self.LIBRARY_TEMPLATE_PROPNAME) @@ -179,7 +200,7 @@ class PlateInitializer: In case the plate is not found in Abase return None. """ queryService = state.getDataSourceQueryService() - queryResult = queryService.select(self.ABASE_DATA_SOURCE, self.ABASE_QUERY, [plate.code]) + queryResult = queryService.select(self.ABASE_DATA_SOURCE, self.ABASE_QUERY, [self.plateCode]) sanofiMaterials = [] for materialMap in list(queryResult): @@ -240,7 +261,10 @@ class PlateInitializer: controlWellTypes = { "H" : self.POSITIVE_CONTROL_TYPE, \ "L" : self.NEGATIVE_CONTROL_TYPE}; - for wellCode in library: + for wellCode in library: + if not library[wellCode]: + continue + libraryValue = library[wellCode].upper() prefixedWellCode = self.plateCode + ":" + wellCode @@ -255,9 +279,10 @@ class PlateInitializer: try: float(concentration) except ValueError: - raise RuntimeError("The specified value for well %(wellCode)s in the property " - " %(LIBRARY_TEMPLATE_PROPNAME)s of experiment %(experimentId)s is invalid. " - "Allowed values are 'H', 'L' or number, but %(libraryValue)s' was found." % vars()) + raise RuntimeError("The specified value for well %s in the property " + " %s of experiment %s is invalid. Allowed values are 'H', 'L'" + " or number, but '%s' was found." % \ + (wellCode, self.LIBRARY_TEMPLATE_PROPNAME, self.experimentId, libraryValue)) well = self.transaction.createNewSample(prefixedWellCode, self.COMPOUND_WELL_TYPE) well.setContainer(self.plate) diff --git a/sanofi/sourceTest/java/ch/systemsx/cisd/sanofi/dss/test/SanofiDropboxJythonTest.java b/sanofi/sourceTest/java/ch/systemsx/cisd/sanofi/dss/test/SanofiDropboxJythonTest.java index bc56399d187..de1c95805f0 100644 --- a/sanofi/sourceTest/java/ch/systemsx/cisd/sanofi/dss/test/SanofiDropboxJythonTest.java +++ b/sanofi/sourceTest/java/ch/systemsx/cisd/sanofi/dss/test/SanofiDropboxJythonTest.java @@ -80,7 +80,7 @@ public class SanofiDropboxJythonTest extends AbstractJythonDataSetHandlerTest private static final String[] EXPERIMENT_RECIPIENTS = new String[] { "admin@sanofi.com", "mickey@mouse.org" }; - private static final String MATERIAL_TYPE = "COMPOUND_BATCH"; + private static final String MATERIAL_TYPE = "COMPOUND"; private static final String POSITIVE_CONTROL_TYPE = "POSITIVE_CONTROL"; @@ -88,9 +88,9 @@ public class SanofiDropboxJythonTest extends AbstractJythonDataSetHandlerTest private static final String COMPOUND_WELL_TYPE = "COMPOUND_WELL"; - private static final String COMPOUND_WELL_CONCENTRATION_PROPNAME = "CONCENTRATION"; + private static final String COMPOUND_WELL_CONCENTRATION_PROPNAME = "CONCENTRATION_M"; - private static final String COMPOUND_WELL_MATERIAL_PROPNAME = "COMPOUND_BATCH"; + private static final String COMPOUND_WELL_MATERIAL_PROPNAME = "COMPOUND"; private static final String DATASET_DIR_NAME = "batchNr_plateCode.variant_2011.06.28"; @@ -116,16 +116,15 @@ public class SanofiDropboxJythonTest extends AbstractJythonDataSetHandlerTest createHandler(properties, false, true); createData(); - final String libraryTemplate = "1.45\t20.701\tH\n0.12\t0.002\tL"; + final String libraryTemplate = "1.45\t\tH\n0.12\t0.002\tL"; final Sample plate = createPlate(libraryTemplate, "6_WELLS_2X3"); setUpPlateSearchExpectations(plate); setUpLibraryTemplateExpectations(plate); final MockDataSet<Map<String, Object>> queryResult = new MockDataSet<Map<String, Object>>(); - queryResult.add(createQueryResult("A0")); queryResult.add(createQueryResult("A1")); - queryResult.add(createQueryResult("B0")); queryResult.add(createQueryResult("B1")); + queryResult.add(createQueryResult("B2")); final RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails> atomicatOperationDetails = new RecordingMatcher<ch.systemsx.cisd.openbis.generic.shared.dto.AtomicEntityOperationDetails>(); @@ -143,7 +142,7 @@ public class SanofiDropboxJythonTest extends AbstractJythonDataSetHandlerTest one(openBisService).listMaterials(with(materialCriteria), with(equal(true))); will(returnValue(Collections.emptyList())); - exactly(6).of(openBisService).createPermId(); + exactly(5).of(openBisService).createPermId(); will(returnValue("well-permId")); one(openBisService).createDataSetCode(); @@ -178,14 +177,13 @@ public class SanofiDropboxJythonTest extends AbstractJythonDataSetHandlerTest List<NewSample> registeredSamples = atomicatOperationDetails.recordedObject().getSampleRegistrations(); - assertEquals(6, registeredSamples.size()); + assertEquals(5, registeredSamples.size()); assertAllSamplesHaveContainer(registeredSamples, plate.getIdentifier()); - assertCompoundWell(registeredSamples, "A0", "1.45"); - assertCompoundWell(registeredSamples, "A1", "20.701"); - assertPositiveControl(registeredSamples, "A2"); - assertCompoundWell(registeredSamples, "B0", "0.12"); - assertCompoundWell(registeredSamples, "B1", "0.002"); - assertNegativeControl(registeredSamples, "B2"); + assertCompoundWell(registeredSamples, "A1", "1.45"); + assertPositiveControl(registeredSamples, "A3"); + assertCompoundWell(registeredSamples, "B1", "0.12"); + assertCompoundWell(registeredSamples, "B2", "0.002"); + assertNegativeControl(registeredSamples, "B3"); List<? extends NewExternalData> dataSetsRegistered = atomicatOperationDetails.recordedObject().getDataSetRegistrations(); -- GitLab