• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1/*
2 * Copyright (c) 2022-2023 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 *     http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16import ExtendInterface from "./ExtendInterface";
17import VerificationMode from "./VerificationMode";
18import ArgumentMatchers from "./ArgumentMatchers";
19
20class MockKit {
21
22    constructor() {
23        this.mFunctions = [];
24        this.stubs = new Map();
25        this.recordCalls = new Map();
26        this.currentSetKey = new Map();
27        this.mockObj = null;
28        this.recordMockedMethod = new Map();
29    }
30
31    init() {
32        this.reset();
33    }
34
35    reset() {
36        this.mFunctions = [];
37        this.stubs = {};
38        this.recordCalls = {};
39        this.currentSetKey = new Map();
40        this.mockObj = null;
41        this.recordMockedMethod = new Map();
42    }
43
44    clearAll() {
45        this.reset();
46        var props = Object.keys(this);
47        for (var i = 0; i < props.length; i++) {
48            delete this[props[i]];
49        }
50
51        var props = Object.getOwnPropertyNames(this);
52        for (var i = 0; i < props.length; i++) {
53            delete this[props[i]];
54        }
55        for (var key in this) {
56            delete this[key];
57        }
58    }
59
60    clear(obj) {
61        if (!obj) throw Error("Please enter an object to be cleaned");
62        if (typeof (obj) !== 'object' && typeof (obj) !== 'function') throw new Error('Not a object or static class');
63        this.recordMockedMethod.forEach(function (value, key, map) {
64            if (key) {
65                obj[key] = value;
66            }
67        });
68    }
69
70    ignoreMock(obj, method) {
71        if (typeof (obj) !== 'object' && typeof (obj) !== 'function') throw new Error('Not a object or static class');
72        if (typeof (method) !== 'function') throw new Error('Not a function');
73        let og = this.recordMockedMethod.get(method.propName);
74        if (og) {
75            obj[method.propName] = og;
76            this.recordMockedMethod.set(method.propName, undefined);
77        }
78    }
79
80    extend(dest, source) {
81        dest["stub"] = source["stub"];
82        dest["afterReturn"] = source["afterReturn"];
83        dest["afterReturnNothing"] = source["afterReturnNothing"];
84        dest["afterAction"] = source["afterAction"];
85        dest["afterThrow"] = source["afterThrow"];
86        dest["stubMockedCall"] = source["stubMockedCall"];
87        dest["clear"] = source["clear"];
88        return dest;
89    }
90
91    stubApply(f, params, returnInfo) {
92        let values = this.stubs.get(f);
93        if (!values) {
94            values = new Map();
95        }
96        let key = params[0];
97        if (typeof key == "undefined") {
98            key = "anonymous-mock-" + f.propName;
99        }
100        let matcher = new ArgumentMatchers();
101        if (matcher.matcheStubKey(key)) {
102            key = matcher.matcheStubKey(key);
103            if (key) {
104                this.currentSetKey.set(f, key);
105            }
106        }
107        values.set(key, returnInfo);
108        this.stubs.set(f, values);
109    }
110
111    getReturnInfo(f, params) {
112        let values = this.stubs.get(f);
113        if (!values) {
114            return undefined;
115        }
116        let retrunKet = params[0];
117        if (typeof retrunKet == "undefined") {
118            retrunKet = "anonymous-mock-" + f.propName;
119        }
120        let stubSetKey = this.currentSetKey.get(f);
121
122        if (stubSetKey && (typeof (retrunKet) !== "undefined")) {
123            retrunKet = stubSetKey;
124        }
125        let matcher = new ArgumentMatchers();
126        if (matcher.matcheReturnKey(params[0], undefined, stubSetKey) && matcher.matcheReturnKey(params[0], undefined, stubSetKey) !== stubSetKey) {
127            retrunKet = params[0];
128        }
129
130        values.forEach(function (value, key, map) {
131            if (ArgumentMatchers.isRegExp(key) && matcher.matcheReturnKey(params[0], key)) {
132                retrunKet = key;
133            }
134        });
135
136        return values.get(retrunKet);
137    }
138
139    findName(obj, value) {
140        let properties = this.findProperties(obj);
141        let name = null;
142        properties.filter(item => (item !== 'caller' && item !== 'arguments')).forEach(
143            function (va1, idx, array) {
144                if (obj[va1] === value) {
145                    name = va1;
146                }
147            }
148        );
149        return name;
150    }
151
152    isFunctionFromPrototype(f, container, propName) {
153        if (container.constructor !== Object && container.constructor.prototype !== container) {
154            return container.constructor.prototype[propName] === f;
155        }
156        return false;
157    }
158
159    findProperties(obj, ...arg) {
160        function getProperty(new_obj) {
161            if (new_obj.__proto__ === null) {
162                return [];
163            }
164            let properties = Object.getOwnPropertyNames(new_obj);
165            return [...properties, ...getProperty(new_obj.__proto__)];
166        }
167        return getProperty(obj);
168    }
169
170    recordMethodCall(originalMethod, args) {
171        Function.prototype.getName = function () {
172            return this.name || this.toString().match(/function\s*([^(]*)\(/)[1];
173        };
174        let name = originalMethod.getName();
175        let arglistString = name + '(' + Array.from(args).toString() + ')';
176        let records = this.recordCalls.get(arglistString);
177        if (!records) {
178            records = 0;
179        }
180        records++;
181        this.recordCalls.set(arglistString, records);
182    }
183
184    mockFunc(originalObject, originalMethod) {
185        let tmp = this;
186        this.originalMethod = originalMethod;
187        let f = function () {
188            let args = arguments;
189            let action = tmp.getReturnInfo(f, args);
190            if (originalMethod) {
191                tmp.recordMethodCall(originalMethod, args);
192            }
193            if (action) {
194                return action.apply(this, args);
195            }
196        };
197
198        f.container = null || originalObject;
199        f.original = originalMethod || null;
200
201        if (originalObject && originalMethod) {
202            if (typeof (originalMethod) !== 'function') throw new Error('Not a function');
203            var name = this.findName(originalObject, originalMethod);
204            originalObject[name] = f;
205            this.recordMockedMethod.set(name, originalMethod);
206            f.propName = name;
207            f.originalFromPrototype = this.isFunctionFromPrototype(f.original, originalObject, f.propName);
208        }
209        f.mocker = this;
210        this.mFunctions.push(f);
211        this.extend(f, new ExtendInterface(this));
212        return f;
213    }
214
215    verify(methodName, argsArray) {
216        if (!methodName) {
217            throw Error("not a function name");
218        }
219        let a = this.recordCalls.get(methodName + '(' + argsArray.toString() + ')');
220        return new VerificationMode(a ? a : 0);
221    }
222
223    mockObject(object) {
224        if (!object || typeof object === "string") {
225            throw Error(`this ${object} cannot be mocked`);
226        }
227        const _this = this;
228        let mockedObject = {};
229        let keys = Reflect.ownKeys(object);
230        keys.filter(key => (typeof Reflect.get(object, key)) === 'function')
231            .forEach(key => {
232                mockedObject[key] = object[key];
233                mockedObject[key] = _this.mockFunc(mockedObject, mockedObject[key]);
234            });
235        return mockedObject;
236    }
237}
238
239function ifMockedFunction(f) {
240    if (Object.prototype.toString.call(f) !== "[object Function]" &&
241        Object.prototype.toString.call(f) !== "[object AsyncFunction]") {
242        throw Error("not a function");
243    }
244    if (!f.stub) {
245        throw Error("not a mock function");
246    }
247    return true;
248}
249
250function when(f) {
251    if (ifMockedFunction(f)) {
252        return f.stub.bind(f);
253    }
254}
255
256export {MockKit, when};