2022-01-06 20:43:57 +01:00
import bcrypt from 'bcryptjs' ;
2021-04-09 13:46:53 +02:00
import owasp from 'owasp-password-strength-test' ;
import Joi from 'joi' ;
2021-04-16 15:29:23 +02:00
import { URL } from 'url' ;
2021-04-09 13:46:53 +02:00
import { Logger } from '../logger' ;
2023-11-24 16:06:37 +01:00
import User , { IUser , IUserWithRootRole } from '../types/user' ;
2021-04-09 13:46:53 +02:00
import isEmail from '../util/is-email' ;
2021-08-12 15:04:37 +02:00
import { AccessService } from './access-service' ;
2021-04-16 15:29:23 +02:00
import ResetTokenService from './reset-token-service' ;
import InvalidTokenError from '../error/invalid-token-error' ;
import NotFoundError from '../error/notfound-error' ;
import OwaspValidationError from '../error/owasp-validation-error' ;
import { EmailService } from './email-service' ;
2023-10-06 09:07:06 +02:00
import { IAuthOption , IUnleashConfig } from '../types/option' ;
2021-04-27 09:16:44 +02:00
import SessionService from './session-service' ;
import { IUnleashStores } from '../types/stores' ;
2021-04-27 15:35:10 +02:00
import PasswordUndefinedError from '../error/password-undefined' ;
2023-11-24 16:06:37 +01:00
import {
USER_UPDATED ,
USER_CREATED ,
USER_DELETED ,
UserCreatedEvent ,
UserUpdatedEvent ,
UserDeletedEvent ,
} from '../types/events' ;
2022-06-22 14:55:43 +02:00
import { IUserStore } from '../types/stores/user-store' ;
2021-08-12 15:04:37 +02:00
import { RoleName } from '../types/model' ;
2021-10-29 10:25:42 +02:00
import SettingService from './setting-service' ;
import { SimpleAuthSettings } from '../server-impl' ;
2022-08-26 09:09:48 +02:00
import { simpleAuthSettingsKey } from '../types/settings/simple-auth-settings' ;
2022-01-26 13:45:22 +01:00
import DisabledError from '../error/disabled-error' ;
2022-06-22 14:55:43 +02:00
import BadDataError from '../error/bad-data-error' ;
import { isDefined } from '../util/isDefined' ;
2022-06-22 15:37:26 +02:00
import { TokenUserSchema } from '../openapi/spec/token-user-schema' ;
Clean up old errors (#3633)
This PR attempts to improve the error handling introduced in #3607.
## About the changes
## **tl;dr:**
- Make `UnleashError` constructor protected
- Make all custom errors inherit from `UnleashError`.
- Add tests to ensure that all special error cases include their
relevant data
- Remove `PasswordMismatchError` and `BadRequestError`. These don't
exist.
- Add a few new error types: `ContentTypeError`, `NotImplementedError`,
`UnauthorizedError`
- Remove the `...rest` parameter from error constructor
- Add an unexported `GenericUnleashError` class
- Move OpenAPI conversion function to `BadDataError` clas
- Remove explicit `Error.captureStackTrace`. This is done automatically.
- Extract `getPropFromString` function and add tests
### **In a more verbose fashion**
The main thing is that all our internal errors now inherit
from`UnleashError`. This allows us to simplify the `UnleashError`
constructor and error handling in general while still giving us the
extra benefits we added to that class. However, it _does_ also mean that
I've had to update **all** existing error classes.
The constructor for `UnleashError` is now protected and all places that
called that constructor directly have been updated. Because the base
error isn't available anymore, I've added three new errors to cover use
cases that we didn't already have covered: `NotImplementedError`,
`UnauthorizedError`, `ContentTypeError`. This is to stay consistent in
how we report errors to the user.
There is also an internal class, `GenericUnleashError` that inherits
from the base error. This class is only used in conversions for cases
where we don't know what the error is. It is not exported.
In making all the errors inherit, I've also removed the `...rest`
parameter from the `UnleashError` constructor. We don't need this
anymore.
Following on from the fixes with missing properties in #3638, I have
added tests for all errors that contain extra data.
Some of the error names that were originally used when creating the list
don't exist in the backend. `BadRequestError` and
`PasswordMismatchError` have been removed.
The `BadDataError` class now contains the conversion code for OpenAPI
validation errors. In doing so, I extracted and tested the
`getPropFromString` function.
### Main files
Due to the nature of the changes, there's a lot of files to look at. So
to make it easier to know where to turn your attention:
The changes in `api-error.ts` contain the main changes: protected
constructor, removal of OpenAPI conversion (moved into `BadDataError`.
`api-error.test.ts` contains tests to make sure that errors work as
expected.
Aside from `get-prop-from-string.ts` and the tests, everything else is
just the required updates to go through with the changes.
## Discussion points
I've gone for inheritance of the Error type over composition. This is in
large part because throwing actual Error instances instead of just
objects is preferable (because they collect stack traces, for instance).
However, it's quite possible that we could solve the same thing in a
more elegant fashion using composition.
## For later / suggestions for further improvements
The `api-error` files still contain a lot of code. I think it might be
beneficial to break each Error into a separate folder that includes the
error, its tests, and its schema (if required). It would help decouple
it a bit.
We don't currently expose the schema anywhere, so it's not available in
the openapi spec. We should look at exposing it too.
Finally, it would be good to go through each individual error message
and update each one to be as helpful as possible.
2023-05-11 11:10:57 +02:00
import PasswordMismatch from '../error/password-mismatch' ;
2023-09-27 15:23:05 +02:00
import EventService from './event-service' ;
2021-04-27 20:47:11 +02:00
const systemUser = new User ( { id : - 1 , username : 'system' } ) ;
2021-04-09 13:46:53 +02:00
export interface ICreateUser {
name? : string ;
email? : string ;
username? : string ;
password? : string ;
2021-04-30 13:25:24 +02:00
rootRole : number | RoleName ;
2021-04-09 13:46:53 +02:00
}
export interface IUpdateUser {
id : number ;
name? : string ;
email? : string ;
2021-04-30 13:25:24 +02:00
rootRole? : number | RoleName ;
2021-04-09 13:46:53 +02:00
}
2021-08-23 12:11:29 +02:00
export interface ILoginUserRequest {
email : string ;
name? : string ;
rootRole? : number | RoleName ;
autoCreate? : boolean ;
}
2021-04-09 13:46:53 +02:00
const saltRounds = 10 ;
class UserService {
private logger : Logger ;
2021-08-12 15:04:37 +02:00
private store : IUserStore ;
2021-04-09 13:46:53 +02:00
2023-09-27 15:23:05 +02:00
private eventService : EventService ;
2021-04-27 20:47:11 +02:00
2021-04-09 13:46:53 +02:00
private accessService : AccessService ;
2021-04-16 15:29:23 +02:00
private resetTokenService : ResetTokenService ;
2021-04-27 09:16:44 +02:00
private sessionService : SessionService ;
2021-04-16 15:29:23 +02:00
private emailService : EmailService ;
2021-10-29 10:25:42 +02:00
private settingService : SettingService ;
2022-09-28 10:24:43 +02:00
private passwordResetTimeouts : { [ key : string ] : NodeJS . Timeout } = { } ;
feat: unify error responses (#3607)
This PR implements the first version of a suggested unification (and
documentation) of the errors that we return from the API today.
The goal is for this to be the first step towards the error type defined
in this internal [linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type
'Define the new API error type').
## The state of things today
As things stand, we currently have no (or **very** little) documentation
of the errors that are returned from the API. We mention error codes,
but never what the errors may contain.
Second, there is no specified format for errors, so what they return is
arbitrary, and based on ... Who knows? As a result, we have multiple
different errors returned by the API depending on what operation you're
trying to do. What's more, with OpenAPI validation in the mix, it's
absolutely possible for you to get two completely different error
objects for operations to the same endpoint.
Third, the errors we do return are usually pretty vague and don't really
provide any real help to the user. "You don't have the right
permissions". Great. Well what permissions do I need? And how would I
know? "BadDataError". Sick. Why is it bad?
... You get it.
## What we want to achieve
The ultimate goal is for error messages to serve both humans and
machines. When the user provides bad data, we should tell them what
parts of the data are bad and what they can do to fix it. When they
don't have the right permissions, we should tell them what permissions
they need.
Additionally, it would be nice if we could provide an ID for each error
instance, so that you (or an admin) can look through the logs and locate
he incident.
## What's included in **this** PR?
This PR does not aim to implement everything above. It's not intended to
magically fix everything. Its goal is to implement the necessary
**breaking** changes, so that they can be included in v5. Changing error
messages is a slightly grayer area than changing APIs directly, but
changing the format is definitely something I'd consider breaking.
So this PR:
- defines a minimal version of the error type defined in the [API error
definition linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type).
- aims to catch all errors we return today and wrap them in the error
type
- updates tests to match the new expectations.
An important point: because we are cutting v5 very soon and because work
for this wasn't started until last week, the code here isn't necessarily
very polished. But it doesn't need to be. The internals can be as messy
as we want, as long as the API surface is stable.
That said, I'm very open to feedback about design and code completeness,
etc, but this has intentionally been done quickly.
Please also see my inline comments on the changes for more specific
details.
### Proposed follow-ups
As mentioned, this is the first step to implementing the error type. The
public API error type only exposes `id`, `name`, and `message`. This is
barely any more than most of the previous messages, but they are now all
using the same format. Any additional properties, such as `suggestion`,
`help`, `documentationLink` etc can be added as features without
breaking the current format. This is an intentional limitation of this
PR.
Regarding additional properties: there are some error responses that
must contain extra properties. Some of these are documented in the types
of the new error constructor, but not all. This includes `path` and
`type` properties on 401 errors, `details` on validation errors, and
more.
Also, because it was put together quickly, I don't yet know exactly how
we (as developers) would **prefer** to use these new error messages
within the code, so the internal API (the new type, name, etc), is just
a suggestion. This can evolve naturally over time if (based on feedback
and experience) without changing the public API.
## Returning multiple errors
Most of the time when we return errors today, we only return a single
error (even if many things are wrong). AJV, the OpenAPI integration we
use does have a setting that allows it to return all errors in a request
instead of a single one. I suggest we turn that on, but that we do it in
a separate PR (because it updates a number of other snapshots).
When returning errors that point to `details`, the objects in the
`details` now contain a new `description` property. This "deprecates"
the `message` property. Due to our general deprecation policy, this
should be kept around for another full major and can be removed in v6.
```json
{
"name": "BadDataError",
"message": "Something went wrong. Check the `details` property for more information."
"details": [{
"message": "The .params property must be an object. You provided an array.",
"description": "The .params property must be an object. You provided an array.",
}]
}
```
2023-04-25 15:40:46 +02:00
private baseUriPath : string ;
2021-04-09 13:46:53 +02:00
constructor (
2023-09-27 15:23:05 +02:00
stores : Pick < IUnleashStores , ' userStore ' > ,
2021-04-22 10:07:10 +02:00
{
feat: unify error responses (#3607)
This PR implements the first version of a suggested unification (and
documentation) of the errors that we return from the API today.
The goal is for this to be the first step towards the error type defined
in this internal [linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type
'Define the new API error type').
## The state of things today
As things stand, we currently have no (or **very** little) documentation
of the errors that are returned from the API. We mention error codes,
but never what the errors may contain.
Second, there is no specified format for errors, so what they return is
arbitrary, and based on ... Who knows? As a result, we have multiple
different errors returned by the API depending on what operation you're
trying to do. What's more, with OpenAPI validation in the mix, it's
absolutely possible for you to get two completely different error
objects for operations to the same endpoint.
Third, the errors we do return are usually pretty vague and don't really
provide any real help to the user. "You don't have the right
permissions". Great. Well what permissions do I need? And how would I
know? "BadDataError". Sick. Why is it bad?
... You get it.
## What we want to achieve
The ultimate goal is for error messages to serve both humans and
machines. When the user provides bad data, we should tell them what
parts of the data are bad and what they can do to fix it. When they
don't have the right permissions, we should tell them what permissions
they need.
Additionally, it would be nice if we could provide an ID for each error
instance, so that you (or an admin) can look through the logs and locate
he incident.
## What's included in **this** PR?
This PR does not aim to implement everything above. It's not intended to
magically fix everything. Its goal is to implement the necessary
**breaking** changes, so that they can be included in v5. Changing error
messages is a slightly grayer area than changing APIs directly, but
changing the format is definitely something I'd consider breaking.
So this PR:
- defines a minimal version of the error type defined in the [API error
definition linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type).
- aims to catch all errors we return today and wrap them in the error
type
- updates tests to match the new expectations.
An important point: because we are cutting v5 very soon and because work
for this wasn't started until last week, the code here isn't necessarily
very polished. But it doesn't need to be. The internals can be as messy
as we want, as long as the API surface is stable.
That said, I'm very open to feedback about design and code completeness,
etc, but this has intentionally been done quickly.
Please also see my inline comments on the changes for more specific
details.
### Proposed follow-ups
As mentioned, this is the first step to implementing the error type. The
public API error type only exposes `id`, `name`, and `message`. This is
barely any more than most of the previous messages, but they are now all
using the same format. Any additional properties, such as `suggestion`,
`help`, `documentationLink` etc can be added as features without
breaking the current format. This is an intentional limitation of this
PR.
Regarding additional properties: there are some error responses that
must contain extra properties. Some of these are documented in the types
of the new error constructor, but not all. This includes `path` and
`type` properties on 401 errors, `details` on validation errors, and
more.
Also, because it was put together quickly, I don't yet know exactly how
we (as developers) would **prefer** to use these new error messages
within the code, so the internal API (the new type, name, etc), is just
a suggestion. This can evolve naturally over time if (based on feedback
and experience) without changing the public API.
## Returning multiple errors
Most of the time when we return errors today, we only return a single
error (even if many things are wrong). AJV, the OpenAPI integration we
use does have a setting that allows it to return all errors in a request
instead of a single one. I suggest we turn that on, but that we do it in
a separate PR (because it updates a number of other snapshots).
When returning errors that point to `details`, the objects in the
`details` now contain a new `description` property. This "deprecates"
the `message` property. Due to our general deprecation policy, this
should be kept around for another full major and can be removed in v6.
```json
{
"name": "BadDataError",
"message": "Something went wrong. Check the `details` property for more information."
"details": [{
"message": "The .params property must be an object. You provided an array.",
"description": "The .params property must be an object. You provided an array.",
}]
}
```
2023-04-25 15:40:46 +02:00
server ,
2021-04-22 10:07:10 +02:00
getLogger ,
authentication ,
feat: unify error responses (#3607)
This PR implements the first version of a suggested unification (and
documentation) of the errors that we return from the API today.
The goal is for this to be the first step towards the error type defined
in this internal [linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type
'Define the new API error type').
## The state of things today
As things stand, we currently have no (or **very** little) documentation
of the errors that are returned from the API. We mention error codes,
but never what the errors may contain.
Second, there is no specified format for errors, so what they return is
arbitrary, and based on ... Who knows? As a result, we have multiple
different errors returned by the API depending on what operation you're
trying to do. What's more, with OpenAPI validation in the mix, it's
absolutely possible for you to get two completely different error
objects for operations to the same endpoint.
Third, the errors we do return are usually pretty vague and don't really
provide any real help to the user. "You don't have the right
permissions". Great. Well what permissions do I need? And how would I
know? "BadDataError". Sick. Why is it bad?
... You get it.
## What we want to achieve
The ultimate goal is for error messages to serve both humans and
machines. When the user provides bad data, we should tell them what
parts of the data are bad and what they can do to fix it. When they
don't have the right permissions, we should tell them what permissions
they need.
Additionally, it would be nice if we could provide an ID for each error
instance, so that you (or an admin) can look through the logs and locate
he incident.
## What's included in **this** PR?
This PR does not aim to implement everything above. It's not intended to
magically fix everything. Its goal is to implement the necessary
**breaking** changes, so that they can be included in v5. Changing error
messages is a slightly grayer area than changing APIs directly, but
changing the format is definitely something I'd consider breaking.
So this PR:
- defines a minimal version of the error type defined in the [API error
definition linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type).
- aims to catch all errors we return today and wrap them in the error
type
- updates tests to match the new expectations.
An important point: because we are cutting v5 very soon and because work
for this wasn't started until last week, the code here isn't necessarily
very polished. But it doesn't need to be. The internals can be as messy
as we want, as long as the API surface is stable.
That said, I'm very open to feedback about design and code completeness,
etc, but this has intentionally been done quickly.
Please also see my inline comments on the changes for more specific
details.
### Proposed follow-ups
As mentioned, this is the first step to implementing the error type. The
public API error type only exposes `id`, `name`, and `message`. This is
barely any more than most of the previous messages, but they are now all
using the same format. Any additional properties, such as `suggestion`,
`help`, `documentationLink` etc can be added as features without
breaking the current format. This is an intentional limitation of this
PR.
Regarding additional properties: there are some error responses that
must contain extra properties. Some of these are documented in the types
of the new error constructor, but not all. This includes `path` and
`type` properties on 401 errors, `details` on validation errors, and
more.
Also, because it was put together quickly, I don't yet know exactly how
we (as developers) would **prefer** to use these new error messages
within the code, so the internal API (the new type, name, etc), is just
a suggestion. This can evolve naturally over time if (based on feedback
and experience) without changing the public API.
## Returning multiple errors
Most of the time when we return errors today, we only return a single
error (even if many things are wrong). AJV, the OpenAPI integration we
use does have a setting that allows it to return all errors in a request
instead of a single one. I suggest we turn that on, but that we do it in
a separate PR (because it updates a number of other snapshots).
When returning errors that point to `details`, the objects in the
`details` now contain a new `description` property. This "deprecates"
the `message` property. Due to our general deprecation policy, this
should be kept around for another full major and can be removed in v6.
```json
{
"name": "BadDataError",
"message": "Something went wrong. Check the `details` property for more information."
"details": [{
"message": "The .params property must be an object. You provided an array.",
"description": "The .params property must be an object. You provided an array.",
}]
}
```
2023-04-25 15:40:46 +02:00
} : Pick < IUnleashConfig , ' getLogger ' | ' authentication ' | ' server ' > ,
2021-08-12 15:04:37 +02:00
services : {
accessService : AccessService ;
resetTokenService : ResetTokenService ;
emailService : EmailService ;
2023-09-27 15:23:05 +02:00
eventService : EventService ;
2021-08-12 15:04:37 +02:00
sessionService : SessionService ;
2021-10-29 10:25:42 +02:00
settingService : SettingService ;
2021-08-12 15:04:37 +02:00
} ,
2021-04-09 13:46:53 +02:00
) {
2021-04-22 10:07:10 +02:00
this . logger = getLogger ( 'service/user-service.js' ) ;
2021-04-09 13:46:53 +02:00
this . store = stores . userStore ;
2023-09-27 15:23:05 +02:00
this . eventService = services . eventService ;
2021-08-12 15:04:37 +02:00
this . accessService = services . accessService ;
this . resetTokenService = services . resetTokenService ;
this . emailService = services . emailService ;
this . sessionService = services . sessionService ;
2021-10-29 10:25:42 +02:00
this . settingService = services . settingService ;
2023-10-06 09:07:06 +02:00
if ( authentication . createAdminUser !== false ) {
process . nextTick ( ( ) = >
this . initAdminUser ( {
createAdminUser : authentication.createAdminUser ,
initialAdminUser : authentication.initialAdminUser ,
} ) ,
) ;
2021-04-09 13:46:53 +02:00
}
feat: unify error responses (#3607)
This PR implements the first version of a suggested unification (and
documentation) of the errors that we return from the API today.
The goal is for this to be the first step towards the error type defined
in this internal [linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type
'Define the new API error type').
## The state of things today
As things stand, we currently have no (or **very** little) documentation
of the errors that are returned from the API. We mention error codes,
but never what the errors may contain.
Second, there is no specified format for errors, so what they return is
arbitrary, and based on ... Who knows? As a result, we have multiple
different errors returned by the API depending on what operation you're
trying to do. What's more, with OpenAPI validation in the mix, it's
absolutely possible for you to get two completely different error
objects for operations to the same endpoint.
Third, the errors we do return are usually pretty vague and don't really
provide any real help to the user. "You don't have the right
permissions". Great. Well what permissions do I need? And how would I
know? "BadDataError". Sick. Why is it bad?
... You get it.
## What we want to achieve
The ultimate goal is for error messages to serve both humans and
machines. When the user provides bad data, we should tell them what
parts of the data are bad and what they can do to fix it. When they
don't have the right permissions, we should tell them what permissions
they need.
Additionally, it would be nice if we could provide an ID for each error
instance, so that you (or an admin) can look through the logs and locate
he incident.
## What's included in **this** PR?
This PR does not aim to implement everything above. It's not intended to
magically fix everything. Its goal is to implement the necessary
**breaking** changes, so that they can be included in v5. Changing error
messages is a slightly grayer area than changing APIs directly, but
changing the format is definitely something I'd consider breaking.
So this PR:
- defines a minimal version of the error type defined in the [API error
definition linear
task](https://linear.app/unleash/issue/1-629/define-the-error-type).
- aims to catch all errors we return today and wrap them in the error
type
- updates tests to match the new expectations.
An important point: because we are cutting v5 very soon and because work
for this wasn't started until last week, the code here isn't necessarily
very polished. But it doesn't need to be. The internals can be as messy
as we want, as long as the API surface is stable.
That said, I'm very open to feedback about design and code completeness,
etc, but this has intentionally been done quickly.
Please also see my inline comments on the changes for more specific
details.
### Proposed follow-ups
As mentioned, this is the first step to implementing the error type. The
public API error type only exposes `id`, `name`, and `message`. This is
barely any more than most of the previous messages, but they are now all
using the same format. Any additional properties, such as `suggestion`,
`help`, `documentationLink` etc can be added as features without
breaking the current format. This is an intentional limitation of this
PR.
Regarding additional properties: there are some error responses that
must contain extra properties. Some of these are documented in the types
of the new error constructor, but not all. This includes `path` and
`type` properties on 401 errors, `details` on validation errors, and
more.
Also, because it was put together quickly, I don't yet know exactly how
we (as developers) would **prefer** to use these new error messages
within the code, so the internal API (the new type, name, etc), is just
a suggestion. This can evolve naturally over time if (based on feedback
and experience) without changing the public API.
## Returning multiple errors
Most of the time when we return errors today, we only return a single
error (even if many things are wrong). AJV, the OpenAPI integration we
use does have a setting that allows it to return all errors in a request
instead of a single one. I suggest we turn that on, but that we do it in
a separate PR (because it updates a number of other snapshots).
When returning errors that point to `details`, the objects in the
`details` now contain a new `description` property. This "deprecates"
the `message` property. Due to our general deprecation policy, this
should be kept around for another full major and can be removed in v6.
```json
{
"name": "BadDataError",
"message": "Something went wrong. Check the `details` property for more information."
"details": [{
"message": "The .params property must be an object. You provided an array.",
"description": "The .params property must be an object. You provided an array.",
}]
}
```
2023-04-25 15:40:46 +02:00
this . baseUriPath = server . baseUriPath || '' ;
2021-04-09 13:46:53 +02:00
}
validatePassword ( password : string ) : boolean {
2021-04-27 15:35:10 +02:00
if ( password ) {
const result = owasp . test ( password ) ;
if ( ! result . strong ) {
throw new OwaspValidationError ( result ) ;
} else return true ;
} else {
throw new PasswordUndefinedError ( ) ;
}
2021-04-09 13:46:53 +02:00
}
2023-10-06 09:07:06 +02:00
async initAdminUser (
initialAdminUserConfig : Pick <
IAuthOption ,
'createAdminUser' | 'initialAdminUser'
> ,
) : Promise < void > {
let username : string ;
let password : string ;
if (
initialAdminUserConfig . createAdminUser !== false &&
initialAdminUserConfig . initialAdminUser
) {
username = initialAdminUserConfig . initialAdminUser . username ;
password = initialAdminUserConfig . initialAdminUser . password ;
} else {
username = 'admin' ;
password = 'unleash4all' ;
}
2021-10-12 21:27:06 +02:00
const userCount = await this . store . count ( ) ;
2021-04-09 13:46:53 +02:00
2023-10-06 09:07:06 +02:00
if ( userCount === 0 && username && password ) {
2021-04-09 13:46:53 +02:00
// create default admin user
try {
this . logger . info (
2023-10-06 09:07:06 +02:00
` Creating default user ' ${ username } ' with password ' ${ password } ' ` ,
2021-04-09 13:46:53 +02:00
) ;
2021-04-22 23:40:52 +02:00
const user = await this . store . insert ( {
2023-10-06 09:07:06 +02:00
username ,
2021-04-22 23:40:52 +02:00
} ) ;
2023-10-06 09:07:06 +02:00
const passwordHash = await bcrypt . hash ( password , saltRounds ) ;
2021-04-09 13:46:53 +02:00
await this . store . setPasswordHash ( user . id , passwordHash ) ;
2021-08-12 15:04:37 +02:00
await this . accessService . setUserRootRole (
user . id ,
RoleName . ADMIN ,
) ;
2021-04-09 13:46:53 +02:00
} catch ( e ) {
2023-10-06 09:07:06 +02:00
this . logger . error (
` Unable to create default user ' ${ username } ' ` ,
) ;
2021-04-09 13:46:53 +02:00
}
}
}
2023-11-24 16:06:37 +01:00
async getAll ( ) : Promise < IUserWithRootRole [ ] > {
2021-04-09 13:46:53 +02:00
const users = await this . store . getAll ( ) ;
2023-11-24 16:06:37 +01:00
const defaultRole = await this . accessService . getPredefinedRole (
2021-04-22 10:07:10 +02:00
RoleName . VIEWER ,
) ;
2021-04-09 13:46:53 +02:00
const userRoles = await this . accessService . getRootRoleForAllUsers ( ) ;
2021-08-12 15:04:37 +02:00
const usersWithRootRole = users . map ( ( u ) = > {
const rootRole = userRoles . find ( ( r ) = > r . userId === u . id ) ;
2021-04-09 13:46:53 +02:00
const roleId = rootRole ? rootRole.roleId : defaultRole.id ;
return { . . . u , rootRole : roleId } ;
2023-01-18 13:12:44 +01:00
} ) ;
return usersWithRootRole ;
}
2023-11-24 16:06:37 +01:00
async getUser ( id : number ) : Promise < IUserWithRootRole > {
2021-08-12 15:04:37 +02:00
const user = await this . store . get ( id ) ;
2023-11-24 16:06:37 +01:00
const rootRole = await this . accessService . getRootRoleForUser ( id ) ;
return { . . . user , rootRole : rootRole.id } ;
2021-04-09 13:46:53 +02:00
}
2022-06-22 14:55:43 +02:00
async search ( query : string ) : Promise < IUser [ ] > {
2021-04-09 13:46:53 +02:00
return this . store . search ( query ) ;
}
2021-08-12 15:04:37 +02:00
async getByEmail ( email : string ) : Promise < IUser > {
return this . store . getByQuery ( { email } ) ;
2021-04-16 15:29:23 +02:00
}
2021-04-27 20:47:11 +02:00
async createUser (
{ username , email , name , password , rootRole } : ICreateUser ,
2023-11-06 10:46:59 +01:00
updatedBy? : IUser ,
2023-11-24 16:06:37 +01:00
) : Promise < IUserWithRootRole > {
2022-06-22 14:55:43 +02:00
if ( ! username && ! email ) {
throw new BadDataError ( 'You must specify username or email' ) ;
}
2021-04-09 13:46:53 +02:00
if ( email ) {
Joi . assert ( email , Joi . string ( ) . email ( ) , 'Email' ) ;
}
const exists = await this . store . hasUser ( { username , email } ) ;
if ( exists ) {
2023-07-06 08:24:46 +02:00
throw new BadDataError ( 'User already exists' ) ;
2021-04-09 13:46:53 +02:00
}
2021-04-22 23:40:52 +02:00
const user = await this . store . insert ( {
username ,
email ,
name ,
} ) ;
2021-04-09 13:46:53 +02:00
await this . accessService . setUserRootRole ( user . id , rootRole ) ;
if ( password ) {
const passwordHash = await bcrypt . hash ( password , saltRounds ) ;
await this . store . setPasswordHash ( user . id , passwordHash ) ;
}
2023-11-24 16:06:37 +01:00
const userCreated = await this . getUser ( user . id ) ;
2021-04-27 20:47:11 +02:00
2023-11-24 16:06:37 +01:00
await this . eventService . storeEvent (
new UserCreatedEvent ( {
createdBy : this.getCreatedBy ( updatedBy ) ,
userCreated ,
} ) ,
) ;
return userCreated ;
2021-04-09 13:46:53 +02:00
}
2023-11-06 10:46:59 +01:00
private getCreatedBy ( updatedBy : IUser = systemUser ) {
2021-11-12 13:15:51 +01:00
return updatedBy . username || updatedBy . email ;
}
2021-04-27 20:47:11 +02:00
async updateUser (
{ id , name , email , rootRole } : IUpdateUser ,
2023-11-06 10:46:59 +01:00
updatedBy? : IUser ,
2023-11-24 16:06:37 +01:00
) : Promise < IUserWithRootRole > {
const preUser = await this . getUser ( id ) ;
2021-11-12 13:15:51 +01:00
2022-06-22 14:55:43 +02:00
if ( email ) {
Joi . assert ( email , Joi . string ( ) . email ( ) , 'Email' ) ;
}
2021-04-09 13:46:53 +02:00
if ( rootRole ) {
await this . accessService . setUserRootRole ( id , rootRole ) ;
}
2022-06-22 14:55:43 +02:00
const payload : Partial < IUser > = {
name : name || preUser . name ,
email : email || preUser . email ,
} ;
// Empty updates will throw, so make sure we have something to update.
const user = Object . values ( payload ) . some ( isDefined )
? await this . store . update ( id , payload )
: preUser ;
2021-04-27 20:47:11 +02:00
2023-11-24 16:06:37 +01:00
const storedUser = await this . getUser ( user . id ) ;
2021-04-27 20:47:11 +02:00
2023-11-24 16:06:37 +01:00
await this . eventService . storeEvent (
new UserUpdatedEvent ( {
createdBy : this.getCreatedBy ( updatedBy ) ,
preUser : preUser ,
postUser : storedUser ,
} ) ,
) ;
return storedUser ;
2021-04-09 13:46:53 +02:00
}
2023-11-06 10:46:59 +01:00
async deleteUser ( userId : number , updatedBy? : IUser ) : Promise < void > {
2023-11-24 16:06:37 +01:00
const user = await this . getUser ( userId ) ;
2022-11-23 08:30:54 +01:00
await this . accessService . wipeUserPermissions ( userId ) ;
2021-11-12 13:15:51 +01:00
await this . sessionService . deleteSessionsForUser ( userId ) ;
await this . store . delete ( userId ) ;
2023-11-24 16:06:37 +01:00
await this . eventService . storeEvent (
new UserDeletedEvent ( {
createdBy : this.getCreatedBy ( updatedBy ) ,
deletedUser : user ,
} ) ,
) ;
2021-11-12 13:15:51 +01:00
}
2021-08-12 15:04:37 +02:00
async loginUser ( usernameOrEmail : string , password : string ) : Promise < IUser > {
2021-10-29 10:25:42 +02:00
const settings = await this . settingService . get < SimpleAuthSettings > (
2022-08-26 09:09:48 +02:00
simpleAuthSettingsKey ,
2021-10-29 10:25:42 +02:00
) ;
2022-01-26 13:45:22 +01:00
if ( settings ? . disabled ) {
throw new DisabledError (
2021-10-29 10:25:42 +02:00
'Logging in with username/password has been disabled.' ,
) ;
}
2021-04-09 13:46:53 +02:00
const idQuery = isEmail ( usernameOrEmail )
? { email : usernameOrEmail }
: { username : usernameOrEmail } ;
2023-08-03 08:51:13 +02:00
let user , passwordHash ;
2023-07-04 10:31:54 +02:00
try {
user = await this . store . getByQuery ( idQuery ) ;
2023-08-03 08:51:13 +02:00
passwordHash = await this . store . getPasswordHash ( user . id ) ;
2023-07-04 10:31:54 +02:00
} catch ( error ) { }
2023-08-03 08:51:13 +02:00
if ( user && passwordHash ) {
2023-07-04 10:31:54 +02:00
const match = await bcrypt . compare ( password , passwordHash ) ;
if ( match ) {
await this . store . successfullyLogin ( user ) ;
return user ;
}
2021-04-09 13:46:53 +02:00
}
Clean up old errors (#3633)
This PR attempts to improve the error handling introduced in #3607.
## About the changes
## **tl;dr:**
- Make `UnleashError` constructor protected
- Make all custom errors inherit from `UnleashError`.
- Add tests to ensure that all special error cases include their
relevant data
- Remove `PasswordMismatchError` and `BadRequestError`. These don't
exist.
- Add a few new error types: `ContentTypeError`, `NotImplementedError`,
`UnauthorizedError`
- Remove the `...rest` parameter from error constructor
- Add an unexported `GenericUnleashError` class
- Move OpenAPI conversion function to `BadDataError` clas
- Remove explicit `Error.captureStackTrace`. This is done automatically.
- Extract `getPropFromString` function and add tests
### **In a more verbose fashion**
The main thing is that all our internal errors now inherit
from`UnleashError`. This allows us to simplify the `UnleashError`
constructor and error handling in general while still giving us the
extra benefits we added to that class. However, it _does_ also mean that
I've had to update **all** existing error classes.
The constructor for `UnleashError` is now protected and all places that
called that constructor directly have been updated. Because the base
error isn't available anymore, I've added three new errors to cover use
cases that we didn't already have covered: `NotImplementedError`,
`UnauthorizedError`, `ContentTypeError`. This is to stay consistent in
how we report errors to the user.
There is also an internal class, `GenericUnleashError` that inherits
from the base error. This class is only used in conversions for cases
where we don't know what the error is. It is not exported.
In making all the errors inherit, I've also removed the `...rest`
parameter from the `UnleashError` constructor. We don't need this
anymore.
Following on from the fixes with missing properties in #3638, I have
added tests for all errors that contain extra data.
Some of the error names that were originally used when creating the list
don't exist in the backend. `BadRequestError` and
`PasswordMismatchError` have been removed.
The `BadDataError` class now contains the conversion code for OpenAPI
validation errors. In doing so, I extracted and tested the
`getPropFromString` function.
### Main files
Due to the nature of the changes, there's a lot of files to look at. So
to make it easier to know where to turn your attention:
The changes in `api-error.ts` contain the main changes: protected
constructor, removal of OpenAPI conversion (moved into `BadDataError`.
`api-error.test.ts` contains tests to make sure that errors work as
expected.
Aside from `get-prop-from-string.ts` and the tests, everything else is
just the required updates to go through with the changes.
## Discussion points
I've gone for inheritance of the Error type over composition. This is in
large part because throwing actual Error instances instead of just
objects is preferable (because they collect stack traces, for instance).
However, it's quite possible that we could solve the same thing in a
more elegant fashion using composition.
## For later / suggestions for further improvements
The `api-error` files still contain a lot of code. I think it might be
beneficial to break each Error into a separate folder that includes the
error, its tests, and its schema (if required). It would help decouple
it a bit.
We don't currently expose the schema anywhere, so it's not available in
the openapi spec. We should look at exposing it too.
Finally, it would be good to go through each individual error message
and update each one to be as helpful as possible.
2023-05-11 11:10:57 +02:00
throw new PasswordMismatch (
` The combination of password and username you provided is invalid. If you have forgotten your password, visit ${ this . baseUriPath } /forgotten-password or get in touch with your instance administrator. ` ,
) ;
2021-04-09 13:46:53 +02:00
}
/ * *
* Used to login users without specifying password . Used when integrating
* with external identity providers .
*
* @param usernameOrEmail
* @param autoCreateUser
* @returns
* /
async loginUserWithoutPassword (
email : string ,
autoCreateUser : boolean = false ,
2021-08-12 15:04:37 +02:00
) : Promise < IUser > {
2021-08-23 12:11:29 +02:00
return this . loginUserSSO ( { email , autoCreate : autoCreateUser } ) ;
}
async loginUserSSO ( {
email ,
name ,
rootRole ,
autoCreate = false ,
} : ILoginUserRequest ) : Promise < IUser > {
2021-08-12 15:04:37 +02:00
let user : IUser ;
2021-04-09 13:46:53 +02:00
try {
2021-08-12 15:04:37 +02:00
user = await this . store . getByQuery ( { email } ) ;
2021-08-23 12:11:29 +02:00
// Update user if autCreate is enabled.
2021-08-25 12:43:42 +02:00
if ( name && user . name !== name ) {
2021-08-23 12:11:29 +02:00
user = await this . store . update ( user . id , { name , email } ) ;
}
2021-04-09 13:46:53 +02:00
} catch ( e ) {
2023-10-06 09:07:06 +02:00
// User does not exists. Create if 'autoCreate' is enabled
2021-08-23 12:11:29 +02:00
if ( autoCreate ) {
2021-04-09 13:46:53 +02:00
user = await this . createUser ( {
email ,
2021-08-23 12:11:29 +02:00
name ,
rootRole : rootRole || RoleName . EDITOR ,
2021-04-09 13:46:53 +02:00
} ) ;
} else {
throw e ;
}
}
2022-09-23 14:19:17 +02:00
await this . store . successfullyLogin ( user ) ;
2021-04-09 13:46:53 +02:00
return user ;
}
async changePassword ( userId : number , password : string ) : Promise < void > {
this . validatePassword ( password ) ;
const passwordHash = await bcrypt . hash ( password , saltRounds ) ;
2022-09-23 14:19:17 +02:00
await this . store . setPasswordHash ( userId , passwordHash ) ;
await this . sessionService . deleteSessionsForUser ( userId ) ;
2023-04-05 11:39:52 +02:00
await this . resetTokenService . expireExistingTokensForUser ( userId ) ;
2021-04-09 13:46:53 +02:00
}
2023-06-05 11:58:25 +02:00
async changePasswordWithVerification (
userId : number ,
newPassword : string ,
oldPassword : string ,
) : Promise < void > {
const currentPasswordHash = await this . store . getPasswordHash ( userId ) ;
const match = await bcrypt . compare ( oldPassword , currentPasswordHash ) ;
if ( ! match ) {
throw new PasswordMismatch (
` The old password you provided is invalid. If you have forgotten your password, visit ${ this . baseUriPath } /forgotten-password or get in touch with your instance administrator. ` ,
) ;
}
await this . changePassword ( userId , newPassword ) ;
}
2022-06-22 15:37:26 +02:00
async getUserForToken ( token : string ) : Promise < TokenUserSchema > {
2023-11-27 13:42:58 +01:00
const { createdBy , userId } =
await this . resetTokenService . isValid ( token ) ;
2021-04-16 15:29:23 +02:00
const user = await this . getUser ( userId ) ;
2022-01-13 11:14:17 +01:00
const role = await this . accessService . getRoleData ( user . rootRole ) ;
2021-04-16 15:29:23 +02:00
return {
token ,
createdBy ,
email : user.email ,
name : user.name ,
id : user.id ,
role : {
2022-06-22 14:55:43 +02:00
id : user.rootRole ,
2021-04-16 15:29:23 +02:00
description : role.role.description ,
type : role . role . type ,
name : role.role.name ,
} ,
} ;
}
2021-04-27 09:16:44 +02:00
/ * *
* If the password is a strong password will update password and delete all sessions for the user we ' re changing the password for
* @param token - the token authenticating this request
* @param password - new password
* /
2021-04-16 15:29:23 +02:00
async resetPassword ( token : string , password : string ) : Promise < void > {
this . validatePassword ( password ) ;
const user = await this . getUserForToken ( token ) ;
const allowed = await this . resetTokenService . useAccessToken ( {
userId : user.id ,
token ,
} ) ;
if ( allowed ) {
await this . changePassword ( user . id , password ) ;
} else {
throw new InvalidTokenError ( ) ;
}
}
async createResetPasswordEmail (
receiverEmail : string ,
2023-11-06 10:46:59 +01:00
user : IUser = systemUser ,
2021-04-16 15:29:23 +02:00
) : Promise < URL > {
const receiver = await this . getByEmail ( receiverEmail ) ;
if ( ! receiver ) {
throw new NotFoundError ( ` Could not find ${ receiverEmail } ` ) ;
}
2022-09-28 10:24:43 +02:00
if ( this . passwordResetTimeouts [ receiver . id ] ) {
return ;
}
2021-04-16 15:29:23 +02:00
const resetLink = await this . resetTokenService . createResetPasswordUrl (
receiver . id ,
2021-04-27 20:47:11 +02:00
user . username || user . email ,
2021-04-16 15:29:23 +02:00
) ;
2022-09-28 10:24:43 +02:00
this . passwordResetTimeouts [ receiver . id ] = setTimeout ( ( ) = > {
delete this . passwordResetTimeouts [ receiver . id ] ;
} , 1000 * 60 ) ; // 1 minute
2021-04-16 15:29:23 +02:00
await this . emailService . sendResetMail (
receiver . name ,
receiver . email ,
resetLink . toString ( ) ,
) ;
return resetLink ;
}
2021-04-09 13:46:53 +02:00
}
export default UserService ;