diff --git a/libraries/hfm/src/hfm/HFM.h b/libraries/hfm/src/hfm/HFM.h index aadc9e49ab..d082f30dc5 100644 --- a/libraries/hfm/src/hfm/HFM.h +++ b/libraries/hfm/src/hfm/HFM.h @@ -342,6 +342,26 @@ public: FlowData flowData; void debugDump(); + + /** + * @brief Get the number of warnings that were generated when loading this model. + * + * These may indicate non-compliance with the spec, or usage of deprecated functionality. + * This function is intended to be used for testing. + * + * @return Count + */ + int loadWarningCount = 0; + + /** + * @brief Get the number of errors that were generated when loading this model. + * + * Errors indicate the model is probably broken and unusable. + * + * @return Count + */ + int loadErrorCount = 0; + }; }; diff --git a/libraries/model-serializers/src/GLTFSerializer.cpp b/libraries/model-serializers/src/GLTFSerializer.cpp index 01de57d980..2b8acb6429 100755 --- a/libraries/model-serializers/src/GLTFSerializer.cpp +++ b/libraries/model-serializers/src/GLTFSerializer.cpp @@ -1498,6 +1498,32 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash& } for (int c = 0; c < clusterJoints.size(); ++c) { + if (mesh.clusterIndices.length() <= prevMeshClusterIndexCount + c) { + qCWarning(modelformat) << "Trying to write past end of clusterIndices at" << prevMeshClusterIndexCount + c; + hfmModel.loadErrorCount++; + continue; + } + + if ( clusterJoints.length() <= c) { + qCWarning(modelformat) << "Trying to read past end of clusterJoints at" << c; + hfmModel.loadErrorCount++; + continue; + } + + if ( _file.skins.length() <= node.skin ) { + qCWarning(modelformat) << "Trying to read past end of _file.skins at" << node.skin; + hfmModel.loadErrorCount++; + continue; + } + + if ( _file.skins[node.skin].joints.length() <= clusterJoints[c]) { + qCWarning(modelformat) << "Trying to read past end of _file.skins[node.skin].joints at" << clusterJoints[c] + << "; there are only" << _file.skins[node.skin].joints.length() << "for skin" << node.skin; + hfmModel.loadErrorCount++; + continue; + } + + mesh.clusterIndices[prevMeshClusterIndexCount + c] = originalToNewNodeIndexMap[_file.skins[node.skin].joints[clusterJoints[c]]]; } diff --git a/tests/model-serializers/CMakeLists.txt b/tests/model-serializers/CMakeLists.txt new file mode 100644 index 0000000000..c4df515279 --- /dev/null +++ b/tests/model-serializers/CMakeLists.txt @@ -0,0 +1,10 @@ + +# Declare dependencies +macro (setup_testcase_dependencies) + # link in the shared libraries + link_hifi_libraries(shared test-utils model-serializers networking model-networking hfm graphics gpu image) + + package_libraries_for_deployment() +endmacro () + +setup_hifi_testcase(Script Network) diff --git a/tests/model-serializers/src/ModelSerializersTests.cpp b/tests/model-serializers/src/ModelSerializersTests.cpp new file mode 100644 index 0000000000..13a83fadde --- /dev/null +++ b/tests/model-serializers/src/ModelSerializersTests.cpp @@ -0,0 +1,119 @@ +// +// AABoxCubeTests.h +// tests/octree/src +// +// Created by Brad Hefta-Gaub on 06/04/2014. +// Copyright 2014 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#include "ModelSerializersTests.h" +#include "GLTFSerializer.h" +#include "FBXSerializer.h" +#include "OBJSerializer.h" + +#include "Gzip.h" +#include "model-networking/ModelLoader.h" +#include +#include "DependencyManager.h" +#include "ResourceManager.h" +#include "AssetClient.h" +#include "LimitedNodeList.h" +#include "NodeList.h" + +#include +#include +#include +#include +#include +#include + + +QTEST_MAIN(ModelSerializersTests) + +void ModelSerializersTests::initTestCase() { + qRegisterMetaType("QNetworkReply*"); + + DependencyManager::registerInheritance(); + DependencyManager::set(NodeType::Agent, INVALID_PORT); + + DependencyManager::set(); // ModelFormatRegistry must be defined before ModelCache. See the ModelCache constructor. + DependencyManager::set(); + DependencyManager::set(); + + + + auto modelFormatRegistry = DependencyManager::get(); + modelFormatRegistry->addFormat(FBXSerializer()); + modelFormatRegistry->addFormat(OBJSerializer()); + modelFormatRegistry->addFormat(GLTFSerializer()); +} + +void ModelSerializersTests::loadGLTF_data() { + QTest::addColumn("filename"); + QTest::addColumn("expectWarnings"); + QTest::addColumn("expectErrors"); + + QTest::newRow("crash1") << "635d84711260644e7e393e0b.glb.gz" << false << true; + QTest::newRow("crash2") << "dude.glb.gz" << false << true; + +} + +void ModelSerializersTests::loadGLTF() { + QFETCH(QString, filename); + QFETCH(bool, expectWarnings); + QFETCH(bool, expectErrors); + + + QFile gltf_file(filename); + QVERIFY(gltf_file.open(QIODevice::ReadOnly)); + + QByteArray data = gltf_file.readAll(); + QByteArray uncompressedData; + QUrl url("https://example.com"); + + qInfo() << "URL: " << url; + + if (filename.toLower().endsWith(".gz")) { + url.setPath("/" + filename.chopped(3)); + + if (gunzip(data, uncompressedData)) { + qInfo() << "Uncompressed into" << uncompressedData.length(); + } else { + qCritical() << "Failed to uncompress"; + } + } else { + url.setPath("/" + filename); + uncompressedData = data; + } + + + ModelLoader loader; + QMultiHash serializerMapping; + std::string webMediaType; + + serializerMapping.insert("combineParts", true); + serializerMapping.insert("deduplicateIndices", true); + + qInfo() << "Loading model from" << uncompressedData.length() << "bytes data, url" << url; + + // Check that we can find a serializer for this + auto serializer = DependencyManager::get()->getSerializerForMediaType(uncompressedData, url, webMediaType); + QVERIFY(serializer); + + + + hfm::Model::Pointer model = loader.load(uncompressedData, serializerMapping, url, webMediaType); + QVERIFY(model); + QVERIFY(!model->meshes.empty()); + QVERIFY(!model->joints.empty()); + + qInfo() << "Model was loaded with" << model->meshes.count() << "meshes and" << model->joints.count() << "joints. Found" << model->loadWarningCount << "warnings and" << model->loadErrorCount << "errors"; + + // Some models we test are expected to be broken. We're testing that we can load the model without blowing up, + // so loading it with errors is still a successful test. + QVERIFY(expectWarnings == (model->loadWarningCount>0)); + QVERIFY(expectErrors == (model->loadErrorCount>0)); +} diff --git a/tests/model-serializers/src/ModelSerializersTests.h b/tests/model-serializers/src/ModelSerializersTests.h new file mode 100644 index 0000000000..45b913a5d7 --- /dev/null +++ b/tests/model-serializers/src/ModelSerializersTests.h @@ -0,0 +1,27 @@ +// +// ModelSerializersTests.h +// tests/model-serializers/src +// +// Created by Dale Glass on 20/11/2022. +// Copyright 2022 Overte e.V. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef overte_ModelSerializersTests_h +#define overte_ModelSerializersTests_h + +#include + +class ModelSerializersTests : public QObject { + Q_OBJECT + +private slots: + void initTestCase(); + void loadGLTF_data(); + void loadGLTF(); + +}; + +#endif // overte_ModelSerializersTests_h