Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add new use cases to authorize user, register SPs, and get id tokens #22

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 51 additions & 0 deletions server/scripts/approve_client.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
from configparser import ConfigParser
from datetime import datetime
import sys
import base64
import psycopg2

def approve_client():
if(len(sys.argv)!=2):
print("Invalid number of cmd line arguments provided.")
client_id = base64.b64decode(sys.argv[1])
Comment on lines +8 to +10
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no need to reinvent the wheel, let's use argparse instead

with conn.cursor() as cursor:
cursor.execute("UPDATE auth_secret SET is_verified = 1 WHERE auth_secret.client_id = {client_id}".format(
client_id=client_id
))
conn.commit()
print("Auth Secret updated")


def connect():
conn = None
print("Connecting to PostgreSQL server...")

parser = ConfigParser()
with open("db.ini") as f:
parser.read_file(f)

keys = parser["postgresql"]
try:
conn = psycopg2.connect(
host=keys.get("host"),
database=keys.get("database"),
user=keys.get("user"),
password=keys.get("password"))
print("Connection Successful")
cursor = conn.cursor()
cursor.execute("SELECT version()")
print(cursor.fetchone())
return conn
except (Exception, psycopg2.DatabaseError) as error:
print(error)
if conn is not None:
conn.close()
print("Connection Closed")


conn = connect()

if conn is None:
exit()

approve_client()
1 change: 1 addition & 0 deletions server/scripts/requirements.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
psycopg2==2.8.6
12 changes: 12 additions & 0 deletions server/src/migrations/Migration20210829055229.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import { Migration } from '@mikro-orm/migrations';

