235 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			235 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
| /*
 | |
| Copyright 2022 The Matrix.org Foundation C.I.C.
 | |
| 
 | |
| 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 { logger } from "matrix-js-sdk/src/logger";
 | |
| 
 | |
| import {
 | |
|     GeolocationError,
 | |
|     getGeoUri,
 | |
|     mapGeolocationError,
 | |
|     mapGeolocationPositionToTimedGeo,
 | |
|     watchPosition,
 | |
| } from "../../../src/utils/beacon";
 | |
| import { getCurrentPosition } from "../../../src/utils/beacon/geolocation";
 | |
| import {
 | |
|     makeGeolocationPosition,
 | |
|     mockGeolocation,
 | |
|     getMockGeolocationPositionError,
 | |
| } from "../../test-utils";
 | |
| 
 | |
| describe('geolocation utilities', () => {
 | |
|     let geolocation;
 | |
|     const defaultPosition = makeGeolocationPosition({});
 | |
| 
 | |
|     beforeEach(() => {
 | |
|         geolocation = mockGeolocation();
 | |
|     });
 | |
| 
 | |
|     afterEach(() => {
 | |
|         jest.spyOn(logger, 'error').mockRestore();
 | |
|     });
 | |
| 
 | |
|     describe('getGeoUri', () => {
 | |
|         it("Renders a URI with only lat and lon", () => {
 | |
|             const pos = {
 | |
|                 latitude: 43.2,
 | |
|                 longitude: 12.4,
 | |
|                 altitude: undefined,
 | |
|                 accuracy: undefined,
 | |
| 
 | |
|                 timestamp: 12334,
 | |
|             };
 | |
|             expect(getGeoUri(pos)).toEqual("geo:43.2,12.4");
 | |
|         });
 | |
| 
 | |
|         it("Nulls in location are not shown in URI", () => {
 | |
|             const pos = {
 | |
|                 latitude: 43.2,
 | |
|                 longitude: 12.4,
 | |
|                 altitude: null,
 | |
|                 accuracy: null,
 | |
| 
 | |
|                 timestamp: 12334,
 | |
|             };
 | |
|             expect(getGeoUri(pos)).toEqual("geo:43.2,12.4");
 | |
|         });
 | |
| 
 | |
|         it("Renders a URI with 3 coords", () => {
 | |
|             const pos = {
 | |
|                 latitude: 43.2,
 | |
|                 longitude: 12.4,
 | |
|                 altitude: 332.54,
 | |
|                 accuracy: undefined,
 | |
|                 timestamp: 12334,
 | |
|             };
 | |
|             expect(getGeoUri(pos)).toEqual("geo:43.2,12.4,332.54");
 | |
|         });
 | |
| 
 | |
|         it("Renders a URI with accuracy", () => {
 | |
|             const pos = {
 | |
|                 latitude: 43.2,
 | |
|                 longitude: 12.4,
 | |
|                 altitude: undefined,
 | |
|                 accuracy: 21,
 | |
|                 timestamp: 12334,
 | |
|             };
 | |
|             expect(getGeoUri(pos)).toEqual("geo:43.2,12.4;u=21");
 | |
|         });
 | |
| 
 | |
|         it("Renders a URI with accuracy and altitude", () => {
 | |
|             const pos = {
 | |
|                 latitude: 43.2,
 | |
|                 longitude: 12.4,
 | |
|                 altitude: 12.3,
 | |
|                 accuracy: 21,
 | |
|                 timestamp: 12334,
 | |
|             };
 | |
|             expect(getGeoUri(pos)).toEqual("geo:43.2,12.4,12.3;u=21");
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe('mapGeolocationError', () => {
 | |
|         beforeEach(() => {
 | |
|             // suppress expected errors from test log
 | |
|             jest.spyOn(logger, 'error').mockImplementation(() => { });
 | |
|         });
 | |
| 
 | |
|         it('returns default for other error', () => {
 | |
|             const error = new Error('oh no..');
 | |
|             expect(mapGeolocationError(error)).toEqual(GeolocationError.Default);
 | |
|         });
 | |
| 
 | |
|         it('returns unavailable for unavailable error', () => {
 | |
|             const error = new Error(GeolocationError.Unavailable);
 | |
|             expect(mapGeolocationError(error)).toEqual(GeolocationError.Unavailable);
 | |
|         });
 | |
| 
 | |
|         it('maps geo error permissiondenied correctly', () => {
 | |
|             const error = getMockGeolocationPositionError(1, 'message');
 | |
|             expect(mapGeolocationError(error)).toEqual(GeolocationError.PermissionDenied);
 | |
|         });
 | |
| 
 | |
|         it('maps geo position unavailable error correctly', () => {
 | |
|             const error = getMockGeolocationPositionError(2, 'message');
 | |
|             expect(mapGeolocationError(error)).toEqual(GeolocationError.PositionUnavailable);
 | |
|         });
 | |
| 
 | |
|         it('maps geo timeout error correctly', () => {
 | |
|             const error = getMockGeolocationPositionError(3, 'message');
 | |
|             expect(mapGeolocationError(error)).toEqual(GeolocationError.Timeout);
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe('mapGeolocationPositionToTimedGeo()', () => {
 | |
|         it('maps geolocation position correctly', () => {
 | |
|             expect(mapGeolocationPositionToTimedGeo(defaultPosition)).toEqual({
 | |
|                 timestamp: 1647256791840, geoUri: 'geo:54.001927,-8.253491;u=1',
 | |
|             });
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe('watchPosition()', () => {
 | |
|         it('throws with unavailable error when geolocation is not available', () => {
 | |
|             // suppress expected errors from test log
 | |
|             jest.spyOn(logger, 'error').mockImplementation(() => { });
 | |
| 
 | |
|             // remove the mock we added
 | |
|             // @ts-ignore illegal assignment to readonly property
 | |
|             navigator.geolocation = undefined;
 | |
| 
 | |
|             const positionHandler = jest.fn();
 | |
|             const errorHandler = jest.fn();
 | |
| 
 | |
|             expect(() => watchPosition(positionHandler, errorHandler)).toThrow(GeolocationError.Unavailable);
 | |
|         });
 | |
| 
 | |
|         it('sets up position handler with correct options', () => {
 | |
|             const positionHandler = jest.fn();
 | |
|             const errorHandler = jest.fn();
 | |
|             watchPosition(positionHandler, errorHandler);
 | |
| 
 | |
|             const [, , options] = geolocation.watchPosition.mock.calls[0];
 | |
|             expect(options).toEqual({
 | |
|                 maximumAge: 2000,
 | |
|                 timeout: 5000,
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         it('returns clearWatch function', () => {
 | |
|             const watchId = 1;
 | |
|             geolocation.watchPosition.mockReturnValue(watchId);
 | |
|             const positionHandler = jest.fn();
 | |
|             const errorHandler = jest.fn();
 | |
|             const clearWatch = watchPosition(positionHandler, errorHandler);
 | |
| 
 | |
|             clearWatch();
 | |
| 
 | |
|             expect(geolocation.clearWatch).toHaveBeenCalledWith(watchId);
 | |
|         });
 | |
| 
 | |
|         it('calls position handler with position', () => {
 | |
|             const positionHandler = jest.fn();
 | |
|             const errorHandler = jest.fn();
 | |
|             watchPosition(positionHandler, errorHandler);
 | |
| 
 | |
|             expect(positionHandler).toHaveBeenCalledWith(defaultPosition);
 | |
|         });
 | |
| 
 | |
|         it('maps geolocation position error and calls error handler', () => {
 | |
|             // suppress expected errors from test log
 | |
|             jest.spyOn(logger, 'error').mockImplementation(() => { });
 | |
|             geolocation.watchPosition.mockImplementation(
 | |
|                 (_callback, error) => error(getMockGeolocationPositionError(1, 'message')),
 | |
|             );
 | |
|             const positionHandler = jest.fn();
 | |
|             const errorHandler = jest.fn();
 | |
|             watchPosition(positionHandler, errorHandler);
 | |
| 
 | |
|             expect(errorHandler).toHaveBeenCalledWith(GeolocationError.PermissionDenied);
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     describe('getCurrentPosition()', () => {
 | |
|         it('throws with unavailable error when geolocation is not available', async () => {
 | |
|             // suppress expected errors from test log
 | |
|             jest.spyOn(logger, 'error').mockImplementation(() => { });
 | |
| 
 | |
|             // remove the mock we added
 | |
|             // @ts-ignore illegal assignment to readonly property
 | |
|             navigator.geolocation = undefined;
 | |
| 
 | |
|             await expect(() => getCurrentPosition()).rejects.toThrow(GeolocationError.Unavailable);
 | |
|         });
 | |
| 
 | |
|         it('throws with geolocation error when geolocation.getCurrentPosition fails', async () => {
 | |
|             // suppress expected errors from test log
 | |
|             jest.spyOn(logger, 'error').mockImplementation(() => { });
 | |
| 
 | |
|             const timeoutError = getMockGeolocationPositionError(3, 'message');
 | |
|             geolocation.getCurrentPosition.mockImplementation((callback, error) => error(timeoutError));
 | |
| 
 | |
|             await expect(() => getCurrentPosition()).rejects.toThrow(GeolocationError.Timeout);
 | |
|         });
 | |
| 
 | |
|         it('resolves with current location', async () => {
 | |
|             geolocation.getCurrentPosition.mockImplementation((callback, error) => callback(defaultPosition));
 | |
| 
 | |
|             const result = await getCurrentPosition();
 | |
|             expect(result).toEqual(defaultPosition);
 | |
|         });
 | |
|     });
 | |
| });
 |