@smarterplan/ngx-smarterplan-core 1.2.45 → 1.2.47
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -24
- package/esm2020/lib/components/csv-export/csv-export.component.mjs +59 -59
- package/esm2020/lib/components/loader/loader.component.mjs +23 -23
- package/esm2020/lib/components/menu-bar/avatar/avatar.component.mjs +80 -80
- package/esm2020/lib/components/menu-bar/menu-bar.component.mjs +99 -99
- package/esm2020/lib/components/menu-bar/navigation-bar/navigation-bar.component.mjs +384 -384
- package/esm2020/lib/components/menu-bar/range-date-picker/range-date-picker.component.mjs +147 -147
- package/esm2020/lib/components/modal-switch-visit/modal-switch-visit.component.mjs +40 -40
- package/esm2020/lib/components/search-bar/search-bar.component.mjs +63 -63
- package/esm2020/lib/components/support-modal/support-modal.component.mjs +66 -66
- package/esm2020/lib/config.mjs +4 -4
- package/esm2020/lib/helpers.service.mjs +470 -470
- package/esm2020/lib/matterport-extensions/hsl-loader/HlsLoader.mjs +69 -69
- package/esm2020/lib/matterport-extensions/nest-thermostat/CanvasImage.mjs +51 -51
- package/esm2020/lib/matterport-extensions/nest-thermostat/CanvasRenderer.mjs +61 -61
- package/esm2020/lib/matterport-extensions/nest-thermostat/NestThermostat.mjs +158 -158
- package/esm2020/lib/matterport-extensions/nest-thermostat/PlaneRenderer.mjs +85 -85
- package/esm2020/lib/matterport-extensions/scene-component/SceneComponent.mjs +128 -128
- package/esm2020/lib/matterport-extensions/security-camera/SecurityCamera.mjs +249 -249
- package/esm2020/lib/matterport-extensions/tv-player/TvPlayer.mjs +98 -98
- package/esm2020/lib/matterport-extensions/video-renderer/VideoRenderer.mjs +64 -64
- package/esm2020/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.mjs +221 -221
- package/esm2020/lib/mattertagData.mjs +165 -165
- package/esm2020/lib/ngx-smarterplan-core.module.mjs +122 -122
- package/esm2020/lib/ngx-smarterplan-core.service.mjs +14 -14
- package/esm2020/lib/pipes/duration-to-string.pipe.mjs +66 -66
- package/esm2020/lib/pipes/format-date-number-to-digits.pipe.mjs +30 -30
- package/esm2020/lib/pipes/hashtag-from-id.pipe.mjs +26 -26
- package/esm2020/lib/pipes/safe-url.pipe.mjs +20 -20
- package/esm2020/lib/pipes/time-date-to-local-string.pipe.mjs +104 -104
- package/esm2020/lib/pipes/username-from-id.pipe.mjs +29 -29
- package/esm2020/lib/services/amplify-cache.service.mjs +72 -72
- package/esm2020/lib/services/base-tab.service.mjs +24 -24
- package/esm2020/lib/services/baseVisibility.service.mjs +18 -18
- package/esm2020/lib/services/content.service.mjs +135 -135
- package/esm2020/lib/services/filter.service.mjs +599 -599
- package/esm2020/lib/services/intervention.service.mjs +236 -236
- package/esm2020/lib/services/locale.service.mjs +45 -45
- package/esm2020/lib/services/matterport-import.service.mjs +340 -340
- package/esm2020/lib/services/matterport.service.mjs +1587 -1587
- package/esm2020/lib/services/models/affectation.service.mjs +60 -60
- package/esm2020/lib/services/models/base-object.service.mjs +70 -70
- package/esm2020/lib/services/models/capture.service.mjs +34 -34
- package/esm2020/lib/services/models/comment.service.mjs +98 -98
- package/esm2020/lib/services/models/domain.service.mjs +78 -78
- package/esm2020/lib/services/models/equipment.service.mjs +683 -683
- package/esm2020/lib/services/models/event.service.mjs +128 -128
- package/esm2020/lib/services/models/feature.service.mjs +380 -380
- package/esm2020/lib/services/models/hashtag.service.mjs +38 -38
- package/esm2020/lib/services/models/layer.service.mjs +33 -33
- package/esm2020/lib/services/models/measurement.service.mjs +199 -199
- package/esm2020/lib/services/models/mission.service.mjs +206 -206
- package/esm2020/lib/services/models/navigation.service.mjs +92 -92
- package/esm2020/lib/services/models/node.service.mjs +31 -31
- package/esm2020/lib/services/models/object3D.service.mjs +364 -364
- package/esm2020/lib/services/models/operation.service.mjs +59 -59
- package/esm2020/lib/services/models/organisation.service.mjs +73 -73
- package/esm2020/lib/services/models/plan.service.mjs +799 -799
- package/esm2020/lib/services/models/poi.service.mjs +103 -103
- package/esm2020/lib/services/models/profile.service.mjs +58 -58
- package/esm2020/lib/services/models/property.service.mjs +44 -44
- package/esm2020/lib/services/models/space.service.mjs +204 -204
- package/esm2020/lib/services/models/template.service.mjs +41 -41
- package/esm2020/lib/services/models/ticket.service.mjs +526 -526
- package/esm2020/lib/services/models/visit.service.mjs +130 -130
- package/esm2020/lib/services/models/zone.service.mjs +225 -225
- package/esm2020/lib/services/navigator.service.mjs +212 -212
- package/esm2020/lib/services/s3.service.mjs +137 -137
- package/esm2020/lib/services/search.service.mjs +124 -124
- package/esm2020/lib/services/support.service.mjs +42 -42
- package/esm2020/lib/services/tag.service.mjs +111 -111
- package/esm2020/lib/services/user.service.mjs +501 -501
- package/esm2020/lib/services/validators.service.mjs +50 -50
- package/esm2020/lib/services/viewer.service.mjs +389 -389
- package/esm2020/lib/services/zone-drawer.service.mjs +76 -76
- package/esm2020/lib/services/zoneChange.service.mjs +30 -30
- package/esm2020/lib/types.service.mjs +311 -311
- package/esm2020/lib/validators/email.directive.mjs +7 -7
- package/esm2020/lib/validators/no-empty.directive.mjs +12 -12
- package/esm2020/lib/validators/number.directive.mjs +12 -12
- package/esm2020/lib/validators/text.directive.mjs +12 -12
- package/esm2020/public-api.mjs +72 -72
- package/esm2020/smarterplan-ngx-smarterplan-core.mjs +4 -4
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs +13014 -13014
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs +12263 -12263
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/lib/components/csv-export/csv-export.component.d.ts +18 -18
- package/lib/components/loader/loader.component.d.ts +10 -10
- package/lib/components/menu-bar/avatar/avatar.component.d.ts +21 -21
- package/lib/components/menu-bar/menu-bar.component.d.ts +38 -38
- package/lib/components/menu-bar/navigation-bar/navigation-bar.component.d.ts +73 -73
- package/lib/components/menu-bar/range-date-picker/range-date-picker.component.d.ts +35 -35
- package/lib/components/modal-switch-visit/modal-switch-visit.component.d.ts +22 -22
- package/lib/components/search-bar/search-bar.component.d.ts +16 -16
- package/lib/components/support-modal/support-modal.component.d.ts +26 -26
- package/lib/config.d.ts +22 -22
- package/lib/helpers.service.d.ts +79 -79
- package/lib/matterport-extensions/hsl-loader/HlsLoader.d.ts +26 -26
- package/lib/matterport-extensions/nest-thermostat/CanvasImage.d.ts +31 -31
- package/lib/matterport-extensions/nest-thermostat/CanvasRenderer.d.ts +37 -37
- package/lib/matterport-extensions/nest-thermostat/NestThermostat.d.ts +42 -42
- package/lib/matterport-extensions/nest-thermostat/PlaneRenderer.d.ts +46 -46
- package/lib/matterport-extensions/scene-component/SceneComponent.d.ts +388 -388
- package/lib/matterport-extensions/security-camera/SecurityCamera.d.ts +47 -47
- package/lib/matterport-extensions/tv-player/TvPlayer.d.ts +26 -26
- package/lib/matterport-extensions/video-renderer/VideoRenderer.d.ts +26 -26
- package/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.d.ts +43 -43
- package/lib/mattertagData.d.ts +70 -70
- package/lib/ngx-smarterplan-core.module.d.ts +29 -29
- package/lib/ngx-smarterplan-core.service.d.ts +6 -6
- package/lib/pipes/duration-to-string.pipe.d.ts +12 -12
- package/lib/pipes/format-date-number-to-digits.pipe.d.ts +10 -10
- package/lib/pipes/hashtag-from-id.pipe.d.ts +10 -10
- package/lib/pipes/safe-url.pipe.d.ts +10 -10
- package/lib/pipes/time-date-to-local-string.pipe.d.ts +16 -16
- package/lib/pipes/username-from-id.pipe.d.ts +11 -11
- package/lib/services/amplify-cache.service.d.ts +37 -37
- package/lib/services/base-tab.service.d.ts +10 -10
- package/lib/services/baseVisibility.service.d.ts +9 -9
- package/lib/services/content.service.d.ts +28 -28
- package/lib/services/filter.service.d.ts +60 -60
- package/lib/services/intervention.service.d.ts +25 -25
- package/lib/services/locale.service.d.ts +23 -23
- package/lib/services/matterport-import.service.d.ts +53 -53
- package/lib/services/matterport.service.d.ts +336 -336
- package/lib/services/models/affectation.service.d.ts +14 -14
- package/lib/services/models/base-object.service.d.ts +20 -20
- package/lib/services/models/capture.service.d.ts +13 -13
- package/lib/services/models/comment.service.d.ts +26 -26
- package/lib/services/models/domain.service.d.ts +19 -19
- package/lib/services/models/equipment.service.d.ts +93 -93
- package/lib/services/models/event.service.d.ts +43 -43
- package/lib/services/models/feature.service.d.ts +75 -75
- package/lib/services/models/hashtag.service.d.ts +13 -13
- package/lib/services/models/layer.service.d.ts +11 -11
- package/lib/services/models/measurement.service.d.ts +51 -51
- package/lib/services/models/mission.service.d.ts +39 -39
- package/lib/services/models/navigation.service.d.ts +29 -29
- package/lib/services/models/node.service.d.ts +12 -12
- package/lib/services/models/object3D.service.d.ts +57 -57
- package/lib/services/models/operation.service.d.ts +15 -15
- package/lib/services/models/organisation.service.d.ts +19 -19
- package/lib/services/models/plan.service.d.ts +133 -133
- package/lib/services/models/poi.service.d.ts +25 -25
- package/lib/services/models/profile.service.d.ts +16 -16
- package/lib/services/models/property.service.d.ts +13 -13
- package/lib/services/models/space.service.d.ts +46 -46
- package/lib/services/models/template.service.d.ts +15 -15
- package/lib/services/models/ticket.service.d.ts +93 -93
- package/lib/services/models/visit.service.d.ts +24 -24
- package/lib/services/models/zone.service.d.ts +50 -50
- package/lib/services/navigator.service.d.ts +61 -61
- package/lib/services/s3.service.d.ts +14 -14
- package/lib/services/search.service.d.ts +20 -20
- package/lib/services/support.service.d.ts +17 -17
- package/lib/services/tag.service.d.ts +29 -29
- package/lib/services/user.service.d.ts +118 -118
- package/lib/services/validators.service.d.ts +18 -18
- package/lib/services/viewer.service.d.ts +110 -110
- package/lib/services/zone-drawer.service.d.ts +7 -7
- package/lib/services/zoneChange.service.d.ts +17 -17
- package/lib/types.service.d.ts +842 -842
- package/lib/validators/email.directive.d.ts +2 -2
- package/lib/validators/no-empty.directive.d.ts +2 -2
- package/lib/validators/number.directive.d.ts +2 -2
- package/lib/validators/text.directive.d.ts +2 -2
- package/package.json +2 -2
- package/public-api.d.ts +64 -64
- package/smarterplan-ngx-smarterplan-core.d.ts +5 -5
|
@@ -1,501 +1,501 @@
|
|
|
1
|
-
import { Inject, Injectable } from "@angular/core";
|
|
2
|
-
import { Subject } from "rxjs";
|
|
3
|
-
import { LevelStatus, ProfileEntity, ProfileStatus, PropertyType, RoleStatus } from "../types.service";
|
|
4
|
-
import { enumToArray, filterUniqueArrayByID, getHighestLevelForMissions, getHighestRoleForMissions, getLevelsBelow, getRolesBelowForManager } from "../helpers.service";
|
|
5
|
-
import { DateTime } from "luxon";
|
|
6
|
-
import { Auth } from "aws-amplify";
|
|
7
|
-
import * as i0 from "@angular/core";
|
|
8
|
-
import * as i1 from "./models/profile.service";
|
|
9
|
-
import * as i2 from "./models/mission.service";
|
|
10
|
-
import * as i3 from "./models/property.service";
|
|
11
|
-
import * as i4 from "@angular/router";
|
|
12
|
-
import * as i5 from "@angular/common";
|
|
13
|
-
import * as i6 from "@ngx-translate/core";
|
|
14
|
-
import * as i7 from "../types.service";
|
|
15
|
-
/**
|
|
16
|
-
* UserService handles user's lifecycle, from creation to deletion
|
|
17
|
-
*/
|
|
18
|
-
export class BaseUserService {
|
|
19
|
-
constructor(profileService, missionService, propertyService, router, location, ngZone, translate, spModuleInjected) {
|
|
20
|
-
this.profileService = profileService;
|
|
21
|
-
this.missionService = missionService;
|
|
22
|
-
this.propertyService = propertyService;
|
|
23
|
-
this.router = router;
|
|
24
|
-
this.location = location;
|
|
25
|
-
this.ngZone = ngZone;
|
|
26
|
-
this.translate = translate;
|
|
27
|
-
this.isChanged = new Subject();
|
|
28
|
-
this.updating = false;
|
|
29
|
-
this.managers = [
|
|
30
|
-
RoleStatus.ADMIN,
|
|
31
|
-
RoleStatus.MANAGER,
|
|
32
|
-
RoleStatus.OWNER,
|
|
33
|
-
RoleStatus.GUIDE_MUSEUM,
|
|
34
|
-
];
|
|
35
|
-
this.cu = null;
|
|
36
|
-
// this.checkConnectedAWSUser();
|
|
37
|
-
this.missionService.currentMissionsUpdated.subscribe(async (data) => {
|
|
38
|
-
this.currentMissions =
|
|
39
|
-
await this.missionService.instantMissionsOfCurrent(this.cu.id);
|
|
40
|
-
this.isChanged.next(true);
|
|
41
|
-
});
|
|
42
|
-
this.spModule = spModuleInjected;
|
|
43
|
-
}
|
|
44
|
-
/**
|
|
45
|
-
* Checks if AWS user is connected and assigng it to current user (this.cu)
|
|
46
|
-
**/
|
|
47
|
-
async checkConnectedAWSUser() {
|
|
48
|
-
if (!this.cu) {
|
|
49
|
-
// console.log("check AWS user");
|
|
50
|
-
let cau = null;
|
|
51
|
-
let user = null;
|
|
52
|
-
try {
|
|
53
|
-
if (!this.updating) {
|
|
54
|
-
cau = await Auth.currentAuthenticatedUser();
|
|
55
|
-
// console.log("connected", cau);
|
|
56
|
-
user = await this.current(cau);
|
|
57
|
-
// console.log("current of userService", user);
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
catch { }
|
|
61
|
-
if (!user && !this.updating) {
|
|
62
|
-
// console.log("no user in check connected aws user");
|
|
63
|
-
this.router.navigate(["auth"]);
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
/**
|
|
68
|
-
* createUser spawns a new User, along with:
|
|
69
|
-
* - a proper AWSCognito signup
|
|
70
|
-
* - a Profile entry holding the AWSCognito ID
|
|
71
|
-
* - a Mission entry holding the Organisation ID
|
|
72
|
-
*
|
|
73
|
-
* @see 174
|
|
74
|
-
* @see https://docs.amplify.aws/lib/auth/emailpassword/q/platform/js#sign-up
|
|
75
|
-
* @todo wonder if we can mash together Profile, Mission and User entries
|
|
76
|
-
* @param u User to be created
|
|
77
|
-
*/
|
|
78
|
-
async createUserAWSandProfile(u, prevalidated = false) {
|
|
79
|
-
const attributes = {
|
|
80
|
-
email: u.email.trim(),
|
|
81
|
-
given_name: u.firstName,
|
|
82
|
-
family_name: u.lastName,
|
|
83
|
-
zoneinfo: "fr",
|
|
84
|
-
locale: "fr",
|
|
85
|
-
"custom:prevalidated": "false",
|
|
86
|
-
};
|
|
87
|
-
if (prevalidated) {
|
|
88
|
-
attributes["custom:prevalidated"] = "true";
|
|
89
|
-
}
|
|
90
|
-
// AWSCognito
|
|
91
|
-
const password = u.password ? u.password : "temporaryp";
|
|
92
|
-
let error;
|
|
93
|
-
try {
|
|
94
|
-
const { userSub } = await Auth.signUp({
|
|
95
|
-
username: u.email,
|
|
96
|
-
password,
|
|
97
|
-
attributes,
|
|
98
|
-
clientMetadata: {
|
|
99
|
-
phone: u.phone,
|
|
100
|
-
},
|
|
101
|
-
});
|
|
102
|
-
if (userSub) {
|
|
103
|
-
return this.createProfile(userSub, u);
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
catch (error_) {
|
|
107
|
-
switch (error_.code) {
|
|
108
|
-
case "InvalidPasswordException":
|
|
109
|
-
error = "auth.error.wrong_password";
|
|
110
|
-
break;
|
|
111
|
-
case "UserNotConfirmedException":
|
|
112
|
-
error = "auth.error.user_not_confirmed";
|
|
113
|
-
break;
|
|
114
|
-
case "InvalidParameterException":
|
|
115
|
-
error = "auth.error.invalid_parameter";
|
|
116
|
-
break;
|
|
117
|
-
default:
|
|
118
|
-
error = error_.message;
|
|
119
|
-
}
|
|
120
|
-
console.log(`AWS Sign UP error : ${error}`);
|
|
121
|
-
}
|
|
122
|
-
return Promise.reject(error);
|
|
123
|
-
}
|
|
124
|
-
/**
|
|
125
|
-
* - Creates AWS user in Cognito for Hotel User Pool Schema (only email is required)
|
|
126
|
-
* - Creates Profile in DynamoDB
|
|
127
|
-
*
|
|
128
|
-
* @param u User to be created
|
|
129
|
-
*/
|
|
130
|
-
async createHotelUserAWSandProfile(user) {
|
|
131
|
-
const attributes = {
|
|
132
|
-
email: user.email.trim()
|
|
133
|
-
};
|
|
134
|
-
// AWSCognito
|
|
135
|
-
const password = "temporaryp";
|
|
136
|
-
let error;
|
|
137
|
-
try {
|
|
138
|
-
const { userSub } = await Auth.signUp({
|
|
139
|
-
username: user.email.trim(),
|
|
140
|
-
password,
|
|
141
|
-
attributes,
|
|
142
|
-
});
|
|
143
|
-
if (userSub) {
|
|
144
|
-
return this.createProfile(userSub, user);
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
catch (error_) {
|
|
148
|
-
switch (error_.code) {
|
|
149
|
-
case "InvalidPasswordException":
|
|
150
|
-
error = "auth.error.wrong_password";
|
|
151
|
-
break;
|
|
152
|
-
case "UserNotConfirmedException":
|
|
153
|
-
error = "auth.error.user_not_confirmed";
|
|
154
|
-
break;
|
|
155
|
-
case "InvalidParameterException":
|
|
156
|
-
error = "auth.error.invalid_parameter";
|
|
157
|
-
break;
|
|
158
|
-
default:
|
|
159
|
-
error = error_.message;
|
|
160
|
-
}
|
|
161
|
-
console.log(`AWS Sign UP error : ${error}`);
|
|
162
|
-
}
|
|
163
|
-
return Promise.reject(error);
|
|
164
|
-
}
|
|
165
|
-
async createProfile(awsUserID, u) {
|
|
166
|
-
const profile = await this.profileService.createProfile({
|
|
167
|
-
userID: awsUserID,
|
|
168
|
-
metadata: JSON.stringify({
|
|
169
|
-
email: u.email,
|
|
170
|
-
phone: u.phone,
|
|
171
|
-
lastName: u.lastName,
|
|
172
|
-
firstName: u.firstName,
|
|
173
|
-
}),
|
|
174
|
-
status: ProfileStatus.WAITING_FOR_ACTIVATION,
|
|
175
|
-
});
|
|
176
|
-
return profile;
|
|
177
|
-
}
|
|
178
|
-
currentOrganisation(spaceID = null) {
|
|
179
|
-
if (this.cu) {
|
|
180
|
-
if (!spaceID) {
|
|
181
|
-
return this.currentMissions[0].organisation;
|
|
182
|
-
}
|
|
183
|
-
return this.currentMission(spaceID).organisation;
|
|
184
|
-
}
|
|
185
|
-
return null;
|
|
186
|
-
}
|
|
187
|
-
async current(user, fastConnect = false) {
|
|
188
|
-
if (!this.cu && !this.updating) {
|
|
189
|
-
this.updating = true;
|
|
190
|
-
this.cu = await this.userFromAWSID(user.username);
|
|
191
|
-
if (!this.cu && user.attributes.email) {
|
|
192
|
-
console.log(`No profile for userID 🔪 ${user.attributes.email}`);
|
|
193
|
-
// There's no profiles, profiles.items.length === 0
|
|
194
|
-
// It's maybe the 🤮 case where the profile has been saved with an email
|
|
195
|
-
// instead of an hashed id
|
|
196
|
-
// If found, we need to replace the id in both the Profile
|
|
197
|
-
const emailProfile = await this.userFromAWSID(user.attributes.email);
|
|
198
|
-
if (emailProfile) {
|
|
199
|
-
console.log(`👏 doing the spacehip reversal email on ${user.attributes.email}`);
|
|
200
|
-
this.cu = await this.userFromAWSID(user.username);
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
if (!this.cu) {
|
|
204
|
-
this.updating = false;
|
|
205
|
-
await Auth.signOut();
|
|
206
|
-
return null;
|
|
207
|
-
}
|
|
208
|
-
const missions = await this.missionService.instantMissionsOfCurrent(this.cu.id);
|
|
209
|
-
this.currentMissions = missions;
|
|
210
|
-
let okToLogin = true;
|
|
211
|
-
if (!fastConnect && !this.hasManagerRoleInAnyMission()) {
|
|
212
|
-
// check access token if not Fast connect (fast connect is for the guest connection via link(/redirect?user=))
|
|
213
|
-
// checks visitors(museum or not), occupants, service
|
|
214
|
-
okToLogin = await this.checkAccessToken(user.signInUserSession.accessToken.jwtToken);
|
|
215
|
-
}
|
|
216
|
-
if (!okToLogin) {
|
|
217
|
-
this.cu = null;
|
|
218
|
-
this.currentMissions = [];
|
|
219
|
-
this.updating = false;
|
|
220
|
-
await Auth.signOut();
|
|
221
|
-
return null;
|
|
222
|
-
}
|
|
223
|
-
await this.profileService.updateProfile({
|
|
224
|
-
id: this.cu.id,
|
|
225
|
-
lastConnectedAt: Date.now(),
|
|
226
|
-
});
|
|
227
|
-
if (!missions && this.location.path() !== "/no-mission") {
|
|
228
|
-
this.ngZone.run(() => {
|
|
229
|
-
this.router.navigate(["/no-mission"]);
|
|
230
|
-
});
|
|
231
|
-
}
|
|
232
|
-
// console.log("user is changed emitting");
|
|
233
|
-
this.isChanged.next(true);
|
|
234
|
-
this.updating = false;
|
|
235
|
-
}
|
|
236
|
-
return this.cu;
|
|
237
|
-
}
|
|
238
|
-
async userFromAWSID(userId) {
|
|
239
|
-
const profile = await this.profileService.getProfileFromAWSID(userId);
|
|
240
|
-
if (profile) {
|
|
241
|
-
return new ProfileEntity(profile);
|
|
242
|
-
}
|
|
243
|
-
return null;
|
|
244
|
-
}
|
|
245
|
-
async userFromUserID(userID) {
|
|
246
|
-
const profile = await this.profileService.getProfile(userID);
|
|
247
|
-
return new ProfileEntity(profile);
|
|
248
|
-
}
|
|
249
|
-
currentMission(spaceID) {
|
|
250
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
251
|
-
return mission;
|
|
252
|
-
}
|
|
253
|
-
isManager(spaceID) {
|
|
254
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
255
|
-
if (mission) {
|
|
256
|
-
return this.managers.includes(mission.role);
|
|
257
|
-
}
|
|
258
|
-
return false;
|
|
259
|
-
}
|
|
260
|
-
isService(spaceID) {
|
|
261
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
262
|
-
if (mission) {
|
|
263
|
-
return mission.role === RoleStatus.SERVICE;
|
|
264
|
-
}
|
|
265
|
-
return false;
|
|
266
|
-
}
|
|
267
|
-
isAdmin(spaceID) {
|
|
268
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
269
|
-
if (mission) {
|
|
270
|
-
return mission.role === RoleStatus.ADMIN;
|
|
271
|
-
}
|
|
272
|
-
return false;
|
|
273
|
-
}
|
|
274
|
-
isSmarterPlan() {
|
|
275
|
-
if (!this.currentMissions) {
|
|
276
|
-
return false;
|
|
277
|
-
}
|
|
278
|
-
const spMissions = this.currentMissions.filter((m) => m.organisationID === "6e28201f-4679-4c29-9d2d-8a325f57cee9" &&
|
|
279
|
-
this.managers.includes(m.role)); // ID of SmarterPlan
|
|
280
|
-
return spMissions.length > 0;
|
|
281
|
-
}
|
|
282
|
-
isSPAdmin() {
|
|
283
|
-
if (!this.currentMissions) {
|
|
284
|
-
return false;
|
|
285
|
-
}
|
|
286
|
-
const spMissions = this.currentMissions.filter((m) => m.organisationID === "6e28201f-4679-4c29-9d2d-8a325f57cee9" &&
|
|
287
|
-
m.role === RoleStatus.ADMIN); // ID of SmarterPlan
|
|
288
|
-
return spMissions.length > 0;
|
|
289
|
-
}
|
|
290
|
-
getMissionForSpace(spaceID) {
|
|
291
|
-
if (!this.currentMissions) {
|
|
292
|
-
return null;
|
|
293
|
-
}
|
|
294
|
-
return this.currentMissions.find((m) => m.spaceID === spaceID);
|
|
295
|
-
}
|
|
296
|
-
hasManagerRoleInAnyMission() {
|
|
297
|
-
if (!this.currentMissions) {
|
|
298
|
-
return false;
|
|
299
|
-
}
|
|
300
|
-
return this.currentMissions.some((mis) => this.managers.includes(mis.role));
|
|
301
|
-
}
|
|
302
|
-
hasServiceRoleInAnyMission() {
|
|
303
|
-
if (!this.currentMissions) {
|
|
304
|
-
return false;
|
|
305
|
-
}
|
|
306
|
-
return this.currentMissions.some((mis) => mis.role === RoleStatus.SERVICE);
|
|
307
|
-
}
|
|
308
|
-
hasAdminRoleInAnyMission() {
|
|
309
|
-
if (!this.currentMissions) {
|
|
310
|
-
return false;
|
|
311
|
-
}
|
|
312
|
-
return this.currentMissions.some((mis) => mis.role === RoleStatus.ADMIN);
|
|
313
|
-
}
|
|
314
|
-
async createOrUpdateMissionForOrgWithProperty(missionInput, orgID, create = true) {
|
|
315
|
-
const mission = await (create
|
|
316
|
-
? this.missionService.create(missionInput)
|
|
317
|
-
: this.missionService.update(missionInput));
|
|
318
|
-
if (this.managers.includes(mission.role) && mission.spaceID) {
|
|
319
|
-
const properties = await this.propertyService.getPropertiesForOrganisation(orgID);
|
|
320
|
-
const propertySpaceIDs = properties.map((property_) => property_.spaceID);
|
|
321
|
-
if (propertySpaceIDs.includes(mission.spaceID)) {
|
|
322
|
-
return mission;
|
|
323
|
-
}
|
|
324
|
-
// we delegate the rights of the space
|
|
325
|
-
await this.propertyService.createProperty({
|
|
326
|
-
spaceID: mission.spaceID,
|
|
327
|
-
organisationID: orgID,
|
|
328
|
-
type: PropertyType.SHARED,
|
|
329
|
-
});
|
|
330
|
-
}
|
|
331
|
-
return mission;
|
|
332
|
-
}
|
|
333
|
-
async softDeleteUser(userID) {
|
|
334
|
-
const missions = await this.missionService.getMissionsFromUserID(userID);
|
|
335
|
-
if (missions.length === 0) {
|
|
336
|
-
await this.profileService.updateProfile({ id: userID, deletedAt: Date.now() });
|
|
337
|
-
console.log("Profile soft deleted");
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
/**
|
|
341
|
-
* Checks if the Profile (by id) can be removed completely from the DB (no missions in the DB)
|
|
342
|
-
* @param profileID
|
|
343
|
-
* @returns boolean
|
|
344
|
-
*/
|
|
345
|
-
async canHardDeleteUser(profileID) {
|
|
346
|
-
const missions = (await this.missionService.getMissionsFromUserID(profileID));
|
|
347
|
-
return missions.length === 0;
|
|
348
|
-
}
|
|
349
|
-
async hardDeleteUser(userID) {
|
|
350
|
-
const deleted = await this.profileService.deleteProfile(userID);
|
|
351
|
-
console.log("Profile hard deleted");
|
|
352
|
-
return deleted;
|
|
353
|
-
}
|
|
354
|
-
getManagerMissions() {
|
|
355
|
-
if (!this.currentMissions) {
|
|
356
|
-
return [];
|
|
357
|
-
}
|
|
358
|
-
return this.currentMissions.filter((mis) => this.managers.includes(mis.role));
|
|
359
|
-
}
|
|
360
|
-
getAvalableLevelsForCurrentUser() {
|
|
361
|
-
if (!this.currentMissions) {
|
|
362
|
-
return [LevelStatus.MINIMUM];
|
|
363
|
-
}
|
|
364
|
-
const levels = enumToArray(LevelStatus);
|
|
365
|
-
const highestLevel = getHighestLevelForMissions(this.currentMissions);
|
|
366
|
-
return levels.filter((level) => getLevelsBelow(highestLevel).includes(level));
|
|
367
|
-
}
|
|
368
|
-
getAvailableRolesForCurrentUser() {
|
|
369
|
-
if (!this.currentMissions) {
|
|
370
|
-
return [RoleStatus.VISITOR];
|
|
371
|
-
}
|
|
372
|
-
const roles = enumToArray(RoleStatus);
|
|
373
|
-
const highestRole = getHighestRoleForMissions(this.currentMissions);
|
|
374
|
-
return roles.filter((role) => getRolesBelowForManager(highestRole).includes(role));
|
|
375
|
-
}
|
|
376
|
-
/** Test if the user have the role given in parameter : "roleStatus" */
|
|
377
|
-
isRole(roleStatus) {
|
|
378
|
-
if (!this.currentMissions) {
|
|
379
|
-
return false;
|
|
380
|
-
}
|
|
381
|
-
return this.currentMissions.some((m) => m.role === roleStatus);
|
|
382
|
-
}
|
|
383
|
-
isGuide(spaceID = null) {
|
|
384
|
-
if (spaceID) {
|
|
385
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
386
|
-
if (mission) {
|
|
387
|
-
return mission.role === RoleStatus.GUIDE_MUSEUM;
|
|
388
|
-
}
|
|
389
|
-
}
|
|
390
|
-
return this.isRole(RoleStatus.GUIDE_MUSEUM);
|
|
391
|
-
}
|
|
392
|
-
isMuseumVisitor(spaceID = null) {
|
|
393
|
-
if (spaceID) {
|
|
394
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
395
|
-
if (mission) {
|
|
396
|
-
return mission.role === RoleStatus.VISITOR_MUSEUM;
|
|
397
|
-
}
|
|
398
|
-
}
|
|
399
|
-
return this.isRole(RoleStatus.VISITOR_MUSEUM);
|
|
400
|
-
}
|
|
401
|
-
isOccupant() {
|
|
402
|
-
return this.isRole(RoleStatus.OCCUPANT);
|
|
403
|
-
}
|
|
404
|
-
isVisitor(spaceID = null) {
|
|
405
|
-
if (spaceID) {
|
|
406
|
-
const mission = this.getMissionForSpace(spaceID);
|
|
407
|
-
if (mission) {
|
|
408
|
-
return mission.role === RoleStatus.VISITOR;
|
|
409
|
-
}
|
|
410
|
-
}
|
|
411
|
-
return this.isRole(RoleStatus.VISITOR);
|
|
412
|
-
}
|
|
413
|
-
isMuseumUser(spaceID = null) {
|
|
414
|
-
return this.isGuide(spaceID) || this.isMuseumVisitor(spaceID);
|
|
415
|
-
}
|
|
416
|
-
hasOnlyMuseumRole() {
|
|
417
|
-
if (this.currentMissions) {
|
|
418
|
-
return this.currentMissions.every((m) => m.role === RoleStatus.GUIDE_MUSEUM ||
|
|
419
|
-
m.role === RoleStatus.VISITOR_MUSEUM);
|
|
420
|
-
}
|
|
421
|
-
return false;
|
|
422
|
-
}
|
|
423
|
-
async profileForMail(mail) {
|
|
424
|
-
const usersWithMail = await this.profileService.getUsersForMail(mail);
|
|
425
|
-
return usersWithMail.length > 0 ? usersWithMail[0] : null;
|
|
426
|
-
}
|
|
427
|
-
currentOrderedOrgList() {
|
|
428
|
-
if (!this.currentMissions) {
|
|
429
|
-
return [];
|
|
430
|
-
}
|
|
431
|
-
const orderedOrgList = this.currentMissions.map((mis) => mis.orderedOrganisation);
|
|
432
|
-
return filterUniqueArrayByID(orderedOrgList);
|
|
433
|
-
}
|
|
434
|
-
userHasAccessToAllZonesInSpace(spaceID) {
|
|
435
|
-
const missionsForSpace = this.currentMissions.filter((mis) => mis.spaceID === spaceID);
|
|
436
|
-
return missionsForSpace.some((mission) => !mission.zoneID || (mission.zone && !mission.zone.parentID));
|
|
437
|
-
}
|
|
438
|
-
async logoutCurrentAndSignOutAuth() {
|
|
439
|
-
await this.profileService.updateProfile({
|
|
440
|
-
id: this.cu.id,
|
|
441
|
-
token: null,
|
|
442
|
-
});
|
|
443
|
-
this.cu = null;
|
|
444
|
-
this.currentMissions = null;
|
|
445
|
-
this.isChanged.next(false);
|
|
446
|
-
await Auth.signOut();
|
|
447
|
-
}
|
|
448
|
-
async logoutCurrentWihtoutAuth() {
|
|
449
|
-
await this.profileService.updateProfile({
|
|
450
|
-
id: this.cu.id,
|
|
451
|
-
token: null,
|
|
452
|
-
});
|
|
453
|
-
this.cu = null;
|
|
454
|
-
this.currentMissions = null;
|
|
455
|
-
this.isChanged.next(false);
|
|
456
|
-
}
|
|
457
|
-
async checkAccessToken(jwtToken) {
|
|
458
|
-
const now = DateTime.local();
|
|
459
|
-
if (this.cu.token && this.cu.tokenExpiredAt > now.toMillis()) {
|
|
460
|
-
// check token
|
|
461
|
-
if (this.cu.token === jwtToken) {
|
|
462
|
-
// console.log("same aws token");
|
|
463
|
-
return true;
|
|
464
|
-
}
|
|
465
|
-
alert(this.translate.instant("auth.error.activeToken"));
|
|
466
|
-
return false;
|
|
467
|
-
}
|
|
468
|
-
await this.profileService.updateProfile({
|
|
469
|
-
id: this.cu.id,
|
|
470
|
-
tokenExpiredAt: now.plus({ minutes: 5 }).toMillis(),
|
|
471
|
-
token: jwtToken,
|
|
472
|
-
status: ProfileStatus.ACTIVE,
|
|
473
|
-
});
|
|
474
|
-
return true;
|
|
475
|
-
}
|
|
476
|
-
async clearUserToken(id) {
|
|
477
|
-
await this.profileService.updateProfile({
|
|
478
|
-
id: id,
|
|
479
|
-
token: null,
|
|
480
|
-
});
|
|
481
|
-
}
|
|
482
|
-
updateCurrentUser(profile) {
|
|
483
|
-
this.cu = new ProfileEntity(profile);
|
|
484
|
-
this.isChanged.next(true);
|
|
485
|
-
}
|
|
486
|
-
getSpModule() {
|
|
487
|
-
return this.spModule;
|
|
488
|
-
}
|
|
489
|
-
}
|
|
490
|
-
BaseUserService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, deps: [{ token: i1.ProfileService }, { token: i2.MissionService }, { token: i3.PropertyService }, { token: i4.Router }, { token: i5.Location }, { token: i0.NgZone }, { token: i6.TranslateService }, { token: 'SPModule' }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
491
|
-
BaseUserService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, providedIn: "root" });
|
|
492
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, decorators: [{
|
|
493
|
-
type: Injectable,
|
|
494
|
-
args: [{
|
|
495
|
-
providedIn: "root",
|
|
496
|
-
}]
|
|
497
|
-
}], ctorParameters: function () { return [{ type: i1.ProfileService }, { type: i2.MissionService }, { type: i3.PropertyService }, { type: i4.Router }, { type: i5.Location }, { type: i0.NgZone }, { type: i6.TranslateService }, { type: i7.SpModule, decorators: [{
|
|
498
|
-
type: Inject,
|
|
499
|
-
args: ['SPModule']
|
|
500
|
-
}] }]; } });
|
|
501
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-smarterplan-core/src/lib/services/user.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, NgZone } from \"@angular/core\";\nimport { Subject } from \"rxjs\";\nimport { Router } from \"@angular/router\";\nimport { Location } from \"@angular/common\";\nimport { TranslateService } from \"@ngx-translate/core\";\nimport { LevelStatus, Mission, Organisation, Profile, ProfileEntity, ProfileStatus, PropertyType, RoleStatus, SpModule } from \"../types.service\";\nimport { enumToArray, filterUniqueArrayByID, getHighestLevelForMissions, getHighestRoleForMissions, getLevelsBelow, getRolesBelowForManager } from \"../helpers.service\";\nimport { ProfileService } from \"./models/profile.service\";\nimport { MissionService } from \"./models/mission.service\";\nimport { PropertyService } from \"./models/property.service\";\nimport { DateTime } from \"luxon\";\nimport { Auth } from \"aws-amplify\";\n\ninterface CreateUser {\n    email: string;\n    phone?: string;\n    lastName: string;\n    firstName: string;\n    password?: string;\n}\n\n/**\n * UserService handles user's lifecycle, from creation to deletion\n */\n@Injectable({\n    providedIn: \"root\",\n})\nexport class BaseUserService {\n\n    isChanged: Subject<boolean> = new Subject<boolean>();\n\n    updating = false;\n\n    managers = [\n        RoleStatus.ADMIN,\n        RoleStatus.MANAGER,\n        RoleStatus.OWNER,\n        RoleStatus.GUIDE_MUSEUM,\n    ];\n\n    currentMissions: Mission[];\n\n    lastExpiredMission: Mission;\n\n    nextMissionToStart: Mission;\n\n    cu: ProfileEntity = null;\n\n    spModule: SpModule;\n\n    constructor(\n        private profileService: ProfileService,\n        private missionService: MissionService,\n        private propertyService: PropertyService,\n        private router: Router,\n        private location: Location,\n        private ngZone: NgZone,\n        private translate: TranslateService,\n        @Inject('SPModule') spModuleInjected: SpModule,\n    ) {\n        // this.checkConnectedAWSUser();\n        this.missionService.currentMissionsUpdated.subscribe(async (data) => {\n            this.currentMissions =\n                await this.missionService.instantMissionsOfCurrent(this.cu.id);\n            this.isChanged.next(true);\n        });\n        this.spModule = spModuleInjected;\n    }\n\n    /**\n     * Checks if AWS user is connected and assigng it to current user (this.cu)\n    **/\n    async checkConnectedAWSUser() {\n        if (!this.cu) {\n            // console.log(\"check AWS user\");\n            let cau = null;\n            let user = null;\n            try {\n                if (!this.updating) {\n                    cau = await Auth.currentAuthenticatedUser();\n                    // console.log(\"connected\", cau);\n                    user = await this.current(cau);\n                    // console.log(\"current of userService\", user);\n                }\n            } catch { }\n            if (!user && !this.updating) {\n                // console.log(\"no user in check connected aws user\");\n                this.router.navigate([\"auth\"]);\n            }\n        }\n    }\n\n    /**\n     * createUser spawns a new User, along with:\n     * - a proper AWSCognito signup\n     * - a Profile entry holding the AWSCognito ID\n     * - a Mission entry holding the Organisation ID\n     *\n     * @see 174\n     * @see https://docs.amplify.aws/lib/auth/emailpassword/q/platform/js#sign-up\n     * @todo wonder if we can mash together Profile, Mission and User entries\n     * @param u User to be created\n     */\n    async createUserAWSandProfile(\n        u: {\n            email: string;\n            phone: string;\n            lastName: string;\n            firstName: string;\n            password?: string;\n        },\n        prevalidated: boolean = false,\n    ): Promise<Profile> {\n        const attributes = {\n            email: u.email.trim(),\n            given_name: u.firstName,\n            family_name: u.lastName,\n            zoneinfo: \"fr\", // @todo deal with zoneinfo and locale\n            locale: \"fr\",\n            \"custom:prevalidated\": \"false\",\n        };\n        if (prevalidated) {\n            attributes[\"custom:prevalidated\"] = \"true\";\n        }\n        // AWSCognito\n        const password = u.password ? u.password : \"temporaryp\";\n        let error;\n        try {\n            const { userSub } = await Auth.signUp({\n                username: u.email, // username should be an email they say\n                password,\n                attributes,\n                clientMetadata: {\n                    phone: u.phone,\n                },\n            });\n            if (userSub) {\n                return this.createProfile(userSub, u);\n            }\n        } catch (error_: any) {\n            switch (error_.code) {\n                case \"InvalidPasswordException\":\n                    error = \"auth.error.wrong_password\";\n                    break;\n                case \"UserNotConfirmedException\":\n                    error = \"auth.error.user_not_confirmed\";\n                    break;\n                case \"InvalidParameterException\":\n                    error = \"auth.error.invalid_parameter\";\n                    break;\n                default:\n                    error = error_.message;\n            }\n            console.log(`AWS Sign UP error : ${error}`);\n        }\n        return Promise.reject(error);\n    }\n\n    /**\n     * - Creates AWS user in Cognito for Hotel User Pool Schema (only email is required)\n     * - Creates Profile in DynamoDB\n     *\n     * @param u User to be created\n     */\n    async createHotelUserAWSandProfile(\n        user: {\n            email: string;\n            lastName: string;\n            firstName: string;\n        },\n    ): Promise<Profile> {\n        const attributes = {\n            email: user.email.trim()\n        };\n        // AWSCognito\n        const password = \"temporaryp\";\n        let error;\n        try {\n            const { userSub } = await Auth.signUp({\n                username: user.email.trim(),\n                password,\n                attributes,\n            });\n            if (userSub) {\n                return this.createProfile(userSub, user);\n            }\n        } catch (error_: any) {\n            switch (error_.code) {\n                case \"InvalidPasswordException\":\n                    error = \"auth.error.wrong_password\";\n                    break;\n                case \"UserNotConfirmedException\":\n                    error = \"auth.error.user_not_confirmed\";\n                    break;\n                case \"InvalidParameterException\":\n                    error = \"auth.error.invalid_parameter\";\n                    break;\n                default:\n                    error = error_.message;\n            }\n            console.log(`AWS Sign UP error : ${error}`);\n        }\n        return Promise.reject(error);\n    }\n\n    async createProfile(\n        awsUserID: string,\n        u: CreateUser,\n    ): Promise<Profile> {\n        const profile = await this.profileService.createProfile({\n            userID: awsUserID,\n            metadata: JSON.stringify({\n                email: u.email,\n                phone: u.phone,\n                lastName: u.lastName,\n                firstName: u.firstName,\n            }),\n            status: ProfileStatus.WAITING_FOR_ACTIVATION,\n        });\n        return profile;\n    }\n\n    currentOrganisation(spaceID: string = null): Organisation {\n        if (this.cu) {\n            if (!spaceID) {\n                return this.currentMissions[0].organisation;\n            }\n            return this.currentMission(spaceID).organisation;\n        }\n        return null;\n    }\n\n    async current(\n        user,\n        fastConnect: boolean = false,\n    ): Promise<ProfileEntity | null> {\n        if (!this.cu && !this.updating) {\n            this.updating = true;\n            this.cu = await this.userFromAWSID(user.username);\n            if (!this.cu && user.attributes.email) {\n                console.log(\n                    `No profile for userID 🔪 ${user.attributes.email}`,\n                );\n                // There's no profiles, profiles.items.length === 0\n                // It's maybe the 🤮 case where the profile has been saved with an email\n                // instead of an hashed id\n                // If found, we need to replace the id in both the Profile\n                const emailProfile = await this.userFromAWSID(\n                    user.attributes.email,\n                );\n                if (emailProfile) {\n                    console.log(\n                        `👏 doing the spacehip reversal email on ${user.attributes.email}`,\n                    );\n                    this.cu = await this.userFromAWSID(user.username);\n                }\n            }\n            if (!this.cu) {\n                this.updating = false;\n                await Auth.signOut();\n                return null;\n            }\n            const missions = await this.missionService.instantMissionsOfCurrent(\n                this.cu.id\n            );\n            this.currentMissions = missions;\n            let okToLogin = true;\n            if (!fastConnect && !this.hasManagerRoleInAnyMission()) {\n                // check access token if not Fast connect (fast connect is for the guest connection via link(/redirect?user=))\n                // checks visitors(museum or not), occupants, service\n                okToLogin = await this.checkAccessToken(\n                    user.signInUserSession.accessToken.jwtToken,\n                );\n            }\n            if (!okToLogin) {\n                this.cu = null;\n                this.currentMissions = [];\n                this.updating = false;\n                await Auth.signOut();\n                return null;\n            }\n            await this.profileService.updateProfile({\n                id: this.cu.id,\n                lastConnectedAt: Date.now(),\n            });\n            if (!missions && this.location.path() !== \"/no-mission\") {\n                this.ngZone.run(() => {\n                    this.router.navigate([\"/no-mission\"]);\n                });\n            }\n            // console.log(\"user is changed emitting\");\n            this.isChanged.next(true);\n            this.updating = false;\n        }\n        return this.cu;\n    }\n\n    async userFromAWSID(userId: string): Promise<ProfileEntity | null> {\n        const profile = await this.profileService.getProfileFromAWSID(userId);\n        if (profile) {\n            return new ProfileEntity(profile);\n        }\n        return null;\n    }\n\n    async userFromUserID(userID: string): Promise<ProfileEntity | null> {\n        const profile = await this.profileService.getProfile(userID);\n        return new ProfileEntity(profile);\n    }\n\n    currentMission(spaceID: string): Mission {\n        const mission = this.getMissionForSpace(spaceID);\n        return mission;\n    }\n\n    isManager(spaceID: string): boolean {\n        const mission = this.getMissionForSpace(spaceID);\n        if (mission) {\n            return this.managers.includes(mission.role);\n        }\n        return false;\n    }\n\n    isService(spaceID: string): boolean {\n        const mission = this.getMissionForSpace(spaceID);\n        if (mission) {\n            return mission.role === RoleStatus.SERVICE;\n        }\n        return false;\n    }\n\n    isAdmin(spaceID: string): boolean {\n        const mission = this.getMissionForSpace(spaceID);\n        if (mission) {\n            return mission.role === RoleStatus.ADMIN;\n        }\n        return false;\n    }\n\n    isSmarterPlan(): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        const spMissions = this.currentMissions.filter(\n            (m) =>\n                m.organisationID === \"6e28201f-4679-4c29-9d2d-8a325f57cee9\" &&\n                this.managers.includes(m.role),\n        ); // ID of SmarterPlan\n        return spMissions.length > 0;\n    }\n\n    isSPAdmin(): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        const spMissions = this.currentMissions.filter(\n            (m) =>\n                m.organisationID === \"6e28201f-4679-4c29-9d2d-8a325f57cee9\" &&\n                m.role === RoleStatus.ADMIN,\n        ); // ID of SmarterPlan\n        return spMissions.length > 0;\n    }\n\n    getMissionForSpace(spaceID: string): Mission {\n        if (!this.currentMissions) {\n            return null;\n        }\n        return this.currentMissions.find((m) => m.spaceID === spaceID);\n    }\n\n    hasManagerRoleInAnyMission(): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        return this.currentMissions.some((mis) =>\n            this.managers.includes(mis.role),\n        );\n    }\n\n    hasServiceRoleInAnyMission(): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        return this.currentMissions.some(\n            (mis) => mis.role === RoleStatus.SERVICE,\n        );\n    }\n\n    hasAdminRoleInAnyMission(): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        return this.currentMissions.some(\n            (mis) => mis.role === RoleStatus.ADMIN,\n        );\n    }\n\n    async createOrUpdateMissionForOrgWithProperty(\n        missionInput: Mission,\n        orgID: string,\n        create: boolean = true,\n    ): Promise<Mission> {\n        const mission = await (create\n            ? this.missionService.create(missionInput)\n            : this.missionService.update(missionInput));\n\n        if (this.managers.includes(mission.role) && mission.spaceID) {\n            const properties =\n                await this.propertyService.getPropertiesForOrganisation(orgID);\n            const propertySpaceIDs = properties.map(\n                (property_) => property_.spaceID,\n            );\n            if (propertySpaceIDs.includes(mission.spaceID)) {\n                return mission;\n            }\n            // we delegate the rights of the space\n            await this.propertyService.createProperty({\n                spaceID: mission.spaceID,\n                organisationID: orgID,\n                type: PropertyType.SHARED,\n            });\n        }\n        return mission;\n    }\n\n    async softDeleteUser(userID: string) {\n        const missions = await this.missionService.getMissionsFromUserID(userID);\n        if (missions.length === 0) {\n            await this.profileService.updateProfile({ id: userID, deletedAt: Date.now() });\n            console.log(\"Profile soft deleted\");\n        }\n    }\n    /**\n     * Checks if the Profile (by id) can be removed completely from the DB (no missions in the DB)\n     * @param profileID\n     * @returns boolean\n     */\n    async canHardDeleteUser(profileID: string): Promise<boolean> {\n        const missions = (await this.missionService.getMissionsFromUserID(profileID));\n        return missions.length === 0;\n    }\n\n    async hardDeleteUser(userID: string) {\n        const deleted = await this.profileService.deleteProfile(userID);\n        console.log(\"Profile hard deleted\");\n        return deleted;\n    }\n\n    getManagerMissions(): Mission[] {\n        if (!this.currentMissions) {\n            return [];\n        }\n        return this.currentMissions.filter((mis) =>\n            this.managers.includes(mis.role),\n        );\n    }\n\n    getAvalableLevelsForCurrentUser(): LevelStatus[] {\n        if (!this.currentMissions) {\n            return [LevelStatus.MINIMUM];\n        }\n        const levels = enumToArray(LevelStatus);\n        const highestLevel = getHighestLevelForMissions(this.currentMissions);\n\n        return levels.filter((level) =>\n            getLevelsBelow(highestLevel).includes(level),\n        );\n    }\n\n    getAvailableRolesForCurrentUser(): RoleStatus[] {\n        if (!this.currentMissions) {\n            return [RoleStatus.VISITOR];\n        }\n        const roles = enumToArray(RoleStatus);\n        const highestRole = getHighestRoleForMissions(this.currentMissions);\n        return roles.filter((role) =>\n            getRolesBelowForManager(highestRole).includes(role),\n        );\n    }\n\n    /** Test if the user have the role given in parameter : \"roleStatus\" */\n    isRole(roleStatus: RoleStatus): boolean {\n        if (!this.currentMissions) {\n            return false;\n        }\n        return this.currentMissions.some((m) => m.role === roleStatus);\n    }\n\n    isGuide(spaceID: string = null): boolean {\n        if (spaceID) {\n            const mission = this.getMissionForSpace(spaceID);\n            if (mission) {\n                return mission.role === RoleStatus.GUIDE_MUSEUM;\n            }\n        }\n        return this.isRole(RoleStatus.GUIDE_MUSEUM);\n    }\n\n    isMuseumVisitor(spaceID: string = null): boolean {\n        if (spaceID) {\n            const mission = this.getMissionForSpace(spaceID);\n            if (mission) {\n                return mission.role === RoleStatus.VISITOR_MUSEUM;\n            }\n        }\n        return this.isRole(RoleStatus.VISITOR_MUSEUM);\n    }\n\n    isOccupant(): boolean {\n        return this.isRole(RoleStatus.OCCUPANT);\n    }\n\n    isVisitor(spaceID: string = null): boolean {\n        if (spaceID) {\n            const mission = this.getMissionForSpace(spaceID);\n            if (mission) {\n                return mission.role === RoleStatus.VISITOR;\n            }\n        }\n        return this.isRole(RoleStatus.VISITOR);\n    }\n\n    isMuseumUser(spaceID: string = null): boolean {\n        return this.isGuide(spaceID) || this.isMuseumVisitor(spaceID);\n    }\n\n    hasOnlyMuseumRole(): boolean {\n        if (this.currentMissions) {\n            return this.currentMissions.every(\n                (m) =>\n                    m.role === RoleStatus.GUIDE_MUSEUM ||\n                    m.role === RoleStatus.VISITOR_MUSEUM,\n            );\n        }\n        return false;\n    }\n\n    async profileForMail(mail: string): Promise<Profile> {\n        const usersWithMail = await this.profileService.getUsersForMail(mail);\n        return usersWithMail.length > 0 ? usersWithMail[0] : null;\n    }\n\n    currentOrderedOrgList(): Organisation[] {\n        if (!this.currentMissions) {\n            return [];\n        }\n        const orderedOrgList = this.currentMissions.map(\n            (mis) => mis.orderedOrganisation,\n        );\n        return filterUniqueArrayByID(orderedOrgList);\n    }\n\n    userHasAccessToAllZonesInSpace(spaceID: string): boolean {\n        const missionsForSpace = this.currentMissions.filter(\n            (mis) => mis.spaceID === spaceID,\n        );\n        return missionsForSpace.some(\n            (mission) =>\n                !mission.zoneID || (mission.zone && !mission.zone.parentID),\n        );\n    }\n\n    async logoutCurrentAndSignOutAuth() {\n        await this.profileService.updateProfile({\n            id: this.cu.id,\n            token: null,\n        });\n        this.cu = null;\n        this.currentMissions = null;\n        this.isChanged.next(false);\n        await Auth.signOut();\n    }\n\n    async logoutCurrentWihtoutAuth() {\n        await this.profileService.updateProfile({\n            id: this.cu.id,\n            token: null,\n        });\n        this.cu = null;\n        this.currentMissions = null;\n        this.isChanged.next(false);\n    }\n\n    async checkAccessToken(jwtToken: string): Promise<boolean> {\n        const now = DateTime.local();\n        if (this.cu.token && this.cu.tokenExpiredAt > now.toMillis()) {\n            // check token\n            if (this.cu.token === jwtToken) {\n                // console.log(\"same aws token\");\n                return true;\n            }\n            alert(this.translate.instant(\"auth.error.activeToken\"));\n            return false;\n        }\n        await this.profileService.updateProfile({\n            id: this.cu.id,\n            tokenExpiredAt: now.plus({ minutes: 5 }).toMillis(),\n            token: jwtToken,\n            status: ProfileStatus.ACTIVE,\n        });\n        return true;\n    }\n\n    async clearUserToken(id) {\n        await this.profileService.updateProfile({\n            id: id,\n            token: null,\n        });\n    }\n\n    updateCurrentUser(profile: Profile) {\n        this.cu = new ProfileEntity(profile);\n        this.isChanged.next(true);\n    }\n\n    getSpModule(): SpModule {\n        return this.spModule;\n    }\n}\n"]}
|
|
1
|
+
import { Inject, Injectable } from "@angular/core";
|
|
2
|
+
import { Subject } from "rxjs";
|
|
3
|
+
import { LevelStatus, ProfileEntity, ProfileStatus, PropertyType, RoleStatus } from "../types.service";
|
|
4
|
+
import { enumToArray, filterUniqueArrayByID, getHighestLevelForMissions, getHighestRoleForMissions, getLevelsBelow, getRolesBelowForManager } from "../helpers.service";
|
|
5
|
+
import { DateTime } from "luxon";
|
|
6
|
+
import { Auth } from "aws-amplify";
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "./models/profile.service";
|
|
9
|
+
import * as i2 from "./models/mission.service";
|
|
10
|
+
import * as i3 from "./models/property.service";
|
|
11
|
+
import * as i4 from "@angular/router";
|
|
12
|
+
import * as i5 from "@angular/common";
|
|
13
|
+
import * as i6 from "@ngx-translate/core";
|
|
14
|
+
import * as i7 from "../types.service";
|
|
15
|
+
/**
|
|
16
|
+
* UserService handles user's lifecycle, from creation to deletion
|
|
17
|
+
*/
|
|
18
|
+
export class BaseUserService {
|
|
19
|
+
constructor(profileService, missionService, propertyService, router, location, ngZone, translate, spModuleInjected) {
|
|
20
|
+
this.profileService = profileService;
|
|
21
|
+
this.missionService = missionService;
|
|
22
|
+
this.propertyService = propertyService;
|
|
23
|
+
this.router = router;
|
|
24
|
+
this.location = location;
|
|
25
|
+
this.ngZone = ngZone;
|
|
26
|
+
this.translate = translate;
|
|
27
|
+
this.isChanged = new Subject();
|
|
28
|
+
this.updating = false;
|
|
29
|
+
this.managers = [
|
|
30
|
+
RoleStatus.ADMIN,
|
|
31
|
+
RoleStatus.MANAGER,
|
|
32
|
+
RoleStatus.OWNER,
|
|
33
|
+
RoleStatus.GUIDE_MUSEUM,
|
|
34
|
+
];
|
|
35
|
+
this.cu = null;
|
|
36
|
+
// this.checkConnectedAWSUser();
|
|
37
|
+
this.missionService.currentMissionsUpdated.subscribe(async (data) => {
|
|
38
|
+
this.currentMissions =
|
|
39
|
+
await this.missionService.instantMissionsOfCurrent(this.cu.id);
|
|
40
|
+
this.isChanged.next(true);
|
|
41
|
+
});
|
|
42
|
+
this.spModule = spModuleInjected;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Checks if AWS user is connected and assigng it to current user (this.cu)
|
|
46
|
+
**/
|
|
47
|
+
async checkConnectedAWSUser() {
|
|
48
|
+
if (!this.cu) {
|
|
49
|
+
// console.log("check AWS user");
|
|
50
|
+
let cau = null;
|
|
51
|
+
let user = null;
|
|
52
|
+
try {
|
|
53
|
+
if (!this.updating) {
|
|
54
|
+
cau = await Auth.currentAuthenticatedUser();
|
|
55
|
+
// console.log("connected", cau);
|
|
56
|
+
user = await this.current(cau);
|
|
57
|
+
// console.log("current of userService", user);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
catch { }
|
|
61
|
+
if (!user && !this.updating) {
|
|
62
|
+
// console.log("no user in check connected aws user");
|
|
63
|
+
this.router.navigate(["auth"]);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* createUser spawns a new User, along with:
|
|
69
|
+
* - a proper AWSCognito signup
|
|
70
|
+
* - a Profile entry holding the AWSCognito ID
|
|
71
|
+
* - a Mission entry holding the Organisation ID
|
|
72
|
+
*
|
|
73
|
+
* @see 174
|
|
74
|
+
* @see https://docs.amplify.aws/lib/auth/emailpassword/q/platform/js#sign-up
|
|
75
|
+
* @todo wonder if we can mash together Profile, Mission and User entries
|
|
76
|
+
* @param u User to be created
|
|
77
|
+
*/
|
|
78
|
+
async createUserAWSandProfile(u, prevalidated = false) {
|
|
79
|
+
const attributes = {
|
|
80
|
+
email: u.email.trim(),
|
|
81
|
+
given_name: u.firstName,
|
|
82
|
+
family_name: u.lastName,
|
|
83
|
+
zoneinfo: "fr",
|
|
84
|
+
locale: "fr",
|
|
85
|
+
"custom:prevalidated": "false",
|
|
86
|
+
};
|
|
87
|
+
if (prevalidated) {
|
|
88
|
+
attributes["custom:prevalidated"] = "true";
|
|
89
|
+
}
|
|
90
|
+
// AWSCognito
|
|
91
|
+
const password = u.password ? u.password : "temporaryp";
|
|
92
|
+
let error;
|
|
93
|
+
try {
|
|
94
|
+
const { userSub } = await Auth.signUp({
|
|
95
|
+
username: u.email,
|
|
96
|
+
password,
|
|
97
|
+
attributes,
|
|
98
|
+
clientMetadata: {
|
|
99
|
+
phone: u.phone,
|
|
100
|
+
},
|
|
101
|
+
});
|
|
102
|
+
if (userSub) {
|
|
103
|
+
return this.createProfile(userSub, u);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
catch (error_) {
|
|
107
|
+
switch (error_.code) {
|
|
108
|
+
case "InvalidPasswordException":
|
|
109
|
+
error = "auth.error.wrong_password";
|
|
110
|
+
break;
|
|
111
|
+
case "UserNotConfirmedException":
|
|
112
|
+
error = "auth.error.user_not_confirmed";
|
|
113
|
+
break;
|
|
114
|
+
case "InvalidParameterException":
|
|
115
|
+
error = "auth.error.invalid_parameter";
|
|
116
|
+
break;
|
|
117
|
+
default:
|
|
118
|
+
error = error_.message;
|
|
119
|
+
}
|
|
120
|
+
console.log(`AWS Sign UP error : ${error}`);
|
|
121
|
+
}
|
|
122
|
+
return Promise.reject(error);
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* - Creates AWS user in Cognito for Hotel User Pool Schema (only email is required)
|
|
126
|
+
* - Creates Profile in DynamoDB
|
|
127
|
+
*
|
|
128
|
+
* @param u User to be created
|
|
129
|
+
*/
|
|
130
|
+
async createHotelUserAWSandProfile(user) {
|
|
131
|
+
const attributes = {
|
|
132
|
+
email: user.email.trim()
|
|
133
|
+
};
|
|
134
|
+
// AWSCognito
|
|
135
|
+
const password = "temporaryp";
|
|
136
|
+
let error;
|
|
137
|
+
try {
|
|
138
|
+
const { userSub } = await Auth.signUp({
|
|
139
|
+
username: user.email.trim(),
|
|
140
|
+
password,
|
|
141
|
+
attributes,
|
|
142
|
+
});
|
|
143
|
+
if (userSub) {
|
|
144
|
+
return this.createProfile(userSub, user);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
catch (error_) {
|
|
148
|
+
switch (error_.code) {
|
|
149
|
+
case "InvalidPasswordException":
|
|
150
|
+
error = "auth.error.wrong_password";
|
|
151
|
+
break;
|
|
152
|
+
case "UserNotConfirmedException":
|
|
153
|
+
error = "auth.error.user_not_confirmed";
|
|
154
|
+
break;
|
|
155
|
+
case "InvalidParameterException":
|
|
156
|
+
error = "auth.error.invalid_parameter";
|
|
157
|
+
break;
|
|
158
|
+
default:
|
|
159
|
+
error = error_.message;
|
|
160
|
+
}
|
|
161
|
+
console.log(`AWS Sign UP error : ${error}`);
|
|
162
|
+
}
|
|
163
|
+
return Promise.reject(error);
|
|
164
|
+
}
|
|
165
|
+
async createProfile(awsUserID, u) {
|
|
166
|
+
const profile = await this.profileService.createProfile({
|
|
167
|
+
userID: awsUserID,
|
|
168
|
+
metadata: JSON.stringify({
|
|
169
|
+
email: u.email,
|
|
170
|
+
phone: u.phone,
|
|
171
|
+
lastName: u.lastName,
|
|
172
|
+
firstName: u.firstName,
|
|
173
|
+
}),
|
|
174
|
+
status: ProfileStatus.WAITING_FOR_ACTIVATION,
|
|
175
|
+
});
|
|
176
|
+
return profile;
|
|
177
|
+
}
|
|
178
|
+
currentOrganisation(spaceID = null) {
|
|
179
|
+
if (this.cu) {
|
|
180
|
+
if (!spaceID) {
|
|
181
|
+
return this.currentMissions[0].organisation;
|
|
182
|
+
}
|
|
183
|
+
return this.currentMission(spaceID).organisation;
|
|
184
|
+
}
|
|
185
|
+
return null;
|
|
186
|
+
}
|
|
187
|
+
async current(user, fastConnect = false) {
|
|
188
|
+
if (!this.cu && !this.updating) {
|
|
189
|
+
this.updating = true;
|
|
190
|
+
this.cu = await this.userFromAWSID(user.username);
|
|
191
|
+
if (!this.cu && user.attributes.email) {
|
|
192
|
+
console.log(`No profile for userID 🔪 ${user.attributes.email}`);
|
|
193
|
+
// There's no profiles, profiles.items.length === 0
|
|
194
|
+
// It's maybe the 🤮 case where the profile has been saved with an email
|
|
195
|
+
// instead of an hashed id
|
|
196
|
+
// If found, we need to replace the id in both the Profile
|
|
197
|
+
const emailProfile = await this.userFromAWSID(user.attributes.email);
|
|
198
|
+
if (emailProfile) {
|
|
199
|
+
console.log(`👏 doing the spacehip reversal email on ${user.attributes.email}`);
|
|
200
|
+
this.cu = await this.userFromAWSID(user.username);
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
if (!this.cu) {
|
|
204
|
+
this.updating = false;
|
|
205
|
+
await Auth.signOut();
|
|
206
|
+
return null;
|
|
207
|
+
}
|
|
208
|
+
const missions = await this.missionService.instantMissionsOfCurrent(this.cu.id);
|
|
209
|
+
this.currentMissions = missions;
|
|
210
|
+
let okToLogin = true;
|
|
211
|
+
if (!fastConnect && !this.hasManagerRoleInAnyMission()) {
|
|
212
|
+
// check access token if not Fast connect (fast connect is for the guest connection via link(/redirect?user=))
|
|
213
|
+
// checks visitors(museum or not), occupants, service
|
|
214
|
+
okToLogin = await this.checkAccessToken(user.signInUserSession.accessToken.jwtToken);
|
|
215
|
+
}
|
|
216
|
+
if (!okToLogin) {
|
|
217
|
+
this.cu = null;
|
|
218
|
+
this.currentMissions = [];
|
|
219
|
+
this.updating = false;
|
|
220
|
+
await Auth.signOut();
|
|
221
|
+
return null;
|
|
222
|
+
}
|
|
223
|
+
await this.profileService.updateProfile({
|
|
224
|
+
id: this.cu.id,
|
|
225
|
+
lastConnectedAt: Date.now(),
|
|
226
|
+
});
|
|
227
|
+
if (!missions && this.location.path() !== "/no-mission") {
|
|
228
|
+
this.ngZone.run(() => {
|
|
229
|
+
this.router.navigate(["/no-mission"]);
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
// console.log("user is changed emitting");
|
|
233
|
+
this.isChanged.next(true);
|
|
234
|
+
this.updating = false;
|
|
235
|
+
}
|
|
236
|
+
return this.cu;
|
|
237
|
+
}
|
|
238
|
+
async userFromAWSID(userId) {
|
|
239
|
+
const profile = await this.profileService.getProfileFromAWSID(userId);
|
|
240
|
+
if (profile) {
|
|
241
|
+
return new ProfileEntity(profile);
|
|
242
|
+
}
|
|
243
|
+
return null;
|
|
244
|
+
}
|
|
245
|
+
async userFromUserID(userID) {
|
|
246
|
+
const profile = await this.profileService.getProfile(userID);
|
|
247
|
+
return new ProfileEntity(profile);
|
|
248
|
+
}
|
|
249
|
+
currentMission(spaceID) {
|
|
250
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
251
|
+
return mission;
|
|
252
|
+
}
|
|
253
|
+
isManager(spaceID) {
|
|
254
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
255
|
+
if (mission) {
|
|
256
|
+
return this.managers.includes(mission.role);
|
|
257
|
+
}
|
|
258
|
+
return false;
|
|
259
|
+
}
|
|
260
|
+
isService(spaceID) {
|
|
261
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
262
|
+
if (mission) {
|
|
263
|
+
return mission.role === RoleStatus.SERVICE;
|
|
264
|
+
}
|
|
265
|
+
return false;
|
|
266
|
+
}
|
|
267
|
+
isAdmin(spaceID) {
|
|
268
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
269
|
+
if (mission) {
|
|
270
|
+
return mission.role === RoleStatus.ADMIN;
|
|
271
|
+
}
|
|
272
|
+
return false;
|
|
273
|
+
}
|
|
274
|
+
isSmarterPlan() {
|
|
275
|
+
if (!this.currentMissions) {
|
|
276
|
+
return false;
|
|
277
|
+
}
|
|
278
|
+
const spMissions = this.currentMissions.filter((m) => m.organisationID === "6e28201f-4679-4c29-9d2d-8a325f57cee9" &&
|
|
279
|
+
this.managers.includes(m.role)); // ID of SmarterPlan
|
|
280
|
+
return spMissions.length > 0;
|
|
281
|
+
}
|
|
282
|
+
isSPAdmin() {
|
|
283
|
+
if (!this.currentMissions) {
|
|
284
|
+
return false;
|
|
285
|
+
}
|
|
286
|
+
const spMissions = this.currentMissions.filter((m) => m.organisationID === "6e28201f-4679-4c29-9d2d-8a325f57cee9" &&
|
|
287
|
+
m.role === RoleStatus.ADMIN); // ID of SmarterPlan
|
|
288
|
+
return spMissions.length > 0;
|
|
289
|
+
}
|
|
290
|
+
getMissionForSpace(spaceID) {
|
|
291
|
+
if (!this.currentMissions) {
|
|
292
|
+
return null;
|
|
293
|
+
}
|
|
294
|
+
return this.currentMissions.find((m) => m.spaceID === spaceID);
|
|
295
|
+
}
|
|
296
|
+
hasManagerRoleInAnyMission() {
|
|
297
|
+
if (!this.currentMissions) {
|
|
298
|
+
return false;
|
|
299
|
+
}
|
|
300
|
+
return this.currentMissions.some((mis) => this.managers.includes(mis.role));
|
|
301
|
+
}
|
|
302
|
+
hasServiceRoleInAnyMission() {
|
|
303
|
+
if (!this.currentMissions) {
|
|
304
|
+
return false;
|
|
305
|
+
}
|
|
306
|
+
return this.currentMissions.some((mis) => mis.role === RoleStatus.SERVICE);
|
|
307
|
+
}
|
|
308
|
+
hasAdminRoleInAnyMission() {
|
|
309
|
+
if (!this.currentMissions) {
|
|
310
|
+
return false;
|
|
311
|
+
}
|
|
312
|
+
return this.currentMissions.some((mis) => mis.role === RoleStatus.ADMIN);
|
|
313
|
+
}
|
|
314
|
+
async createOrUpdateMissionForOrgWithProperty(missionInput, orgID, create = true) {
|
|
315
|
+
const mission = await (create
|
|
316
|
+
? this.missionService.create(missionInput)
|
|
317
|
+
: this.missionService.update(missionInput));
|
|
318
|
+
if (this.managers.includes(mission.role) && mission.spaceID) {
|
|
319
|
+
const properties = await this.propertyService.getPropertiesForOrganisation(orgID);
|
|
320
|
+
const propertySpaceIDs = properties.map((property_) => property_.spaceID);
|
|
321
|
+
if (propertySpaceIDs.includes(mission.spaceID)) {
|
|
322
|
+
return mission;
|
|
323
|
+
}
|
|
324
|
+
// we delegate the rights of the space
|
|
325
|
+
await this.propertyService.createProperty({
|
|
326
|
+
spaceID: mission.spaceID,
|
|
327
|
+
organisationID: orgID,
|
|
328
|
+
type: PropertyType.SHARED,
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
return mission;
|
|
332
|
+
}
|
|
333
|
+
async softDeleteUser(userID) {
|
|
334
|
+
const missions = await this.missionService.getMissionsFromUserID(userID);
|
|
335
|
+
if (missions.length === 0) {
|
|
336
|
+
await this.profileService.updateProfile({ id: userID, deletedAt: Date.now() });
|
|
337
|
+
console.log("Profile soft deleted");
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
/**
|
|
341
|
+
* Checks if the Profile (by id) can be removed completely from the DB (no missions in the DB)
|
|
342
|
+
* @param profileID
|
|
343
|
+
* @returns boolean
|
|
344
|
+
*/
|
|
345
|
+
async canHardDeleteUser(profileID) {
|
|
346
|
+
const missions = (await this.missionService.getMissionsFromUserID(profileID));
|
|
347
|
+
return missions.length === 0;
|
|
348
|
+
}
|
|
349
|
+
async hardDeleteUser(userID) {
|
|
350
|
+
const deleted = await this.profileService.deleteProfile(userID);
|
|
351
|
+
console.log("Profile hard deleted");
|
|
352
|
+
return deleted;
|
|
353
|
+
}
|
|
354
|
+
getManagerMissions() {
|
|
355
|
+
if (!this.currentMissions) {
|
|
356
|
+
return [];
|
|
357
|
+
}
|
|
358
|
+
return this.currentMissions.filter((mis) => this.managers.includes(mis.role));
|
|
359
|
+
}
|
|
360
|
+
getAvalableLevelsForCurrentUser() {
|
|
361
|
+
if (!this.currentMissions) {
|
|
362
|
+
return [LevelStatus.MINIMUM];
|
|
363
|
+
}
|
|
364
|
+
const levels = enumToArray(LevelStatus);
|
|
365
|
+
const highestLevel = getHighestLevelForMissions(this.currentMissions);
|
|
366
|
+
return levels.filter((level) => getLevelsBelow(highestLevel).includes(level));
|
|
367
|
+
}
|
|
368
|
+
getAvailableRolesForCurrentUser() {
|
|
369
|
+
if (!this.currentMissions) {
|
|
370
|
+
return [RoleStatus.VISITOR];
|
|
371
|
+
}
|
|
372
|
+
const roles = enumToArray(RoleStatus);
|
|
373
|
+
const highestRole = getHighestRoleForMissions(this.currentMissions);
|
|
374
|
+
return roles.filter((role) => getRolesBelowForManager(highestRole).includes(role));
|
|
375
|
+
}
|
|
376
|
+
/** Test if the user have the role given in parameter : "roleStatus" */
|
|
377
|
+
isRole(roleStatus) {
|
|
378
|
+
if (!this.currentMissions) {
|
|
379
|
+
return false;
|
|
380
|
+
}
|
|
381
|
+
return this.currentMissions.some((m) => m.role === roleStatus);
|
|
382
|
+
}
|
|
383
|
+
isGuide(spaceID = null) {
|
|
384
|
+
if (spaceID) {
|
|
385
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
386
|
+
if (mission) {
|
|
387
|
+
return mission.role === RoleStatus.GUIDE_MUSEUM;
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
return this.isRole(RoleStatus.GUIDE_MUSEUM);
|
|
391
|
+
}
|
|
392
|
+
isMuseumVisitor(spaceID = null) {
|
|
393
|
+
if (spaceID) {
|
|
394
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
395
|
+
if (mission) {
|
|
396
|
+
return mission.role === RoleStatus.VISITOR_MUSEUM;
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
return this.isRole(RoleStatus.VISITOR_MUSEUM);
|
|
400
|
+
}
|
|
401
|
+
isOccupant() {
|
|
402
|
+
return this.isRole(RoleStatus.OCCUPANT);
|
|
403
|
+
}
|
|
404
|
+
isVisitor(spaceID = null) {
|
|
405
|
+
if (spaceID) {
|
|
406
|
+
const mission = this.getMissionForSpace(spaceID);
|
|
407
|
+
if (mission) {
|
|
408
|
+
return mission.role === RoleStatus.VISITOR;
|
|
409
|
+
}
|
|
410
|
+
}
|
|
411
|
+
return this.isRole(RoleStatus.VISITOR);
|
|
412
|
+
}
|
|
413
|
+
isMuseumUser(spaceID = null) {
|
|
414
|
+
return this.isGuide(spaceID) || this.isMuseumVisitor(spaceID);
|
|
415
|
+
}
|
|
416
|
+
hasOnlyMuseumRole() {
|
|
417
|
+
if (this.currentMissions) {
|
|
418
|
+
return this.currentMissions.every((m) => m.role === RoleStatus.GUIDE_MUSEUM ||
|
|
419
|
+
m.role === RoleStatus.VISITOR_MUSEUM);
|
|
420
|
+
}
|
|
421
|
+
return false;
|
|
422
|
+
}
|
|
423
|
+
async profileForMail(mail) {
|
|
424
|
+
const usersWithMail = await this.profileService.getUsersForMail(mail);
|
|
425
|
+
return usersWithMail.length > 0 ? usersWithMail[0] : null;
|
|
426
|
+
}
|
|
427
|
+
currentOrderedOrgList() {
|
|
428
|
+
if (!this.currentMissions) {
|
|
429
|
+
return [];
|
|
430
|
+
}
|
|
431
|
+
const orderedOrgList = this.currentMissions.map((mis) => mis.orderedOrganisation);
|
|
432
|
+
return filterUniqueArrayByID(orderedOrgList);
|
|
433
|
+
}
|
|
434
|
+
userHasAccessToAllZonesInSpace(spaceID) {
|
|
435
|
+
const missionsForSpace = this.currentMissions.filter((mis) => mis.spaceID === spaceID);
|
|
436
|
+
return missionsForSpace.some((mission) => !mission.zoneID || (mission.zone && !mission.zone.parentID));
|
|
437
|
+
}
|
|
438
|
+
async logoutCurrentAndSignOutAuth() {
|
|
439
|
+
await this.profileService.updateProfile({
|
|
440
|
+
id: this.cu.id,
|
|
441
|
+
token: null,
|
|
442
|
+
});
|
|
443
|
+
this.cu = null;
|
|
444
|
+
this.currentMissions = null;
|
|
445
|
+
this.isChanged.next(false);
|
|
446
|
+
await Auth.signOut();
|
|
447
|
+
}
|
|
448
|
+
async logoutCurrentWihtoutAuth() {
|
|
449
|
+
await this.profileService.updateProfile({
|
|
450
|
+
id: this.cu.id,
|
|
451
|
+
token: null,
|
|
452
|
+
});
|
|
453
|
+
this.cu = null;
|
|
454
|
+
this.currentMissions = null;
|
|
455
|
+
this.isChanged.next(false);
|
|
456
|
+
}
|
|
457
|
+
async checkAccessToken(jwtToken) {
|
|
458
|
+
const now = DateTime.local();
|
|
459
|
+
if (this.cu.token && this.cu.tokenExpiredAt > now.toMillis()) {
|
|
460
|
+
// check token
|
|
461
|
+
if (this.cu.token === jwtToken) {
|
|
462
|
+
// console.log("same aws token");
|
|
463
|
+
return true;
|
|
464
|
+
}
|
|
465
|
+
alert(this.translate.instant("auth.error.activeToken"));
|
|
466
|
+
return false;
|
|
467
|
+
}
|
|
468
|
+
await this.profileService.updateProfile({
|
|
469
|
+
id: this.cu.id,
|
|
470
|
+
tokenExpiredAt: now.plus({ minutes: 5 }).toMillis(),
|
|
471
|
+
token: jwtToken,
|
|
472
|
+
status: ProfileStatus.ACTIVE,
|
|
473
|
+
});
|
|
474
|
+
return true;
|
|
475
|
+
}
|
|
476
|
+
async clearUserToken(id) {
|
|
477
|
+
await this.profileService.updateProfile({
|
|
478
|
+
id: id,
|
|
479
|
+
token: null,
|
|
480
|
+
});
|
|
481
|
+
}
|
|
482
|
+
updateCurrentUser(profile) {
|
|
483
|
+
this.cu = new ProfileEntity(profile);
|
|
484
|
+
this.isChanged.next(true);
|
|
485
|
+
}
|
|
486
|
+
getSpModule() {
|
|
487
|
+
return this.spModule;
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
BaseUserService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, deps: [{ token: i1.ProfileService }, { token: i2.MissionService }, { token: i3.PropertyService }, { token: i4.Router }, { token: i5.Location }, { token: i0.NgZone }, { token: i6.TranslateService }, { token: 'SPModule' }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
491
|
+
BaseUserService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, providedIn: "root" });
|
|
492
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: BaseUserService, decorators: [{
|
|
493
|
+
type: Injectable,
|
|
494
|
+
args: [{
|
|
495
|
+
providedIn: "root",
|
|
496
|
+
}]
|
|
497
|
+
}], ctorParameters: function () { return [{ type: i1.ProfileService }, { type: i2.MissionService }, { type: i3.PropertyService }, { type: i4.Router }, { type: i5.Location }, { type: i0.NgZone }, { type: i6.TranslateService }, { type: i7.SpModule, decorators: [{
|
|
498
|
+
type: Inject,
|
|
499
|
+
args: ['SPModule']
|
|
500
|
+
}] }]; } });
|
|
501
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-smarterplan-core/src/lib/services/user.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, NgZone } from \"@angular/core\";\r\nimport { Subject } from \"rxjs\";\r\nimport { Router } from \"@angular/router\";\r\nimport { Location } from \"@angular/common\";\r\nimport { TranslateService } from \"@ngx-translate/core\";\r\nimport { LevelStatus, Mission, Organisation, Profile, ProfileEntity, ProfileStatus, PropertyType, RoleStatus, SpModule } from \"../types.service\";\r\nimport { enumToArray, filterUniqueArrayByID, getHighestLevelForMissions, getHighestRoleForMissions, getLevelsBelow, getRolesBelowForManager } from \"../helpers.service\";\r\nimport { ProfileService } from \"./models/profile.service\";\r\nimport { MissionService } from \"./models/mission.service\";\r\nimport { PropertyService } from \"./models/property.service\";\r\nimport { DateTime } from \"luxon\";\r\nimport { Auth } from \"aws-amplify\";\r\n\r\ninterface CreateUser {\r\n    email: string;\r\n    phone?: string;\r\n    lastName: string;\r\n    firstName: string;\r\n    password?: string;\r\n}\r\n\r\n/**\r\n * UserService handles user's lifecycle, from creation to deletion\r\n */\r\n@Injectable({\r\n    providedIn: \"root\",\r\n})\r\nexport class BaseUserService {\r\n\r\n    isChanged: Subject<boolean> = new Subject<boolean>();\r\n\r\n    updating = false;\r\n\r\n    managers = [\r\n        RoleStatus.ADMIN,\r\n        RoleStatus.MANAGER,\r\n        RoleStatus.OWNER,\r\n        RoleStatus.GUIDE_MUSEUM,\r\n    ];\r\n\r\n    currentMissions: Mission[];\r\n\r\n    lastExpiredMission: Mission;\r\n\r\n    nextMissionToStart: Mission;\r\n\r\n    cu: ProfileEntity = null;\r\n\r\n    spModule: SpModule;\r\n\r\n    constructor(\r\n        private profileService: ProfileService,\r\n        private missionService: MissionService,\r\n        private propertyService: PropertyService,\r\n        private router: Router,\r\n        private location: Location,\r\n        private ngZone: NgZone,\r\n        private translate: TranslateService,\r\n        @Inject('SPModule') spModuleInjected: SpModule,\r\n    ) {\r\n        // this.checkConnectedAWSUser();\r\n        this.missionService.currentMissionsUpdated.subscribe(async (data) => {\r\n            this.currentMissions =\r\n                await this.missionService.instantMissionsOfCurrent(this.cu.id);\r\n            this.isChanged.next(true);\r\n        });\r\n        this.spModule = spModuleInjected;\r\n    }\r\n\r\n    /**\r\n     * Checks if AWS user is connected and assigng it to current user (this.cu)\r\n    **/\r\n    async checkConnectedAWSUser() {\r\n        if (!this.cu) {\r\n            // console.log(\"check AWS user\");\r\n            let cau = null;\r\n            let user = null;\r\n            try {\r\n                if (!this.updating) {\r\n                    cau = await Auth.currentAuthenticatedUser();\r\n                    // console.log(\"connected\", cau);\r\n                    user = await this.current(cau);\r\n                    // console.log(\"current of userService\", user);\r\n                }\r\n            } catch { }\r\n            if (!user && !this.updating) {\r\n                // console.log(\"no user in check connected aws user\");\r\n                this.router.navigate([\"auth\"]);\r\n            }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * createUser spawns a new User, along with:\r\n     * - a proper AWSCognito signup\r\n     * - a Profile entry holding the AWSCognito ID\r\n     * - a Mission entry holding the Organisation ID\r\n     *\r\n     * @see 174\r\n     * @see https://docs.amplify.aws/lib/auth/emailpassword/q/platform/js#sign-up\r\n     * @todo wonder if we can mash together Profile, Mission and User entries\r\n     * @param u User to be created\r\n     */\r\n    async createUserAWSandProfile(\r\n        u: {\r\n            email: string;\r\n            phone: string;\r\n            lastName: string;\r\n            firstName: string;\r\n            password?: string;\r\n        },\r\n        prevalidated: boolean = false,\r\n    ): Promise<Profile> {\r\n        const attributes = {\r\n            email: u.email.trim(),\r\n            given_name: u.firstName,\r\n            family_name: u.lastName,\r\n            zoneinfo: \"fr\", // @todo deal with zoneinfo and locale\r\n            locale: \"fr\",\r\n            \"custom:prevalidated\": \"false\",\r\n        };\r\n        if (prevalidated) {\r\n            attributes[\"custom:prevalidated\"] = \"true\";\r\n        }\r\n        // AWSCognito\r\n        const password = u.password ? u.password : \"temporaryp\";\r\n        let error;\r\n        try {\r\n            const { userSub } = await Auth.signUp({\r\n                username: u.email, // username should be an email they say\r\n                password,\r\n                attributes,\r\n                clientMetadata: {\r\n                    phone: u.phone,\r\n                },\r\n            });\r\n            if (userSub) {\r\n                return this.createProfile(userSub, u);\r\n            }\r\n        } catch (error_: any) {\r\n            switch (error_.code) {\r\n                case \"InvalidPasswordException\":\r\n                    error = \"auth.error.wrong_password\";\r\n                    break;\r\n                case \"UserNotConfirmedException\":\r\n                    error = \"auth.error.user_not_confirmed\";\r\n                    break;\r\n                case \"InvalidParameterException\":\r\n                    error = \"auth.error.invalid_parameter\";\r\n                    break;\r\n                default:\r\n                    error = error_.message;\r\n            }\r\n            console.log(`AWS Sign UP error : ${error}`);\r\n        }\r\n        return Promise.reject(error);\r\n    }\r\n\r\n    /**\r\n     * - Creates AWS user in Cognito for Hotel User Pool Schema (only email is required)\r\n     * - Creates Profile in DynamoDB\r\n     *\r\n     * @param u User to be created\r\n     */\r\n    async createHotelUserAWSandProfile(\r\n        user: {\r\n            email: string;\r\n            lastName: string;\r\n            firstName: string;\r\n        },\r\n    ): Promise<Profile> {\r\n        const attributes = {\r\n            email: user.email.trim()\r\n        };\r\n        // AWSCognito\r\n        const password = \"temporaryp\";\r\n        let error;\r\n        try {\r\n            const { userSub } = await Auth.signUp({\r\n                username: user.email.trim(),\r\n                password,\r\n                attributes,\r\n            });\r\n            if (userSub) {\r\n                return this.createProfile(userSub, user);\r\n            }\r\n        } catch (error_: any) {\r\n            switch (error_.code) {\r\n                case \"InvalidPasswordException\":\r\n                    error = \"auth.error.wrong_password\";\r\n                    break;\r\n                case \"UserNotConfirmedException\":\r\n                    error = \"auth.error.user_not_confirmed\";\r\n                    break;\r\n                case \"InvalidParameterException\":\r\n                    error = \"auth.error.invalid_parameter\";\r\n                    break;\r\n                default:\r\n                    error = error_.message;\r\n            }\r\n            console.log(`AWS Sign UP error : ${error}`);\r\n        }\r\n        return Promise.reject(error);\r\n    }\r\n\r\n    async createProfile(\r\n        awsUserID: string,\r\n        u: CreateUser,\r\n    ): Promise<Profile> {\r\n        const profile = await this.profileService.createProfile({\r\n            userID: awsUserID,\r\n            metadata: JSON.stringify({\r\n                email: u.email,\r\n                phone: u.phone,\r\n                lastName: u.lastName,\r\n                firstName: u.firstName,\r\n            }),\r\n            status: ProfileStatus.WAITING_FOR_ACTIVATION,\r\n        });\r\n        return profile;\r\n    }\r\n\r\n    currentOrganisation(spaceID: string = null): Organisation {\r\n        if (this.cu) {\r\n            if (!spaceID) {\r\n                return this.currentMissions[0].organisation;\r\n            }\r\n            return this.currentMission(spaceID).organisation;\r\n        }\r\n        return null;\r\n    }\r\n\r\n    async current(\r\n        user,\r\n        fastConnect: boolean = false,\r\n    ): Promise<ProfileEntity | null> {\r\n        if (!this.cu && !this.updating) {\r\n            this.updating = true;\r\n            this.cu = await this.userFromAWSID(user.username);\r\n            if (!this.cu && user.attributes.email) {\r\n                console.log(\r\n                    `No profile for userID 🔪 ${user.attributes.email}`,\r\n                );\r\n                // There's no profiles, profiles.items.length === 0\r\n                // It's maybe the 🤮 case where the profile has been saved with an email\r\n                // instead of an hashed id\r\n                // If found, we need to replace the id in both the Profile\r\n                const emailProfile = await this.userFromAWSID(\r\n                    user.attributes.email,\r\n                );\r\n                if (emailProfile) {\r\n                    console.log(\r\n                        `👏 doing the spacehip reversal email on ${user.attributes.email}`,\r\n                    );\r\n                    this.cu = await this.userFromAWSID(user.username);\r\n                }\r\n            }\r\n            if (!this.cu) {\r\n                this.updating = false;\r\n                await Auth.signOut();\r\n                return null;\r\n            }\r\n            const missions = await this.missionService.instantMissionsOfCurrent(\r\n                this.cu.id\r\n            );\r\n            this.currentMissions = missions;\r\n            let okToLogin = true;\r\n            if (!fastConnect && !this.hasManagerRoleInAnyMission()) {\r\n                // check access token if not Fast connect (fast connect is for the guest connection via link(/redirect?user=))\r\n                // checks visitors(museum or not), occupants, service\r\n                okToLogin = await this.checkAccessToken(\r\n                    user.signInUserSession.accessToken.jwtToken,\r\n                );\r\n            }\r\n            if (!okToLogin) {\r\n                this.cu = null;\r\n                this.currentMissions = [];\r\n                this.updating = false;\r\n                await Auth.signOut();\r\n                return null;\r\n            }\r\n            await this.profileService.updateProfile({\r\n                id: this.cu.id,\r\n                lastConnectedAt: Date.now(),\r\n            });\r\n            if (!missions && this.location.path() !== \"/no-mission\") {\r\n                this.ngZone.run(() => {\r\n                    this.router.navigate([\"/no-mission\"]);\r\n                });\r\n            }\r\n            // console.log(\"user is changed emitting\");\r\n            this.isChanged.next(true);\r\n            this.updating = false;\r\n        }\r\n        return this.cu;\r\n    }\r\n\r\n    async userFromAWSID(userId: string): Promise<ProfileEntity | null> {\r\n        const profile = await this.profileService.getProfileFromAWSID(userId);\r\n        if (profile) {\r\n            return new ProfileEntity(profile);\r\n        }\r\n        return null;\r\n    }\r\n\r\n    async userFromUserID(userID: string): Promise<ProfileEntity | null> {\r\n        const profile = await this.profileService.getProfile(userID);\r\n        return new ProfileEntity(profile);\r\n    }\r\n\r\n    currentMission(spaceID: string): Mission {\r\n        const mission = this.getMissionForSpace(spaceID);\r\n        return mission;\r\n    }\r\n\r\n    isManager(spaceID: string): boolean {\r\n        const mission = this.getMissionForSpace(spaceID);\r\n        if (mission) {\r\n            return this.managers.includes(mission.role);\r\n        }\r\n        return false;\r\n    }\r\n\r\n    isService(spaceID: string): boolean {\r\n        const mission = this.getMissionForSpace(spaceID);\r\n        if (mission) {\r\n            return mission.role === RoleStatus.SERVICE;\r\n        }\r\n        return false;\r\n    }\r\n\r\n    isAdmin(spaceID: string): boolean {\r\n        const mission = this.getMissionForSpace(spaceID);\r\n        if (mission) {\r\n            return mission.role === RoleStatus.ADMIN;\r\n        }\r\n        return false;\r\n    }\r\n\r\n    isSmarterPlan(): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        const spMissions = this.currentMissions.filter(\r\n            (m) =>\r\n                m.organisationID === \"6e28201f-4679-4c29-9d2d-8a325f57cee9\" &&\r\n                this.managers.includes(m.role),\r\n        ); // ID of SmarterPlan\r\n        return spMissions.length > 0;\r\n    }\r\n\r\n    isSPAdmin(): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        const spMissions = this.currentMissions.filter(\r\n            (m) =>\r\n                m.organisationID === \"6e28201f-4679-4c29-9d2d-8a325f57cee9\" &&\r\n                m.role === RoleStatus.ADMIN,\r\n        ); // ID of SmarterPlan\r\n        return spMissions.length > 0;\r\n    }\r\n\r\n    getMissionForSpace(spaceID: string): Mission {\r\n        if (!this.currentMissions) {\r\n            return null;\r\n        }\r\n        return this.currentMissions.find((m) => m.spaceID === spaceID);\r\n    }\r\n\r\n    hasManagerRoleInAnyMission(): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        return this.currentMissions.some((mis) =>\r\n            this.managers.includes(mis.role),\r\n        );\r\n    }\r\n\r\n    hasServiceRoleInAnyMission(): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        return this.currentMissions.some(\r\n            (mis) => mis.role === RoleStatus.SERVICE,\r\n        );\r\n    }\r\n\r\n    hasAdminRoleInAnyMission(): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        return this.currentMissions.some(\r\n            (mis) => mis.role === RoleStatus.ADMIN,\r\n        );\r\n    }\r\n\r\n    async createOrUpdateMissionForOrgWithProperty(\r\n        missionInput: Mission,\r\n        orgID: string,\r\n        create: boolean = true,\r\n    ): Promise<Mission> {\r\n        const mission = await (create\r\n            ? this.missionService.create(missionInput)\r\n            : this.missionService.update(missionInput));\r\n\r\n        if (this.managers.includes(mission.role) && mission.spaceID) {\r\n            const properties =\r\n                await this.propertyService.getPropertiesForOrganisation(orgID);\r\n            const propertySpaceIDs = properties.map(\r\n                (property_) => property_.spaceID,\r\n            );\r\n            if (propertySpaceIDs.includes(mission.spaceID)) {\r\n                return mission;\r\n            }\r\n            // we delegate the rights of the space\r\n            await this.propertyService.createProperty({\r\n                spaceID: mission.spaceID,\r\n                organisationID: orgID,\r\n                type: PropertyType.SHARED,\r\n            });\r\n        }\r\n        return mission;\r\n    }\r\n\r\n    async softDeleteUser(userID: string) {\r\n        const missions = await this.missionService.getMissionsFromUserID(userID);\r\n        if (missions.length === 0) {\r\n            await this.profileService.updateProfile({ id: userID, deletedAt: Date.now() });\r\n            console.log(\"Profile soft deleted\");\r\n        }\r\n    }\r\n    /**\r\n     * Checks if the Profile (by id) can be removed completely from the DB (no missions in the DB)\r\n     * @param profileID\r\n     * @returns boolean\r\n     */\r\n    async canHardDeleteUser(profileID: string): Promise<boolean> {\r\n        const missions = (await this.missionService.getMissionsFromUserID(profileID));\r\n        return missions.length === 0;\r\n    }\r\n\r\n    async hardDeleteUser(userID: string) {\r\n        const deleted = await this.profileService.deleteProfile(userID);\r\n        console.log(\"Profile hard deleted\");\r\n        return deleted;\r\n    }\r\n\r\n    getManagerMissions(): Mission[] {\r\n        if (!this.currentMissions) {\r\n            return [];\r\n        }\r\n        return this.currentMissions.filter((mis) =>\r\n            this.managers.includes(mis.role),\r\n        );\r\n    }\r\n\r\n    getAvalableLevelsForCurrentUser(): LevelStatus[] {\r\n        if (!this.currentMissions) {\r\n            return [LevelStatus.MINIMUM];\r\n        }\r\n        const levels = enumToArray(LevelStatus);\r\n        const highestLevel = getHighestLevelForMissions(this.currentMissions);\r\n\r\n        return levels.filter((level) =>\r\n            getLevelsBelow(highestLevel).includes(level),\r\n        );\r\n    }\r\n\r\n    getAvailableRolesForCurrentUser(): RoleStatus[] {\r\n        if (!this.currentMissions) {\r\n            return [RoleStatus.VISITOR];\r\n        }\r\n        const roles = enumToArray(RoleStatus);\r\n        const highestRole = getHighestRoleForMissions(this.currentMissions);\r\n        return roles.filter((role) =>\r\n            getRolesBelowForManager(highestRole).includes(role),\r\n        );\r\n    }\r\n\r\n    /** Test if the user have the role given in parameter : \"roleStatus\" */\r\n    isRole(roleStatus: RoleStatus): boolean {\r\n        if (!this.currentMissions) {\r\n            return false;\r\n        }\r\n        return this.currentMissions.some((m) => m.role === roleStatus);\r\n    }\r\n\r\n    isGuide(spaceID: string = null): boolean {\r\n        if (spaceID) {\r\n            const mission = this.getMissionForSpace(spaceID);\r\n            if (mission) {\r\n                return mission.role === RoleStatus.GUIDE_MUSEUM;\r\n            }\r\n        }\r\n        return this.isRole(RoleStatus.GUIDE_MUSEUM);\r\n    }\r\n\r\n    isMuseumVisitor(spaceID: string = null): boolean {\r\n        if (spaceID) {\r\n            const mission = this.getMissionForSpace(spaceID);\r\n            if (mission) {\r\n                return mission.role === RoleStatus.VISITOR_MUSEUM;\r\n            }\r\n        }\r\n        return this.isRole(RoleStatus.VISITOR_MUSEUM);\r\n    }\r\n\r\n    isOccupant(): boolean {\r\n        return this.isRole(RoleStatus.OCCUPANT);\r\n    }\r\n\r\n    isVisitor(spaceID: string = null): boolean {\r\n        if (spaceID) {\r\n            const mission = this.getMissionForSpace(spaceID);\r\n            if (mission) {\r\n                return mission.role === RoleStatus.VISITOR;\r\n            }\r\n        }\r\n        return this.isRole(RoleStatus.VISITOR);\r\n    }\r\n\r\n    isMuseumUser(spaceID: string = null): boolean {\r\n        return this.isGuide(spaceID) || this.isMuseumVisitor(spaceID);\r\n    }\r\n\r\n    hasOnlyMuseumRole(): boolean {\r\n        if (this.currentMissions) {\r\n            return this.currentMissions.every(\r\n                (m) =>\r\n                    m.role === RoleStatus.GUIDE_MUSEUM ||\r\n                    m.role === RoleStatus.VISITOR_MUSEUM,\r\n            );\r\n        }\r\n        return false;\r\n    }\r\n\r\n    async profileForMail(mail: string): Promise<Profile> {\r\n        const usersWithMail = await this.profileService.getUsersForMail(mail);\r\n        return usersWithMail.length > 0 ? usersWithMail[0] : null;\r\n    }\r\n\r\n    currentOrderedOrgList(): Organisation[] {\r\n        if (!this.currentMissions) {\r\n            return [];\r\n        }\r\n        const orderedOrgList = this.currentMissions.map(\r\n            (mis) => mis.orderedOrganisation,\r\n        );\r\n        return filterUniqueArrayByID(orderedOrgList);\r\n    }\r\n\r\n    userHasAccessToAllZonesInSpace(spaceID: string): boolean {\r\n        const missionsForSpace = this.currentMissions.filter(\r\n            (mis) => mis.spaceID === spaceID,\r\n        );\r\n        return missionsForSpace.some(\r\n            (mission) =>\r\n                !mission.zoneID || (mission.zone && !mission.zone.parentID),\r\n        );\r\n    }\r\n\r\n    async logoutCurrentAndSignOutAuth() {\r\n        await this.profileService.updateProfile({\r\n            id: this.cu.id,\r\n            token: null,\r\n        });\r\n        this.cu = null;\r\n        this.currentMissions = null;\r\n        this.isChanged.next(false);\r\n        await Auth.signOut();\r\n    }\r\n\r\n    async logoutCurrentWihtoutAuth() {\r\n        await this.profileService.updateProfile({\r\n            id: this.cu.id,\r\n            token: null,\r\n        });\r\n        this.cu = null;\r\n        this.currentMissions = null;\r\n        this.isChanged.next(false);\r\n    }\r\n\r\n    async checkAccessToken(jwtToken: string): Promise<boolean> {\r\n        const now = DateTime.local();\r\n        if (this.cu.token && this.cu.tokenExpiredAt > now.toMillis()) {\r\n            // check token\r\n            if (this.cu.token === jwtToken) {\r\n                // console.log(\"same aws token\");\r\n                return true;\r\n            }\r\n            alert(this.translate.instant(\"auth.error.activeToken\"));\r\n            return false;\r\n        }\r\n        await this.profileService.updateProfile({\r\n            id: this.cu.id,\r\n            tokenExpiredAt: now.plus({ minutes: 5 }).toMillis(),\r\n            token: jwtToken,\r\n            status: ProfileStatus.ACTIVE,\r\n        });\r\n        return true;\r\n    }\r\n\r\n    async clearUserToken(id) {\r\n        await this.profileService.updateProfile({\r\n            id: id,\r\n            token: null,\r\n        });\r\n    }\r\n\r\n    updateCurrentUser(profile: Profile) {\r\n        this.cu = new ProfileEntity(profile);\r\n        this.isChanged.next(true);\r\n    }\r\n\r\n    getSpModule(): SpModule {\r\n        return this.spModule;\r\n    }\r\n}\r\n"]}
|