@genialis/resolwe
Version:
Resolwe frontend libraries
183 lines (181 loc) • 24.7 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
var _ = require("lodash");
var angular = require("angular");
require("angular-mocks");
var mock_1 = require("../api/mock");
var matchers_1 = require("./matchers");
var error_1 = require("../core/errors/error");
var lang_1 = require("../core/utils/lang");
var mock_2 = require("./mock");
require("./matchers");
// Service modules that should be loaded.
require("../core/shared_store/index");
require("../core/components/manager");
/**
* Helper for creating simple shared stores in tests. Also see
* [[SharedStoreProvider.create]].
*
* @param storeId Identifier of the shared store (must be globally unique)
* @param initialState Optional initial state of the shared store
*/
function createSharedStore(storeId, initialState) {
if (initialState === void 0) { initialState = null; }
return {
storeId: storeId,
initialState: initialState,
};
}
exports.createSharedStore = createSharedStore;
/**
* Helper for using shared stores in tests.
*
* @param storeId Identifier of the shared store (must be globally unique)
* @param factory Shared store class
*/
function useSharedStore(storeId, factory) {
return {
storeId: storeId,
factory: factory,
};
}
exports.useSharedStore = useSharedStore;
/**
* Helper function when unit testing components which compiles a component.
*
* @param $compile Compile service
* @param $scope Scope service
* @param template template in the form of a directive, e.g. `'<gen-some-component><gen-some-component>'`
* @returns {ComponentDescriptor} Element and its controller
*/
function getComponent($compile, $scope, template) {
// TODO: instead of having getComponent<T>, we could have getComponent<T extends ComponentBase> and then you would do
// T.asView thus reducing the need for template argument, but looks like typescript does not support that (yet)
// https://github.com/Microsoft/TypeScript/issues/5677
var element = $compile(template)($scope);
$scope.$digest();
var ctrl;
try {
ctrl = $scope['$$childTail'].ctrl;
}
catch (e) {
throw new error_1.GenError("Unable to fetch component controller. Did you load your module in tests?");
}
return {
element: element,
ctrl: ctrl,
};
}
exports.getComponent = getComponent;
/**
* A helper function to ease testing of components. It will take care of mocking the
* usual modules needed for testing components and preparing a module that you can use
* to register test components.
*
* The test case is passed an instance of [[ComponentTester]], which contains some
* useful properties and methods for testing components.
*
* The following modules are automatically loaded:
* * `resolwe.services.shared_store`
*
* If you need to load any additional modules, specify them in the `modules`
* argument.
*
* @param description Test case description
* @param modules List of modules to load for this test
* @param tests Test case body
* @param apiClass Optional mock API class that should be used
* @param baseModules Optional list of modules to load before everything for this test
*/
function describeComponent(description, modules, tests, apiClass, baseModules) {
if (apiClass === void 0) { apiClass = mock_1.MockApi; }
if (baseModules === void 0) { baseModules = []; }
describe(description, function () {
var $compile;
var $scope;
var mockApi;
var moduleName = 'resolwe.tests.' + description.replace(/ /g, '.');
var module = angular.module(moduleName, []);
// Load base modules.
beforeEach(angular.mock.module('resolwe.services.shared_store'));
beforeEach(angular.mock.module('resolwe.services.state_manager'));
baseModules.forEach(function (baseModule) { return beforeEach(angular.mock.module(baseModule)); });
beforeEach(angular.mock.module(function ($provide) {
// Explicitly set root element because tests do not go through usual
// Angular bootstrapping.
$provide.value('$rootElement', angular.element(document.body));
// Replace usual API service with mock API.
$provide.service('api', lang_1.ngCompose([apiClass, mock_2.MockApiService]));
}));
beforeEach(angular.mock.module(moduleName));
// Register any shared stores.
var _a = _.partition(modules, function (m) { return !!m.storeId; }), sharedStores = _a[0], additionalModules = _a[1];
module.config(["sharedStoreManagerProvider", function (sharedStoreManagerProvider) {
_.each(sharedStores, function (descriptor) {
if (descriptor.factory) {
sharedStoreManagerProvider.register(descriptor.storeId, descriptor.factory);
}
else {
sharedStoreManagerProvider.create(descriptor.storeId, descriptor.initialState);
}
});
}]);
_.each(additionalModules, function (additionalModule) {
beforeEach(angular.mock.module(additionalModule));
});
// A container in DOM where we can temporarily append component elements.
var containerElement = null;
function provideRealDOM() {
removeRealDOM();
var body = angular.element(document.body);
containerElement = angular.element('<div id="test-container-element"></div>');
body.append(containerElement);
}
function removeRealDOM() {
if (containerElement) {
containerElement.remove();
containerElement = null;
}
}
beforeEach(function () {
jasmine.addMatchers(matchers_1.ngEqualMatcher);
});
beforeEach(angular.mock.inject(function (_$compile_, _$rootScope_, _api_) {
$compile = _$compile_;
$scope = _$rootScope_.$new();
mockApi = _api_;
if (_.contains(tests.toString(), 'debugger') || _.contains(tests.toString(), ' fit(')) {
provideRealDOM();
mockApi.logUnhandledRequests(true);
}
}));
afterEach(function () {
$scope.$destroy();
removeRealDOM();
});
tests({
module: module,
createComponent: function (template) {
var component = getComponent($compile, $scope, template);
if (containerElement) {
// Append component element to actual DOM. Otherwise, computations like height will not work.
containerElement.append(component.element);
$scope.$digest();
}
return component;
},
digest: function () {
$scope.$digest();
},
api: function () {
return mockApi;
},
scope: function () {
return $scope;
},
provideRealDOM: provideRealDOM,
});
});
}
exports.describeComponent = describeComponent;
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["../src/tests/component.ts"],"names":[],"mappings":"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","file":"tests/component.js","sourcesContent":["import * as _ from 'lodash';\nimport * as angular from 'angular';\nimport 'angular-mocks';\n\nimport {ResolweApi} from '../api/index';\nimport {MockBase, MockApi} from '../api/mock';\nimport {ngEqualMatcher} from './matchers';\nimport {SharedStoreFactory, SharedStoreProvider} from '../core/shared_store/index';\nimport {GenError} from '../core/errors/error';\nimport {ngCompose} from '../core/utils/lang';\nimport {MockApiService} from './mock';\nimport './matchers';\n\n// Service modules that should be loaded.\nimport '../core/shared_store/index';\nimport '../core/components/manager';\n\nexport interface ComponentDescriptor<T> {\n    element: angular.IAugmentedJQuery;\n    ctrl: T;\n}\n\nexport interface SharedStoreDescriptor {\n    storeId: string;\n    initialState?: any;\n    factory?: SharedStoreFactory<any, any>;\n}\n\n/**\n * Helper for creating simple shared stores in tests. Also see\n * [[SharedStoreProvider.create]].\n *\n * @param storeId Identifier of the shared store (must be globally unique)\n * @param initialState Optional initial state of the shared store\n */\nexport function createSharedStore(storeId: string, initialState: any = null): SharedStoreDescriptor {\n    return {\n        storeId,\n        initialState,\n    };\n}\n\n/**\n * Helper for using shared stores in tests.\n *\n * @param storeId Identifier of the shared store (must be globally unique)\n * @param factory Shared store class\n */\nexport function useSharedStore(storeId: string, factory: SharedStoreFactory<any, any>): SharedStoreDescriptor {\n    return {\n        storeId,\n        factory,\n    };\n}\n\n/**\n * Helper function when unit testing components which compiles a component.\n *\n * @param $compile Compile service\n * @param $scope Scope service\n * @param template template in the form of a directive, e.g. `'<gen-some-component><gen-some-component>'`\n * @returns {ComponentDescriptor} Element and its controller\n */\nexport function getComponent<T>($compile: angular.ICompileService, $scope: angular.IScope, template: string): ComponentDescriptor<T> {\n\n    // TODO: instead of having getComponent<T>, we could have getComponent<T extends ComponentBase> and then you would do\n    // T.asView thus reducing the need for template argument, but looks like typescript does not support that (yet)\n    // https://github.com/Microsoft/TypeScript/issues/5677\n\n    const element = $compile(template)($scope);\n    $scope.$digest();\n\n    let ctrl: T;\n    try {\n        ctrl = $scope['$$childTail'].ctrl;\n    } catch (e) {\n        throw new GenError(\"Unable to fetch component controller. Did you load your module in tests?\");\n    }\n\n    return {\n        element: element,\n        ctrl: ctrl,\n    };\n}\n\n/**\n * Interface exposed to test case functions, which are created using [[describeComponent]].\n */\nexport interface ComponentTester {\n    /// Angular module, which may be used to register test components in.\n    module: angular.IModule;\n\n    /**\n     * See [[getComponent]].\n     */\n    createComponent<T>(template: string): ComponentDescriptor<T>;\n\n    /**\n     * Runs an Angular digest cycle.\n     */\n    digest(): void;\n\n    /**\n     * Returns the mock version of the API, which may be used to simulate the backend\n     * when testing components. The mock API is automatically injected into components\n     * and replaces the usual API.\n     */\n    api(): ResolweApi & MockBase & MockApiService;\n\n    /**\n     * Returns the scope.\n     */\n    scope(): angular.IScope;\n\n    /**\n     * Ensures that the tested components are inserted into an actual DOM, so things\n     * like height calculations work as expected. This function must be called before\n     * any [[createComponent]] calls.\n     */\n    provideRealDOM(): void;\n}\n\nexport interface MockApiFactory {\n    new (...args: any[]): ResolweApi & MockBase;\n}\n\n/**\n * A helper function to ease testing of components. It will take care of mocking the\n * usual modules needed for testing components and preparing a module that you can use\n * to register test components.\n *\n * The test case is passed an instance of [[ComponentTester]], which contains some\n * useful properties and methods for testing components.\n *\n * The following modules are automatically loaded:\n * * `resolwe.services.shared_store`\n *\n * If you need to load any additional modules, specify them in the `modules`\n * argument.\n *\n * @param description Test case description\n * @param modules List of modules to load for this test\n * @param tests Test case body\n * @param apiClass Optional mock API class that should be used\n * @param baseModules Optional list of modules to load before everything for this test\n */\nexport function describeComponent(description: string,\n                                  modules: any[],\n                                  tests: (tester: ComponentTester) => void,\n                                  apiClass: MockApiFactory = MockApi,\n                                  baseModules: string[] = []): void {\n    describe(description, () => {\n        let $compile: angular.ICompileService;\n        let $scope: angular.IScope;\n        let mockApi: ResolweApi & MockBase & MockApiService;\n\n        const moduleName = 'resolwe.tests.' + description.replace(/ /g, '.');\n        const module: angular.IModule = angular.module(moduleName, []);\n\n        // Load base modules.\n        beforeEach(angular.mock.module('resolwe.services.shared_store'));\n        beforeEach(angular.mock.module('resolwe.services.state_manager'));\n        baseModules.forEach((baseModule) => beforeEach(angular.mock.module(baseModule)));\n\n        beforeEach(angular.mock.module(($provide: angular.auto.IProvideService) => {\n            // Explicitly set root element because tests do not go through usual\n            // Angular bootstrapping.\n            $provide.value('$rootElement', angular.element(document.body));\n\n            // Replace usual API service with mock API.\n            $provide.service('api', ngCompose([apiClass, MockApiService]));\n        }));\n        beforeEach(angular.mock.module(moduleName));\n\n        // Register any shared stores.\n        const [sharedStores, additionalModules] = _.partition(modules, (m) => !!m.storeId);\n\n        module.config((sharedStoreManagerProvider: SharedStoreProvider) => {\n            _.each(sharedStores, (descriptor: SharedStoreDescriptor) => {\n                if (descriptor.factory) {\n                    sharedStoreManagerProvider.register(descriptor.storeId, descriptor.factory);\n                } else {\n                    sharedStoreManagerProvider.create(descriptor.storeId, descriptor.initialState);\n                }\n            });\n        });\n\n        _.each(additionalModules, (additionalModule) => {\n            beforeEach(angular.mock.module(additionalModule));\n        });\n\n        // A container in DOM where we can temporarily append component elements.\n        let containerElement: angular.IAugmentedJQuery = null;\n\n        function provideRealDOM(): void {\n            removeRealDOM();\n            const body = angular.element(document.body);\n            containerElement = angular.element('<div id=\"test-container-element\"></div>');\n            body.append(containerElement);\n        }\n\n        function removeRealDOM(): void {\n            if (containerElement) {\n                containerElement.remove();\n                containerElement = null;\n            }\n        }\n\n        beforeEach(() => {\n            jasmine.addMatchers(ngEqualMatcher);\n        });\n\n        beforeEach(angular.mock.inject((_$compile_, _$rootScope_, _api_) => {\n            $compile = _$compile_;\n            $scope = _$rootScope_.$new();\n            mockApi = _api_;\n\n            if (_.contains(tests.toString(), 'debugger') || _.contains(tests.toString(), ' fit(')) {\n                provideRealDOM();\n                mockApi.logUnhandledRequests(true);\n            }\n        }));\n\n        afterEach(() => {\n            $scope.$destroy();\n            removeRealDOM();\n        });\n\n        tests({\n            module: module,\n\n            createComponent: function<T>(template: string): ComponentDescriptor<T> {\n                const component = getComponent<T>($compile, $scope, template);\n\n                if (containerElement) {\n                    // Append component element to actual DOM. Otherwise, computations like height will not work.\n                    containerElement.append(component.element);\n                    $scope.$digest();\n                }\n\n                return component;\n            },\n\n            digest: function(): void {\n                $scope.$digest();\n            },\n\n            api: function(): ResolweApi & MockBase & MockApiService {\n                return mockApi;\n            },\n\n            scope: function(): angular.IScope {\n                return $scope;\n            },\n\n            provideRealDOM: provideRealDOM,\n        });\n    });\n}\n"]}
;