diff --git a/change/@itwin-imodel-transformer-d56dd0fb-4a09-43f6-9ce5-9bc6a92c7880.json b/change/@itwin-imodel-transformer-d56dd0fb-4a09-43f6-9ce5-9bc6a92c7880.json new file mode 100644 index 00000000..e7a2dfb8 --- /dev/null +++ b/change/@itwin-imodel-transformer-d56dd0fb-4a09-43f6-9ce5-9bc6a92c7880.json @@ -0,0 +1,7 @@ +{ + "type": "patch", + "comment": "add primitive geometry cleanup routine", + "packageName": "@itwin/imodel-transformer", + "email": "mike.belousov@bentley.com", + "dependentChangeType": "patch" +} diff --git a/packages/test-app/src/Main.ts b/packages/test-app/src/Main.ts index 20c78679..9bb8337d 100644 --- a/packages/test-app/src/Main.ts +++ b/packages/test-app/src/Main.ts @@ -133,6 +133,7 @@ void (async () => { type: "boolean", default: false, }, + combinePhysicalModels: { desc: "Combine all source PhysicalModels into a single PhysicalModel in the target iModel", type: "boolean", @@ -188,6 +189,11 @@ void (async () => { default: "reject" as const, choices: ["reject", "ignore"] as const, }, + cleanupUnusedGeometryParts: { + desc: "cleans up unused geometry parts after the transformation is finished", + type: "boolean", + default: false, + }, }) .parseSync(); diff --git a/packages/transformer/package.json b/packages/transformer/package.json index 392d8667..bf9d8c92 100644 --- a/packages/transformer/package.json +++ b/packages/transformer/package.json @@ -1,6 +1,6 @@ { "name": "@itwin/imodel-transformer", - "version": "0.3.2", + "version": "0.3.3-prune-geom-parts.1", "description": "API for exporting an iModel's parts and also importing them into another iModel", "main": "lib/cjs/transformer.js", "typings": "lib/cjs/transformer", diff --git a/packages/transformer/src/CleanupUnusedGeometryParts.ts b/packages/transformer/src/CleanupUnusedGeometryParts.ts new file mode 100644 index 00000000..72fb1464 --- /dev/null +++ b/packages/transformer/src/CleanupUnusedGeometryParts.ts @@ -0,0 +1,61 @@ +import { IModelDb } from "@itwin/core-backend"; +import { DbResult, Id64String } from "@itwin/core-bentley"; +import { ElementGeometry } from "@itwin/core-common"; + +/** + * delete all geometry parts that are not referenced by any geometric elements. + * This will be replaced by a more integrated approach + * @internal + */ +export function cleanupUnusedGeometryParts(db: IModelDb) { + const unusedGeomPartIds = queryUnusedGeomParts(db); + + db.elements.deleteDefinitionElements([...unusedGeomPartIds]); +} + +/** + * queryEntityIds maxes out at 10K, we may need everything during this temporary solution + * since geometry parts may reference each other + */ +function queryUnusedGeomParts(db: IModelDb) { + const usedGeomParts = new Set(); + + const allGeomElemIdsQuery = ` + SELECT ECInstanceId + FROM bis.GeometricElement + `; + + db.withPreparedStatement(allGeomElemIdsQuery, (geomElemIdStmt) => { + while (geomElemIdStmt.step() === DbResult.BE_SQLITE_ROW) { + const geomElemId = geomElemIdStmt.getValue(0).getId(); + db.elementGeometryRequest({ + elementId: geomElemId, + skipBReps: true, // breps contain no references to geometry parts + onGeometry(geomInfo) { + for (const entry of new ElementGeometry.Iterator(geomInfo)) { + const maybeGeomPart = entry.toGeometryPart(); + if (maybeGeomPart) + usedGeomParts.add(maybeGeomPart); + } + }, + }); + } + }); + + const unusedGeomPartIds = db.withPreparedStatement(` + SELECT ECInstanceId + FROM bis.GeometryPart + WHERE NOT InVirtualSet(?, ECInstanceId) + `, + (stmt) => { + const ids = new Set(); + stmt.bindIdSet(1, [...usedGeomParts]); + while (stmt.step() === DbResult.BE_SQLITE_ROW) { + const id = stmt.getValue(0).getId(); + ids.add(id); + } + return ids; + }); + + return unusedGeomPartIds; +} diff --git a/packages/transformer/src/IModelTransformer.ts b/packages/transformer/src/IModelTransformer.ts index 5cf2fc77..ff3afe3c 100644 --- a/packages/transformer/src/IModelTransformer.ts +++ b/packages/transformer/src/IModelTransformer.ts @@ -31,6 +31,7 @@ import { IModelImporter, IModelImporterState, OptimizeGeometryOptions } from "./ import { TransformerLoggerCategory } from "./TransformerLoggerCategory"; import { PendingReference, PendingReferenceMap } from "./PendingReferenceMap"; import { EntityKey, EntityMap } from "./EntityMap"; +import { cleanupUnusedGeometryParts } from "./CleanupUnusedGeometryParts"; import { IModelCloneContext } from "./IModelCloneContext"; import { EntityUnifier } from "./EntityUnifier"; @@ -148,6 +149,12 @@ export interface IModelTransformOptions { * @beta */ optimizeGeometry?: OptimizeGeometryOptions; + + /** internal option, will definitely be replaced with a more complete API + * remove unused geometry parts during `finalizeTransformation` + * @internal + */ + cleanupUnusedGeometryParts?: boolean; } /** @@ -1129,6 +1136,16 @@ export class IModelTransformer extends IModelExportHandler { partiallyCommittedElem.forceComplete(); } } + + if (this._options.cleanupUnusedGeometryParts) + // FIXME: move to importer + cleanupUnusedGeometryParts(this.targetDb); + + if (this._options.optimizeGeometry) + this.importer.optimizeGeometry(this._options.optimizeGeometry); + + this.importer.computeProjectExtents(); + // this internal is guaranteed stable for just transformer usage /* eslint-disable @itwin/no-internal */ if ("codeValueBehavior" in this.sourceDb as any) { @@ -1450,10 +1467,6 @@ export class IModelTransformer extends IModelExportHandler { await this.detectRelationshipDeletes(); } - if (this._options.optimizeGeometry) - this.importer.optimizeGeometry(this._options.optimizeGeometry); - - this.importer.computeProjectExtents(); this.finalizeTransformation(); } @@ -1685,10 +1698,6 @@ export class IModelTransformer extends IModelExportHandler { await this.exporter.exportChanges(options); await this.processDeferredElements(); // eslint-disable-line deprecation/deprecation - if (this._options.optimizeGeometry) - this.importer.optimizeGeometry(this._options.optimizeGeometry); - - this.importer.computeProjectExtents(); this.finalizeTransformation(); } } diff --git a/packages/transformer/src/test/standalone/IModelTransformer.test.ts b/packages/transformer/src/test/standalone/IModelTransformer.test.ts index ad424d3a..c580ba1c 100644 --- a/packages/transformer/src/test/standalone/IModelTransformer.test.ts +++ b/packages/transformer/src/test/standalone/IModelTransformer.test.ts @@ -5,15 +5,16 @@ import { assert, expect } from "chai"; import * as fs from "fs"; +import * as child_process from "child_process"; import * as path from "path"; import * as Semver from "semver"; import * as sinon from "sinon"; import { CategorySelector, DisplayStyle3d, DocumentListModel, Drawing, DrawingCategory, DrawingGraphic, DrawingModel, ECSqlStatement, Element, ElementMultiAspect, ElementOwnsChildElements, ElementOwnsExternalSourceAspects, ElementOwnsMultiAspects, ElementOwnsUniqueAspect, ElementRefersToElements, - ElementUniqueAspect, ExternalSourceAspect, GenericPhysicalMaterial, GeometricElement, IModelDb, IModelElementCloneContext, IModelHost, IModelJsFs, + ElementUniqueAspect, ExternalSourceAspect, GenericPhysicalMaterial, GeometricElement, GeometryPart, IModelDb, IModelElementCloneContext, IModelHost, IModelJsFs, InformationRecordModel, InformationRecordPartition, LinkElement, Model, ModelSelector, OrthographicViewDefinition, - PhysicalModel, PhysicalObject, PhysicalPartition, PhysicalType, Relationship, RenderMaterialElement, RepositoryLink, Schema, SnapshotDb, SpatialCategory, StandaloneDb, + PhysicalModel, PhysicalObject, PhysicalPartition, PhysicalType, Relationship, RenderMaterialElement, RepositoryLink, SQLiteDb, Schema, SnapshotDb, SpatialCategory, StandaloneDb, SubCategory, Subject, Texture, } from "@itwin/core-backend"; import * as coreBackendPkgJson from "@itwin/core-backend/package.json"; @@ -22,9 +23,9 @@ import * as TestUtils from "../TestUtils"; import { DbResult, Guid, Id64, Id64String, Logger, LogLevel, OpenMode } from "@itwin/core-bentley"; import { AxisAlignedBox3d, BriefcaseIdValue, Code, CodeScopeSpec, CodeSpec, ColorDef, CreateIModelProps, DefinitionElementProps, ElementAspectProps, ElementProps, - ExternalSourceAspectProps, GeometricElement2dProps, ImageSourceFormat, IModel, IModelError, InformationPartitionElementProps, ModelProps, PhysicalElementProps, Placement3d, ProfileOptions, QueryRowFormat, RelatedElement, RelationshipProps, RepositoryLinkProps, + ExternalSourceAspectProps, GeometricElement2dProps, GeometryPartProps, GeometryStreamBuilder, GeometryStreamProps, ImageSourceFormat, IModel, IModelError, InformationPartitionElementProps, ModelProps, PhysicalElementProps, Placement3d, ProfileOptions, QueryRowFormat, RelatedElement, RelationshipProps, RepositoryLinkProps, } from "@itwin/core-common"; -import { Point3d, Range3d, StandardViewIndex, Transform, YawPitchRollAngles } from "@itwin/core-geometry"; +import { Box, Point3d, Range3d, StandardViewIndex, Transform, Vector3d, YawPitchRollAngles } from "@itwin/core-geometry"; import { IModelExporter, IModelExportHandler, IModelTransformer, IModelTransformOptions, TransformerLoggerCategory } from "../../transformer"; import { AspectTrackingImporter, @@ -2635,6 +2636,104 @@ describe("IModelTransformer", () => { targetDb.close(); }); + function createBigGeomPart(offset = 0): GeometryStreamProps { + const geometryStreamBuilder = new GeometryStreamBuilder(); + for (let i = 1; i < 100_000; ++i) { + geometryStreamBuilder.appendGeometry(Box.createDgnBox( + Point3d.createZero(), Vector3d.unitX(), Vector3d.unitY(), new Point3d(0, 0, i + offset), + i + offset, i + offset, i + offset, i + offset, true, + )!); + } + return geometryStreamBuilder.geometryStream; + } + + function createBoxWithGeomParts(geometryPartId: Id64String): GeometryStreamProps { + const geometryStreamBuilder = new GeometryStreamBuilder(); + geometryStreamBuilder.appendGeometry(Box.createDgnBox( + Point3d.createZero(), Vector3d.unitX(), Vector3d.unitY(), new Point3d(0, 0, 0), + 0, 0, 0, 0, true, + )!); + geometryStreamBuilder.appendGeometryPart3d(geometryPartId); + geometryStreamBuilder.appendGeometryPart3d(geometryPartId); + return geometryStreamBuilder.geometryStream; + } + + it("processAll prunes unnecessary geometry parts", async function () { + const sourceDbFile = IModelTransformerTestUtils.prepareOutputFile("IModelTransformer", "PruneGeomParts.bim"); + const sourceDb = SnapshotDb.createEmpty(sourceDbFile, { rootSubject: { name: "PruneGeomPartsSrc" } }); + + const sourceModelId = PhysicalModel.insert(sourceDb, IModel.rootSubjectId, "Physical"); + const categoryId = SpatialCategory.insert(sourceDb, IModel.dictionaryId, "SpatialCategory", { color: ColorDef.green.toJSON() }); + + const [physObj1, physObj2] = [1, 2].map((i) => { + const geometryPartProps: GeometryPartProps = { + classFullName: GeometryPart.classFullName, + model: IModelDb.dictionaryId, + code: GeometryPart.createCode(sourceDb, IModelDb.dictionaryId, `GeometryPart${i}`), + geom: createBigGeomPart(i), + }; + + const geomPartId = sourceDb.elements.insertElement(geometryPartProps); + + const physObjProps: PhysicalElementProps = { + classFullName: PhysicalObject.classFullName, + model: sourceModelId, + category: categoryId, + code: Code.createEmpty(), + userLabel: "PhysicalObject1", + geom: createBoxWithGeomParts(geomPartId), + placement: { + origin: Point3d.create(1, 1, 1), + angles: YawPitchRollAngles.createDegrees(0, 0, 0), + }, + }; + + const physObjId = sourceDb.elements.insertElement(physObjProps); + + return { geomPartId, id: physObjId }; + }); + + sourceDb.saveChanges(); + + const targetDbFile: string = IModelTransformerTestUtils.prepareOutputFile("IModelTransformer", "PruneGeomParts-Target.bim"); + const targetDb = StandaloneDb.createEmpty(targetDbFile, { rootSubject: { name: "PruneGeomParts" } }); + targetDb.saveChanges(); + + const transformer = new IModelTransformer(sourceDb, targetDb, { cleanupUnusedGeometryParts: true }); + // expect this to not reject, adding chai as promised makes the error less readable + await transformer.processSchemas(); + const targetModelId = PhysicalModel.insert(targetDb, IModel.rootSubjectId, "Physical"); + + const physObj1Elem = sourceDb.elements.getElement(physObj1.id); + + transformer.context.remapElement(physObj1Elem.model, targetModelId); + transformer.shouldExportElement = (elem) => elem.id !== physObj2.id; + await transformer.processAll(); + + targetDb.saveChanges(); + + assert(Id64.isValidId64(transformer.context.findTargetElementId(physObj1.id))); + assert(!Id64.isValidId64(transformer.context.findTargetElementId(physObj2.id))); + expect(count(sourceDb, GeometryPart.classFullName)).to.equal(2); + expect(count(targetDb, GeometryPart.classFullName)).to.equal(1); + + targetDb.nativeDb.vacuum(); + targetDb.saveChanges(); + + function printSize(p: string) { + // eslint-disable-next-line + console.log(`Size of ${p}\n`, child_process.execSync(`du -h ${p}`, {encoding: "utf-8"})); + } + + printSize(sourceDbFile); + printSize(targetDbFile); + + // clean up + transformer.dispose(); + sourceDb.close(); + targetDb.close(); + }); + /** unskip to generate a javascript CPU profile on just the processAll portion of an iModel */ it.skip("should profile an IModel transformation", async function () { const sourceDbFile = IModelTransformerTestUtils.prepareOutputFile("IModelTransformer", "ProfileTransformation.bim");