export class Migration20210829055229 extends Migration {

async up(): Promise<void> {
this.addSql('alter table "user" drop column "access_token";');
this.addSql('alter table "user" drop column "refresh_token";');

this.addSql('alter table "auth_secret" add column "decoded_redirect_uri" varchar(255) not null, add column "client_name" varchar(255) not null;');
}

}
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
export namespace AuthenticateUserErrors {
export class AuthenticationFailedError {
public message: string
public constructor(email: string, message: string) {
this.message = `Authentication for user with ${email} failed: ${message}`
}
export class AuthenticationFailedError extends Error {
public constructor(email: string, message: string) {
super()
this.message = `Authentication for user with ${email} failed: ${message}`
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,14 @@ import { AuthenticateUserDTO } from './authenticate-user-dto'
import { AuthenticateUserErrors } from './authenticate-user-errors'

type AuthenticateUserUseCaseError =
AuthenticateUserErrors.AuthenticationFailedError
| AuthenticateUserErrors.AuthenticationFailedError
| AppError.UnexpectedError

export type AuthenticateUserUseCaseResponse = Result<User, AuthenticateUserUseCaseError>

export class AuthenticateUserUseCase
implements UseCaseWithDTO<AuthenticateUserDTO, AuthenticateUserUseCaseResponse> {
implements UseCaseWithDTO<AuthenticateUserDTO, AuthenticateUserUseCaseResponse>
{
private userRepo: UserRepo

constructor(userRepo: UserRepo) {
Expand All @@ -37,14 +38,18 @@ export class AuthenticateUserUseCase
const email = results[0].value
const password = results[1].value

try {
const userByEmailAndPassword = await this.userRepo.getUserByUserEmailandUserPassword(email, password)
if (userByEmailAndPassword.isErr()) {
return Result.err(new AuthenticateUserErrors.AuthenticationFailedError(email.value, userByEmailAndPassword.error.message))
}
return Result.ok(userByEmailAndPassword.value)
} catch (err) {
return Result.err(new AppError.UnexpectedError(err))
const userByEmailAndPassword = await this.userRepo.getUserByUserEmailandUserPassword(
email,
password
)
if (userByEmailAndPassword.isErr()) {
return Result.err(
new AuthenticateUserErrors.AuthenticationFailedError(
email.value,
userByEmailAndPassword.error.message
)
)
}
return Result.ok(userByEmailAndPassword.value)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
import express from 'express'
import httpMocks from 'node-mocks-http'
import { AppError } from '../../../../../../shared/core/app-error'
import { Result } from '../../../../../../shared/core/result'
import { AuthorizeUserDTO } from '../authorize-user-dto'
import { AuthorizeUserErrors } from '../authorize-user-errors'
import { AuthorizeUserUseCase } from '../authorize-user-use-case'
import { AuthorizeUserController } from '../authorize-user-controller'
import { mocks } from '../../../../../../test-utils'
import { ParamList, ParamPair } from '../../../../../../shared/app/param-list'

describe('AuthorizeUserController', () => {
let authorizeUserDTO: AuthorizeUserDTO
let authorizeUserUseCase: AuthorizeUserUseCase
let authorizeUserController: AuthorizeUserController
let mockResponse: express.Response

beforeAll(async () => {
const authorizeUser = await mocks.mockAuthorizeUser()
authorizeUserController = authorizeUser.authorizeUserController
authorizeUserUseCase = authorizeUser.authorizeUserUseCase
mockResponse = httpMocks.createResponse()
authorizeUserDTO = {
req: httpMocks.createRequest(),
params: {
client_id: '6a88757bceaddaf03540dbd891dfb828',
response_type: 'code',
redirect_uri: 'www.loolabs.org',
scope: 'openid',
},
}
})

test('When the AuthorizeUserUseCase returns Ok, the AuthorizeUserController returns 302 Redirect', async () => {
const useCaseResolvedValue = {
redirectParams: new ParamList([new ParamPair('type', 'test')]),
redirectUrl: 'www.loolabs.org',
}
jest.spyOn(authorizeUserUseCase, 'execute').mockResolvedValue(Result.ok(useCaseResolvedValue))

const result = await authorizeUserController.executeImpl(authorizeUserDTO, mockResponse)
expect(result.statusCode).toBe(302)
})

test('When the AuthorizeUserUseCase returns AuthorizeUserErrors.InvalidRequestParameters, AuthorizeUserController returns 400 Bad Request', async () => {
jest
.spyOn(authorizeUserUseCase, 'execute')
.mockResolvedValue(Result.err(new AuthorizeUserErrors.InvalidRequestParameters()))

const result = await authorizeUserController.executeImpl(authorizeUserDTO, mockResponse)

expect(result.statusCode).toBe(400)
})

test('When the AuthorizeUserUseCase returns AuthorizeUserErrors.UserNotAuthenticated, AuthorizeUserController returns 302 Redirect', async () => {
const useCaseErrorValue = {
redirectParams: new ParamList([new ParamPair('type', 'test')]),
redirectUrl: 'www.loolabs.org',
}
jest.spyOn(authorizeUserUseCase, 'execute').mockResolvedValue(Result.err(useCaseErrorValue))

const result = await authorizeUserController.executeImpl(authorizeUserDTO, mockResponse)
expect(result.statusCode).toBe(302)
})

test('When the AuthorizeUserUseCase returns AppError.UnexpectedError, AuthorizeUserController returns 500 Internal Server Error', async () => {
jest
.spyOn(authorizeUserUseCase, 'execute')
.mockResolvedValue(Result.err(new AppError.UnexpectedError('Unexpected error')))

const result = await authorizeUserController.executeImpl(authorizeUserDTO, mockResponse)

expect(result.statusCode).toBe(500)
})
})
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import express from 'express'
import { ControllerWithDTO } from '../../../../../shared/app/controller-with-dto'
import { AuthorizeUserUseCase } from './authorize-user-use-case'
import { AuthorizeUserDTO, AuthorizeUserDTOSchema } from './authorize-user-dto'
import { AuthorizeUserErrors } from './authorize-user-errors'
import { Result } from '../../../../../shared/core/result'
import { ValidationError } from 'joi'

export class AuthorizeUserController extends ControllerWithDTO<AuthorizeUserUseCase> {
constructor(useCase: AuthorizeUserUseCase) {
super(useCase)
}

buildDTO(req: express.Request): Result<AuthorizeUserDTO, Array<ValidationError>> {
let params: any = req.params
const errs: Array<ValidationError> = []
const compiledRequest = {
req,
params,
}
const bodyResult = this.validate(compiledRequest, AuthorizeUserDTOSchema)
if (bodyResult.isOk()) {
const body = bodyResult.value
return Result.ok(body)
} else {
errs.push(bodyResult.error)
return Result.err(errs)
}
}

async executeImpl<Res extends express.Response>(dto: AuthorizeUserDTO, res: Res): Promise<Res> {
try {
const result = await this.useCase.execute(dto)

if (result.isOk()) {
return this.redirect(res, result.value.redirectUrl, result.value.redirectParams)
} else {
const error = result.error
if ('redirectParams' in error) {
return this.redirect(res, error.redirectUrl, error.redirectParams)
}
switch (error.constructor) {
case AuthorizeUserErrors.InvalidRequestParameters:
return this.clientError(res, error.message)
default:
return this.fail(res, error.message)
}
}
} catch (err) {
return this.fail(res, err)
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import Joi from 'joi'
import express from 'express'

export const SUPPORTED_OPEN_ID_RESPONSE_TYPES = ['code']
export const SUPPORTED_OPEN_ID_SCOPE = ['openid']

export interface AuthorizeUserDTOParams {
client_id: string
scope: string
response_type: string
redirect_uri: string
}

export interface AuthorizeUserDTO {
req: express.Request
params: AuthorizeUserDTOParams
}

export const AuthorizeUserDTOParamsSchema = Joi.object<AuthorizeUserDTOParams>({
client_id: Joi.string().required(),
scope: Joi.string()
.valid(...SUPPORTED_OPEN_ID_SCOPE)
.required(),
response_type: Joi.string()
.valid(...SUPPORTED_OPEN_ID_RESPONSE_TYPES)
.required(),
redirect_uri: Joi.string().uri().required(),
}).options({ abortEarly: false })

export const AuthorizeUserDTOSchema = Joi.object<AuthorizeUserDTO>({
req: Joi.object().required(),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I understand correctly, the req object is only needed for its .user field. Can we narrow this part of the schema then?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The issue is that by narrowing the schema at this point, I wouldn't be able to redirect the user to /login if the user doesn't exist, like done in the use-case (we return a 400 for all schema errors currently). The overridable method you showed me earlier would let me change that. Is that something I should include here?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since loolabs/waterpark#221 is merged now, feel free to copy over https://github.com/loolabs/waterpark/blob/main/server/src/shared/app/typed-controller.ts and its dependencies to try it out. A word of warning -- it uses Zod. You might be able to modify it to use Joi validation.

params: AuthorizeUserDTOParamsSchema.optional(), // this ensures that all of the necessary request params for client authentication are present, not just an insufficient subset
}).options({ abortEarly: false })
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
export namespace AuthorizeUserErrors {
export class InvalidRequestParameters extends Error {
public constructor() {
super(`Invalid openid request parameters supplied.`)
}
}
export class UserNotAuthenticated extends Error {
public constructor(email: string) {
super(`The user with email ${email} is not authenticated.`)
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
import { UseCaseWithDTO } from '../../../../../shared/app/use-case-with-dto'
import { AppError } from '../../../../../shared/core/app-error'
import { Result } from '../../../../../shared/core/result'
import { AuthorizeUserDTO } from './authorize-user-dto'
import { AuthorizeUserErrors } from './authorize-user-errors'
import { ParamList, ParamPair } from '../../../../../shared/app/param-list'
import { AuthCodeRepo } from '../../../infra/repos/auth-code-repo/auth-code-repo'
import { AuthSecretRepo } from '../../../infra/repos/auth-secret-repo/auth-secret-repo'
import { AuthCode } from '../../../domain/entities/auth-code'
import { AuthCodeString } from '../../../domain/value-objects/auth-code-string'
import { User } from '../../../domain/entities/user'

export type AuthorizeUserUseCaseClientError =
| AuthorizeUserErrors.InvalidRequestParameters
| AppError.UnexpectedError

export type AuthorizeUserUseCaseRedirectError = {
redirectParams: ParamList
redirectUrl: string
}

export type AuthorizeUserUseCaseError =
| AuthorizeUserUseCaseClientError
| AuthorizeUserUseCaseRedirectError

export interface AuthorizeUserSuccess {
redirectParams: ParamList
redirectUrl: string
}

export type AuthorizeUserUseCaseResponse = Result<AuthorizeUserSuccess, AuthorizeUserUseCaseError>

export class AuthorizeUserUseCase
implements UseCaseWithDTO<AuthorizeUserDTO, AuthorizeUserUseCaseResponse>
{
constructor(private authCodeRepo: AuthCodeRepo, private authSecretRepo: AuthSecretRepo) {}

async execute(dto: AuthorizeUserDTO): Promise<AuthorizeUserUseCaseResponse> {
const params = dto.params
const decodedUri = decodeURI(params.redirect_uri)
const authSecretExists = await this.authSecretRepo.exists(params.client_id, decodedUri)
if (authSecretExists.isErr() || authSecretExists.value === false) {
return Result.err(new AuthorizeUserErrors.InvalidRequestParameters())
}
const user = dto.req.user as User
if (user === undefined) {
const redirectParams = new ParamList(
Object.entries(params).map((paramPair) => new ParamPair(paramPair[0], paramPair[1]))
)
return Result.err({
redirectParams,
redirectUrl: `${process.env.PUBLIC_HOST}/login`,
})
}
const authCode = AuthCode.create({
clientId: params.client_id,
userId: user.userId.id.toString(),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

userId.id? I think we changed something in waterpark to reduce the level of object nesting here. cc @KTong821

userEmail: user.email.value,
userEmailVerified: user.isEmailVerified || false,
authCodeString: new AuthCodeString(),
})
if (authCode.isErr()) {
return Result.err(new AppError.UnexpectedError('Authcode creation failed'))
}
await this.authCodeRepo.save(authCode.value)
const redirectParams = new ParamList([
new ParamPair('code', authCode.value.authCodeString.getValue()),
])
const AuthorizeUserSuccessResponse: AuthorizeUserSuccess = {
redirectParams: redirectParams,
redirectUrl: params.redirect_uri,
}

return Result.ok(AuthorizeUserSuccessResponse)
}
}
Loading