Skip resource updating completely if found in package load. (#4036)

* add ability to skip loading package resources that already exists, to avoid reindex when loading IGs

* Add a test for validating no updates or creates are called on anything but binaries

* Add changelog

Co-authored-by: Craig McClendon <craig.mcclendon@accenture.com>
This commit is contained in:
Tadgh 2022-09-20 08:45:37 -07:00 committed by GitHub
parent 97b82596f6
commit d1f988f98a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 101 additions and 15 deletions

View File

@ -0,0 +1,5 @@
---
type: add
issue: 3989
title: "Provided the ability to have the NPM package installer skip installing a package if it is already installed and matches the version requested. This can be controlled by
the `reloadExisting` attribute in PackageInstallationSpec. It defaults to `true`, which is the existing behaviour. Thanks to Craig McClendon (@XcrigX) for the contribution!"

View File

@ -43,6 +43,8 @@ public class PackageInstallOutcomeJson {
@JsonProperty("resourcesInstalled") @JsonProperty("resourcesInstalled")
private Map<String, Integer> myResourcesInstalled; private Map<String, Integer> myResourcesInstalled;
public List<String> getMessage() { public List<String> getMessage() {
if (myMessage == null) { if (myMessage == null) {
myMessage = new ArrayList<>(); myMessage = new ArrayList<>();

View File

@ -34,12 +34,10 @@ import java.util.List;
import java.util.function.Supplier; import java.util.function.Supplier;
@Schema( @Schema(
name = "PackageInstallationSpec", name = "PackageInstallationSpec", description = "Defines a set of instructions for package installation"
description = )
"Defines a set of instructions for package installation"
)
@JsonPropertyOrder({ @JsonPropertyOrder({
"name", "version", "packageUrl", "installMode", "installResourceTypes", "validationMode" "name", "version", "packageUrl", "installMode", "installResourceTypes", "validationMode", "reloadExisting"
}) })
@ExampleSupplier({PackageInstallationSpec.ExampleSupplier.class, PackageInstallationSpec.ExampleSupplier2.class}) @ExampleSupplier({PackageInstallationSpec.ExampleSupplier.class, PackageInstallationSpec.ExampleSupplier2.class})
@JsonInclude(JsonInclude.Include.NON_NULL) @JsonInclude(JsonInclude.Include.NON_NULL)
@ -64,6 +62,9 @@ public class PackageInstallationSpec {
@Schema(description = "Should dependencies be automatically resolved, fetched and installed with the same settings") @Schema(description = "Should dependencies be automatically resolved, fetched and installed with the same settings")
@JsonProperty("fetchDependencies") @JsonProperty("fetchDependencies")
private boolean myFetchDependencies; private boolean myFetchDependencies;
@Schema(description = "Should existing resources be reloaded. Defaults to true, but can be set to false to avoid re-index operations for existing search parameters")
@JsonProperty("reloadExisting")
private boolean myReloadExisting = true;
@Schema(description = "Any values provided here will be interpreted as a regex. Dependencies with an ID matching any regex will be skipped.") @Schema(description = "Any values provided here will be interpreted as a regex. Dependencies with an ID matching any regex will be skipped.")
private List<String> myDependencyExcludes; private List<String> myDependencyExcludes;
@JsonIgnore @JsonIgnore
@ -145,6 +146,14 @@ public class PackageInstallationSpec {
return this; return this;
} }
public boolean isReloadExisting() {
return myReloadExisting;
}
public void setReloadExisting(boolean reloadExisting) {
this.myReloadExisting = reloadExisting;
}
public PackageInstallationSpec addDependencyExclude(String theExclude) { public PackageInstallationSpec addDependencyExclude(String theExclude) {
getDependencyExcludes().add(theExclude); getDependencyExcludes().add(theExclude);
return this; return this;

View File

@ -92,7 +92,7 @@ public class PackageInstallerSvcImpl implements IPackageInstallerSvc {
"ConceptMap", "ConceptMap",
"SearchParameter", "SearchParameter",
"Subscription" "Subscription"
)); ));
boolean enabled = true; boolean enabled = true;
@Autowired @Autowired
@ -163,7 +163,7 @@ public class PackageInstallerSvcImpl implements IPackageInstallerSvc {
return existing.isPresent(); return existing.isPresent();
}); });
if (exists) { if (exists) {
ourLog.info("Package {}#{} is already installed", theInstallationSpec.getName(), theInstallationSpec.getVersion()); ourLog.info("Package {}#{} is already installed", theInstallationSpec.getName(), theInstallationSpec.getVersion());
} }
NpmPackage npmPackage = myPackageCacheManager.installPackage(theInstallationSpec); NpmPackage npmPackage = myPackageCacheManager.installPackage(theInstallationSpec);
@ -225,7 +225,7 @@ public class PackageInstallerSvcImpl implements IPackageInstallerSvc {
try { try {
next = isStructureDefinitionWithoutSnapshot(next) ? generateSnapshot(next) : next; next = isStructureDefinitionWithoutSnapshot(next) ? generateSnapshot(next) : next;
create(next, theOutcome); create(next, theInstallationSpec, theOutcome);
} catch (Exception e) { } catch (Exception e) {
ourLog.warn("Failed to upload resource of type {} with ID {} - Error: {}", myFhirContext.getResourceType(next), next.getIdElement().getValue(), e.toString()); ourLog.warn("Failed to upload resource of type {} with ID {} - Error: {}", myFhirContext.getResourceType(next), next.getIdElement().getValue(), e.toString());
throw new ImplementationGuideInstallationException(Msg.code(1286) + String.format("Error installing IG %s#%s: %s", name, version, e), e); throw new ImplementationGuideInstallationException(Msg.code(1286) + String.format("Error installing IG %s#%s: %s", name, version, e), e);
@ -323,7 +323,7 @@ public class PackageInstallerSvcImpl implements IPackageInstallerSvc {
return resources; return resources;
} }
private void create(IBaseResource theResource, PackageInstallOutcomeJson theOutcome) { private void create(IBaseResource theResource, PackageInstallationSpec theInstallationSpec, PackageInstallOutcomeJson theOutcome) {
IFhirResourceDao dao = myDaoRegistry.getResourceDao(theResource.getClass()); IFhirResourceDao dao = myDaoRegistry.getResourceDao(theResource.getClass());
SearchParameterMap map = createSearchParameterMapFor(theResource); SearchParameterMap map = createSearchParameterMapFor(theResource);
IBundleProvider searchResult = searchResource(dao, map); IBundleProvider searchResult = searchResource(dao, map);
@ -347,11 +347,15 @@ public class PackageInstallerSvcImpl implements IPackageInstallerSvc {
ourLog.info("Created resource with existing id"); ourLog.info("Created resource with existing id");
} }
} else { } else {
ourLog.info("Updating existing resource matching {}", map.toNormalizedQueryString(myFhirContext)); if (theInstallationSpec.isReloadExisting()) {
theResource.setId(searchResult.getResources(0, 1).get(0).getIdElement().toUnqualifiedVersionless()); ourLog.info("Updating existing resource matching {}", map.toNormalizedQueryString(myFhirContext));
DaoMethodOutcome outcome = updateResource(dao, theResource); theResource.setId(searchResult.getResources(0, 1).get(0).getIdElement().toUnqualifiedVersionless());
if (!outcome.isNop()) { DaoMethodOutcome outcome = updateResource(dao, theResource);
theOutcome.incrementResourcesInstalled(myFhirContext.getResourceType(theResource)); if (!outcome.isNop()) {
theOutcome.incrementResourcesInstalled(myFhirContext.getResourceType(theResource));
}
} else {
ourLog.info("Skipping update of existing resource matching {}", map.toNormalizedQueryString(myFhirContext));
} }
} }
} }

View File

@ -360,5 +360,4 @@ public class SearchParamRegistryImpl implements ISearchParamRegistry, IResourceC
public void setSearchParameterCanonicalizerForUnitTest(SearchParameterCanonicalizer theSearchParameterCanonicalizerForUnitTest) { public void setSearchParameterCanonicalizerForUnitTest(SearchParameterCanonicalizer theSearchParameterCanonicalizerForUnitTest) {
mySearchParameterCanonicalizer = theSearchParameterCanonicalizerForUnitTest; mySearchParameterCanonicalizer = theSearchParameterCanonicalizerForUnitTest;
} }
} }

