blob: e9b7c2a1da96c9a71e43ec3103e3912e7afd8a7f [file] [log] [blame]
/*
* Copyright 2018-present Open Networking Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { TestBed, inject } from '@angular/core/testing';
import { LogService } from '../log.service';
import { ConsoleLoggerService } from '../consolelogger.service';
import { FnService } from './fn.service';
import { ActivatedRoute, Params } from '@angular/router';
import { of } from 'rxjs';
import * as d3 from 'd3';
class MockActivatedRoute extends ActivatedRoute {
constructor(params: Params) {
super();
this.queryParams = of(params);
}
}
/**
* ONOS GUI -- Util -- General Purpose Functions - Unit Tests
*/
describe('FnService', () => {
let ar: ActivatedRoute;
let fs: FnService;
let mockWindow: Window;
let logServiceSpy: jasmine.SpyObj<LogService>;
const someFunction = () => {};
const someArray = [1, 2, 3];
const someObject = { foo: 'bar'};
const someNumber = 42;
const someString = 'xyyzy';
const someDate = new Date();
const stringArray = ['foo', 'bar'];
beforeEach(() => {
const logSpy = jasmine.createSpyObj('LogService', ['debug', 'warn']);
ar = new MockActivatedRoute({'debug': 'TestService'});
mockWindow = <any>{
innerWidth: 400,
innerHeight: 200,
navigator: {
userAgent: 'defaultUA'
}
};
TestBed.configureTestingModule({
providers: [FnService,
{ provide: LogService, useValue: logSpy },
{ provide: ActivatedRoute, useValue: ar },
{ provide: 'Window', useFactory: (() => mockWindow ) }
]
});
fs = TestBed.get(FnService);
logServiceSpy = TestBed.get(LogService);
});
it('should be created', () => {
expect(fs).toBeTruthy();
});
// === Tests for isF()
it('isF(): null for undefined', () => {
expect(fs.isF(undefined)).toBeNull();
});
it('isF(): null for null', () => {
expect(fs.isF(null)).toBeNull();
});
it('isF(): the reference for function', () => {
expect(fs.isF(someFunction)).toBe(someFunction);
});
it('isF(): null for string', () => {
expect(fs.isF(someString)).toBeNull();
});
it('isF(): null for number', () => {
expect(fs.isF(someNumber)).toBeNull();
});
it('isF(): null for Date', () => {
expect(fs.isF(someDate)).toBeNull();
});
it('isF(): null for array', () => {
expect(fs.isF(someArray)).toBeNull();
});
it('isF(): null for object', () => {
expect(fs.isF(someObject)).toBeNull();
});
// === Tests for isA()
it('isA(): null for undefined', () => {
expect(fs.isA(undefined)).toBeNull();
});
it('isA(): null for null', () => {
expect(fs.isA(null)).toBeNull();
});
it('isA(): null for function', () => {
expect(fs.isA(someFunction)).toBeNull();
});
it('isA(): null for string', () => {
expect(fs.isA(someString)).toBeNull();
});
it('isA(): null for number', () => {
expect(fs.isA(someNumber)).toBeNull();
});
it('isA(): null for Date', () => {
expect(fs.isA(someDate)).toBeNull();
});
it('isA(): the reference for array', () => {
expect(fs.isA(someArray)).toBe(someArray);
});
it('isA(): null for object', () => {
expect(fs.isA(someObject)).toBeNull();
});
// === Tests for isS()
it('isS(): null for undefined', () => {
expect(fs.isS(undefined)).toBeNull();
});
it('isS(): null for null', () => {
expect(fs.isS(null)).toBeNull();
});
it('isS(): null for function', () => {
expect(fs.isS(someFunction)).toBeNull();
});
it('isS(): the reference for string', () => {
expect(fs.isS(someString)).toBe(someString);
});
it('isS(): null for number', () => {
expect(fs.isS(someNumber)).toBeNull();
});
it('isS(): null for Date', () => {
expect(fs.isS(someDate)).toBeNull();
});
it('isS(): null for array', () => {
expect(fs.isS(someArray)).toBeNull();
});
it('isS(): null for object', () => {
expect(fs.isS(someObject)).toBeNull();
});
// === Tests for isO()
it('isO(): null for undefined', () => {
expect(fs.isO(undefined)).toBeNull();
});
it('isO(): null for null', () => {
expect(fs.isO(null)).toBeNull();
});
it('isO(): null for function', () => {
expect(fs.isO(someFunction)).toBeNull();
});
it('isO(): null for string', () => {
expect(fs.isO(someString)).toBeNull();
});
it('isO(): null for number', () => {
expect(fs.isO(someNumber)).toBeNull();
});
it('isO(): null for Date', () => {
expect(fs.isO(someDate)).toBeNull();
});
it('isO(): null for array', () => {
expect(fs.isO(someArray)).toBeNull();
});
it('isO(): the reference for object', () => {
expect(fs.isO(someObject)).toBe(someObject);
});
// === Tests for contains()
it('contains(): false for non-array', () => {
expect(fs.contains(null, 1)).toBeFalsy();
});
it('contains(): true for contained item', () => {
expect(fs.contains(someArray, 1)).toBeTruthy();
expect(fs.contains(stringArray, 'bar')).toBeTruthy();
});
it('contains(): false for non-contained item', () => {
expect(fs.contains(someArray, 109)).toBeFalsy();
expect(fs.contains(stringArray, 'zonko')).toBeFalsy();
});
// === Tests for areFunctions()
it('areFunctions(): true for empty-array', () => {
expect(fs.areFunctions({}, [])).toBeTruthy();
});
it('areFunctions(): true for some api', () => {
expect(fs.areFunctions({
a: () => {},
b: () => {}
}, ['b', 'a'])).toBeTruthy();
});
it('areFunctions(): false for some other api', () => {
expect(fs.areFunctions({
a: () => {},
b: 'not-a-function'
}, ['b', 'a'])).toBeFalsy();
});
it('areFunctions(): extraneous stuff NOT ignored', () => {
expect(fs.areFunctions({
a: () => {},
b: () => {},
c: 1,
d: 'foo'
}, ['a', 'b'])).toBeFalsy();
});
it('areFunctions(): extraneous stuff ignored (alternate fn)', () => {
expect(fs.areFunctionsNonStrict({
a: () => {},
b: () => {},
c: 1,
d: 'foo'
}, ['a', 'b'])).toBeTruthy();
});
// == use the now-tested areFunctions() on our own api:
it('should define api functions', () => {
expect(fs.areFunctions(fs, [
'isF', 'isA', 'isS', 'isO', 'contains',
'areFunctions', 'areFunctionsNonStrict', 'windowSize',
'isMobile', 'isChrome', 'isChromeHeadless', 'isSafari',
'isFirefox', 'parseDebugFlags',
'debugOn', 'debug', 'find', 'inArray', 'removeFromArray',
'isEmptyObject', 'cap', 'noPx', 'noPxStyle', 'endsWith',
'inEvilList', 'analyze', 'sanitize', 'sameObjProps', 'containsObj',
'addToTrie', 'removeFromTrie', 'trieLookup'
// 'find', 'inArray', 'removeFromArray', 'isEmptyObject', 'sameObjProps', 'containsObj', 'cap',
// 'eecode', 'noPx', 'noPxStyle', 'endsWith', 'addToTrie', 'removeFromTrie', 'trieLookup',
// 'classNames', 'extend', 'sanitize'
])).toBeTruthy();
});
// === Tests for windowSize()
it('windowSize(): adjust height', () => {
const dim = fs.windowSize(50);
expect(dim.width).toEqual(400);
expect(dim.height).toEqual(150);
});
it('windowSize(): adjust width', () => {
const dim = fs.windowSize(0, 50);
expect(dim.width).toEqual(350);
expect(dim.height).toEqual(200);
});
it('windowSize(): adjust width and height', () => {
const dim = fs.windowSize(101, 201);
expect(dim.width).toEqual(199);
expect(dim.height).toEqual(99);
});
// === Tests for isMobile()
const uaMap = {
chrome: 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_2) ' +
'AppleWebKit/537.36 (KHTML, like Gecko) ' +
'Chrome/41.0.2272.89 Safari/537.36',
iPad: 'Mozilla/5.0 (iPad; CPU OS 7_0 like Mac OS X) ' +
'AppleWebKit/537.51.1 (KHTML, like Gecko) Version/7.0 ' +
'Mobile/11A465 Safari/9537.53',
iPhone: 'Mozilla/5.0 (iPhone; CPU iPhone OS 7_0 like Mac OS X) ' +
'AppleWebKit/537.51.1 (KHTML, like Gecko) Version/7.0 ' +
'Mobile/11A465 Safari/9537.53'
};
function setUa(key) {
const str = uaMap[key];
expect(str).toBeTruthy();
(<any>mockWindow.navigator).userAgent = str;
}
it('isMobile(): should be false for Chrome on Mac OS X', () => {
setUa('chrome');
expect(fs.isMobile()).toBe(false);
});
it('isMobile(): should be true for Safari on iPad', () => {
setUa('iPad');
expect(fs.isMobile()).toBe(true);
});
it('isMobile(): should be true for Safari on iPhone', () => {
setUa('iPhone');
expect(fs.isMobile()).toBe(true);
});
// === Tests for find()
const dataset = [
{ id: 'foo', name: 'Furby'},
{ id: 'bar', name: 'Barbi'},
{ id: 'baz', name: 'Basil'},
{ id: 'goo', name: 'Gabby'},
{ id: 'zoo', name: 'Zevvv'}
];
it('should not find ooo', () => {
expect(fs.find('ooo', dataset)).toEqual(-1);
});
it('should find foo', () => {
expect(fs.find('foo', dataset)).toEqual(0);
});
it('should find zoo', () => {
expect(fs.find('zoo', dataset)).toEqual(4);
});
it('should not find Simon', () => {
expect(fs.find('Simon', dataset, 'name')).toEqual(-1);
});
it('should find Furby', () => {
expect(fs.find('Furby', dataset, 'name')).toEqual(0);
});
it('should find Zevvv', () => {
expect(fs.find('Zevvv', dataset, 'name')).toEqual(4);
});
// === Tests for inArray()
const objRef = { x: 1, y: 2 };
const array = [1, 3.14, 'hey', objRef, 'there', true];
const array2 = ['b', 'a', 'd', 'a', 's', 's'];
it('should not find HOO', () => {
expect(fs.inArray('HOO', array)).toEqual(-1);
});
it('should find 1', () => {
expect(fs.inArray(1, array)).toEqual(0);
});
it('should find pi', () => {
expect(fs.inArray(3.14, array)).toEqual(1);
});
it('should find hey', () => {
expect(fs.inArray('hey', array)).toEqual(2);
});
it('should find the object', () => {
expect(fs.inArray(objRef, array)).toEqual(3);
});
it('should find there', () => {
expect(fs.inArray('there', array)).toEqual(4);
});
it('should find true', () => {
expect(fs.inArray(true, array)).toEqual(5);
});
it('should find the first occurrence A', () => {
expect(fs.inArray('a', array2)).toEqual(1);
});
it('should find the first occurrence S', () => {
expect(fs.inArray('s', array2)).toEqual(4);
});
it('should not find X', () => {
expect(fs.inArray('x', array2)).toEqual(-1);
});
// === Tests for removeFromArray()
it('should keep the array the same, for non-match', () => {
const array1 = [1, 2, 3];
expect(fs.removeFromArray(4, array1)).toBe(false);
expect(array1).toEqual([1, 2, 3]);
});
it('should remove a value', () => {
const array1a = [1, 2, 3];
expect(fs.removeFromArray(2, array1a)).toBe(true);
expect(array1a).toEqual([1, 3]);
});
it('should remove the first occurrence', () => {
const array1b = ['x', 'y', 'z', 'z', 'y'];
expect(fs.removeFromArray('y', array1b)).toBe(true);
expect(array1b).toEqual(['x', 'z', 'z', 'y']);
expect(fs.removeFromArray('x', array1b)).toBe(true);
expect(array1b).toEqual(['z', 'z', 'y']);
});
// === Tests for isEmptyObject()
it('should return true if an object is empty', () => {
expect(fs.isEmptyObject({})).toBe(true);
});
it('should return false if an object is not empty', () => {
expect(fs.isEmptyObject({foo: 'bar'})).toBe(false);
});
// === Tests for cap()
it('should ignore non-alpha', () => {
expect(fs.cap('123')).toEqual('123');
});
it('should capitalize first char', () => {
expect(fs.cap('Foo')).toEqual('Foo');
expect(fs.cap('foo')).toEqual('Foo');
expect(fs.cap('foo bar')).toEqual('Foo bar');
expect(fs.cap('FOO BAR')).toEqual('Foo bar');
expect(fs.cap('foo Bar')).toEqual('Foo bar');
});
// === Tests for noPx()
it('should return the value without px suffix', () => {
expect(fs.noPx('10px')).toBe(10);
expect(fs.noPx('500px')).toBe(500);
expect(fs.noPx('-80px')).toBe(-80);
});
// === Tests for noPxStyle()
it('should give a style\'s property without px suffix', () => {
const d3Elem = d3.select('body')
.append('div')
.attr('id', 'fooElem')
.style('width', '500px')
.style('height', '200px')
.style('font-size', '12px');
expect(fs.noPxStyle(d3Elem, 'width')).toBe(500);
expect(fs.noPxStyle(d3Elem, 'height')).toBe(200);
expect(fs.noPxStyle(d3Elem, 'font-size')).toBe(12);
d3.select('#fooElem').remove();
});
// === Tests for endsWith()
it('should return true if string ends with foo', () => {
expect(fs.endsWith('barfoo', 'foo')).toBe(true);
});
it('should return false if string doesnt end with foo', () => {
expect(fs.endsWith('barfood', 'foo')).toBe(false);
});
// === Tests for sanitize()
it('should return foo', () => {
expect(fs.sanitize('foo')).toEqual('foo');
});
it('should retain < b > tags', () => {
const str = 'foo <b>bar</b> baz';
expect(fs.sanitize(str)).toEqual(str);
});
it('should retain < i > tags', () => {
const str = 'foo <i>bar</i> baz';
expect(fs.sanitize(str)).toEqual(str);
});
it('should retain < p > tags', () => {
const str = 'foo <p>bar</p> baz';
expect(fs.sanitize(str)).toEqual(str);
});
it('should retain < em > tags', () => {
const str = 'foo <em>bar</em> baz';
expect(fs.sanitize(str)).toEqual(str);
});
it('should retain < strong > tags', () => {
const str = 'foo <strong>bar</strong> baz';
expect(fs.sanitize(str)).toEqual(str);
});
it('should reject < a > tags', () => {
expect(fs.sanitize('test <a href="hah">something</a> this'))
.toEqual('test something this');
});
it('should log a warning for < script > tags', () => {
expect(fs.sanitize('<script>alert("foo");</script>'))
.toEqual('');
expect(logServiceSpy.warn).toHaveBeenCalledWith(
'Unsanitary HTML input -- <script> detected!'
);
});
it('should log a warning for < style > tags', () => {
expect(fs.sanitize('<style> h1 {color:red;} </style>'))
.toEqual('');
expect(logServiceSpy.warn).toHaveBeenCalledWith(
'Unsanitary HTML input -- <style> detected!'
);
});
it('should log a warning for < iframe > tags', () => {
expect(fs.sanitize('Foo<iframe><body><h1>fake</h1></body></iframe>Bar'))
.toEqual('FooBar');
expect(logServiceSpy.warn).toHaveBeenCalledWith(
'Unsanitary HTML input -- <iframe> detected!'
);
});
it('should completely strip < script >, remove < a >, retain < i >', () => {
expect(fs.sanitize('Hey <i>this</i> is <script>alert("foo");</script> <a href="meh">cool</a>'))
.toEqual('Hey <i>this</i> is cool');
});
});