Skip to content

Commit

Permalink
Merge pull request #285 from CesiumGS/cli-content-file-validation
Browse files Browse the repository at this point in the history
Add support for validating single content files
  • Loading branch information
lilleyse authored Oct 16, 2023
2 parents 11709b4 + b208ab2 commit 3b3e32b
Show file tree
Hide file tree
Showing 5 changed files with 109 additions and 3 deletions.
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,13 @@ This will validate all tileset files in the given directory and all its subdirec
npx 3d-tiles-validator --tilesetsDirectory specs/data/Samples/ --tilesetGlobPattern **/*.json
```

#### Validate a tile content file
```
npx 3d-tiles-validator --tileContentFile specs/data/gltfExtensions/FeatureIdAttributeAndPropertyTableFeatureIdNotInRange.gltf
```

The input file can be any of the content types that are supported by the validator. This includes B3DM, I3DM, PNTS, CMPT, and glTF/GLB files. The type of the file will be determined, and if it is not a known type, then a validation warning will be created.

### Report Files

By default, validation reports are printed to the console.
Expand Down
30 changes: 30 additions & 0 deletions src/ValidatorMain.ts
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,16 @@ export class ValidatorMain {
config.writeReports,
validationOptions
);
} else if (config.tileContentFile) {
const reportFileName = ValidatorMain.obtainReportFileName(
config,
config.tileContentFile
);
await ValidatorMain.validateTileContentFile(
config.tileContentFile,
reportFileName,
validationOptions
);
} else if (config.metadataSchemaFile) {
const reportFileName = ValidatorMain.obtainReportFileName(
config,
Expand Down Expand Up @@ -172,6 +182,26 @@ export class ValidatorMain {
console.log(` ${numFilesWithInfos} files with infos`);
}

static async validateTileContentFile(
fileName: string,
reportFileName: string | undefined,
options: ValidationOptions | undefined
): Promise<ValidationResult> {
console.log("Validating tile content " + fileName);

const validationResult = await Validators.validateTileContentFile(
fileName,
options
);
if (defined(reportFileName)) {
await writeUnchecked(reportFileName, validationResult.serialize());
} else {
console.log("Validation result:");
console.log(validationResult.serialize());
}
return validationResult;
}

static async validateSchemaFile(
fileName: string,
reportFileName: string | undefined
Expand Down
5 changes: 5 additions & 0 deletions src/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,11 @@ const args = yargs(process.argv.slice(1))
"of the report will be derived from the input file name, " +
"and be written into the same directory as the input file.",
},
tileContentFile: {
type: "string",
alias: "f",
describe: "The tile content input file path",
},
optionsFile: {
type: "string",
alias: "o",
Expand Down
1 change: 1 addition & 0 deletions src/validation/ValidationOptions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ export class ValidationOptions {
constructor() {
this._validateContentData = true;
this._includeContentTypes = undefined;
this._excludeContentTypes = undefined;
}

/**
Expand Down
69 changes: 66 additions & 3 deletions src/validation/Validators.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,10 @@ import path from "path";
import fs from "fs";

import { defined, Schema } from "3d-tiles-tools";
import { LazyContentData } from "3d-tiles-tools";
import { Buffers } from "3d-tiles-tools";

import { ResourceResolvers } from "3d-tiles-tools";
import { TileImplicitTiling } from "3d-tiles-tools";

import { Validator } from "./Validator";
import { TilesetValidator } from "./TilesetValidator";
Expand All @@ -14,12 +15,11 @@ import { SubtreeValidator } from "./SubtreeValidator";
import { ValidationOptions } from "./ValidationOptions";
import { ExtendedObjectsValidators } from "./ExtendedObjectsValidators";
import { TilesetPackageValidator } from "./TilesetPackageValidator";
import { ContentDataValidators } from "./ContentDataValidators";
import { ValidatedElement } from "./ValidatedElement";

import { SchemaValidator } from "./metadata/SchemaValidator";

import { TileImplicitTiling } from "3d-tiles-tools";

import { IoValidationIssues } from "../issues/IoValidationIssue";
import { ContentValidationIssues } from "../issues/ContentValidationIssues";

Expand Down Expand Up @@ -175,6 +175,69 @@ export class Validators {
return context.getResult();
}

/**
* Performs a default validation of the given tile content file, and
* returns a promise to the `ValidationResult`.
*
* The given file may be of any format. The method will detect
* the format, and decide whether it can perform a sensible
* validation, based on the existing validation functionality.
*
* @param filePath - The file path
* @param validationOptions - The `ValidationOptions`. When this
* is not given (or `undefined`), then default validation options
* will be used. See {@link ValidationOptions}.
* @returns A promise to a `ValidationResult` that is fulfilled when
* the validation finished.
* @beta
*/
static async validateTileContentFile(
filePath: string,
validationOptions?: ValidationOptions
): Promise<ValidationResult> {
const directory = path.dirname(filePath);
const fileName = path.basename(filePath);
const resourceResolver =
ResourceResolvers.createFileResourceResolver(directory);
const context = new ValidationContext(
directory,
resourceResolver,
validationOptions
);
const contentData = new LazyContentData(fileName, resourceResolver);

// Check if the file exists, and bail out early if it doesn't
const dataExists = await contentData.exists();
if (!dataExists) {
const message = `Content file ${filePath} could not be resolved`;
const issue = ContentValidationIssues.CONTENT_VALIDATION_ERROR(
filePath,
message
);
context.addIssue(issue);
return context.getResult();
}

// Find the validator for the content data, and bail
// out early if none could be found
const dataValidator = await ContentDataValidators.findContentDataValidator(
contentData
);
if (!defined(dataValidator)) {
const message = `No valid content type could be determined for ${filePath}`;
const issue = ContentValidationIssues.CONTENT_VALIDATION_WARNING(
filePath,
message
);
context.addIssue(issue);
return context.getResult();
}

// Perform the actual validation
await dataValidator.validateObject(fileName, contentData, context);
return context.getResult();
}

/**
* Creates a `SchemaValidator` with an unspecified default configuration.
*
Expand Down

0 comments on commit 3b3e32b

Please sign in to comment.