View File

@ -1,8 +1,12 @@
package ca.uhn.fhir.jpa.packages; package ca.uhn.fhir.jpa.packages;
import ca.uhn.fhir.i18n.Msg; import ca.uhn.fhir.i18n.Msg;
import ca.uhn.fhir.interceptor.api.IAnonymousInterceptor;
import ca.uhn.fhir.interceptor.api.Pointcut;
import ca.uhn.fhir.jpa.api.config.DaoConfig; import ca.uhn.fhir.jpa.api.config.DaoConfig;
import ca.uhn.fhir.jpa.dao.data.INpmPackageVersionDao; import ca.uhn.fhir.jpa.dao.data.INpmPackageVersionDao;
import ca.uhn.fhir.jpa.model.entity.IBaseResourceEntity;
import ca.uhn.fhir.jpa.model.entity.NpmPackageVersionEntity;
import ca.uhn.fhir.jpa.test.BaseJpaDstu3Test; import ca.uhn.fhir.jpa.test.BaseJpaDstu3Test;
import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
import ca.uhn.fhir.rest.server.exceptions.ResourceNotFoundException; import ca.uhn.fhir.rest.server.exceptions.ResourceNotFoundException;
@ -11,6 +15,9 @@ import ca.uhn.fhir.test.utilities.ProxyUtil;
import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.servlet.ServletHandler; import org.eclipse.jetty.servlet.ServletHandler;
import org.eclipse.jetty.servlet.ServletHolder; import org.eclipse.jetty.servlet.ServletHolder;
import org.hl7.fhir.instance.model.api.IBaseBinary;
import org.hl7.fhir.instance.model.api.IBaseResource;
import org.hl7.fhir.r4.model.Patient;
import org.hl7.fhir.utilities.npm.IPackageCacheManager; import org.hl7.fhir.utilities.npm.IPackageCacheManager;
import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.BeforeEach;
@ -19,11 +26,17 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import java.util.Date;
import java.util.Optional;
import java.util.concurrent.Callable;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import static ca.uhn.fhir.util.ClasspathUtil.loadResourceAsByteArray; import static ca.uhn.fhir.util.ClasspathUtil.loadResourceAsByteArray;
import static com.healthmarketscience.sqlbuilder.Conditions.not;
import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.hasItem; import static org.hamcrest.Matchers.hasItem;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertFalse;
@ -168,6 +181,54 @@ public class IgInstallerDstu3Test extends BaseJpaDstu3Test {
runInTransaction(() -> { runInTransaction(() -> {
assertTrue(myPackageVersionDao.findByPackageIdAndVersion("nictiz.fhir.nl.stu3.questionnaires", "1.0.2").isPresent()); assertTrue(myPackageVersionDao.findByPackageIdAndVersion("nictiz.fhir.nl.stu3.questionnaires", "1.0.2").isPresent());
}); });
}
private void ensureNoCreatesOrUpdates(Callable theCallable) throws Exception {
myInterceptorRegistry.registerAnonymousInterceptor(Pointcut.STORAGE_PRESTORAGE_RESOURCE_CREATED, (thePointcut, t) -> {
IBaseResource iBaseResource = t.get(IBaseResource.class);
if (iBaseResource instanceof IBaseBinary) {
return;
}
throw new RuntimeException("Not allowed!");
});
myInterceptorRegistry.registerAnonymousInterceptor(Pointcut.STORAGE_PRESTORAGE_RESOURCE_UPDATED, (thePointcut, t) -> {
IBaseResource iBaseResource = t.get(IBaseResource.class);
if (iBaseResource instanceof IBaseBinary) {
return;
}
throw new RuntimeException("Not allowed!");
});
try {
theCallable.call();
} finally {
myInterceptorRegistry.unregisterAllAnonymousInterceptors();
}
}
@Test
public void testMultipleUploads() throws Exception {
myDaoConfig.setAllowExternalReferences(true);
PackageInstallationSpec installationSpec = new PackageInstallationSpec()
.setName("nictiz.fhir.nl.stu3.questionnaires")
.setVersion("1.0.2")
.setInstallMode(PackageInstallationSpec.InstallModeEnum.STORE_AND_INSTALL)
.setPackageUrl("classpath:/packages/nictiz.fhir.nl.stu3.questionnaires-1.0.2.tgz");
igInstaller.install(installationSpec);
installationSpec.setReloadExisting(true);
try {
ensureNoCreatesOrUpdates(() -> igInstaller.install(installationSpec));
fail();
} catch (RuntimeException e) {
assertThat(e.getMessage(), is(containsString("Not allowed!")));
}
installationSpec.setReloadExisting(false);
ensureNoCreatesOrUpdates(() -> igInstaller.install(installationSpec));
} }

View File

@ -951,4 +951,6 @@ public class NpmR4Test extends BaseJpaR4Test {
}); });
} }
} }

View File

@ -757,6 +757,10 @@
<name>Gjergj Sheldija</name> <name>Gjergj Sheldija</name>
</developer> </developer>
<developer> <developer>
<id>XcrigX</id>
<name>Craig McClendon</name>
</developer>
<developer>
<id>dyoung-work</id> <id>dyoung-work</id>
</developer> </developer>
</developers> </developers>