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

feat(auth): add authentication endpoints #6265

Merged
merged 83 commits into from
Feb 2, 2024
Merged
Show file tree
Hide file tree
Changes from 75 commits
Commits
Show all changes
83 commits
Select commit Hold shift + click to select a range
dacdb27
initial implementation
pKorsholm Jan 26, 2024
cd38e4c
add test for invalid scope
pKorsholm Jan 26, 2024
d3fd771
get config from scope not db
pKorsholm Jan 26, 2024
512f5df
assign config from scope
pKorsholm Jan 26, 2024
908e5e6
fix package.json
pKorsholm Jan 26, 2024
4bd6491
optional providers
pKorsholm Jan 26, 2024
9869c71
make providers options
pKorsholm Jan 26, 2024
36d51ef
rename auth to authentication
pKorsholm Jan 26, 2024
7553ea0
more renaming
pKorsholm Jan 26, 2024
ecd2250
update package name
pKorsholm Jan 26, 2024
333a268
add create-customer and customers/me routes
pKorsholm Jan 26, 2024
036e577
add changeset
pKorsholm Jan 26, 2024
9d49a5d
rm auth endpoints from branch
pKorsholm Jan 26, 2024
99c8e2d
cleanup for pr
pKorsholm Jan 26, 2024
94cf842
add fields to customer creation endpoint
pKorsholm Jan 26, 2024
e6baee6
add fields to customer creation endpoint
pKorsholm Jan 26, 2024
76ca08e
delete file
pKorsholm Jan 26, 2024
0418435
update query-config
pKorsholm Jan 26, 2024
fe2c5c3
rename steps
pKorsholm Jan 29, 2024
d8e5415
add create-customer and customers/me routes
pKorsholm Jan 26, 2024
9c6a487
add changeset
pKorsholm Jan 26, 2024
5cc7742
rm auth endpoints from branch
pKorsholm Jan 26, 2024
44dbffd
cleanup for pr
pKorsholm Jan 26, 2024
d33f05d
add fields to customer creation endpoint
pKorsholm Jan 26, 2024
e4b0808
add fields to customer creation endpoint
pKorsholm Jan 26, 2024
560162a
delete file
pKorsholm Jan 26, 2024
2742d5d
update query-config
pKorsholm Jan 26, 2024
b92d8a0
rename steps
pKorsholm Jan 29, 2024
e6a4d59
add auth folders
pKorsholm Jan 26, 2024
58144bb
add middlewares
pKorsholm Jan 29, 2024
5e04717
case as passport
pKorsholm Jan 29, 2024
21d165d
make callback functions arrow-functions and cast "this"
pKorsholm Jan 29, 2024
a51c569
Merge branch 'develop' into feat/add-create-customer-route-to-v2
pKorsholm Jan 29, 2024
6b9c832
rename scope
pKorsholm Jan 29, 2024
96e12e6
update providers for core integration
pKorsholm Jan 29, 2024
6856205
add auth handlers
pKorsholm Jan 29, 2024
9b05eda
add initial middlewares for authentication
pKorsholm Jan 29, 2024
f4b02a0
add scope and medusa_id types to session type
pKorsholm Jan 29, 2024
b58d304
update yarn.lock
pKorsholm Jan 29, 2024
48f41e2
feat: create/update/delete customers
srindom Jan 29, 2024
099f1e0
fix: cleanup directories
srindom Jan 29, 2024
3e0e16d
fix: typo
srindom Jan 29, 2024
6b91e87
fix: faulty export
srindom Jan 29, 2024
201a690
fix: missing export
srindom Jan 29, 2024
6b494a6
Merge remote-tracking branch 'origin/feat/add-create-customer-route-t…
srindom Jan 29, 2024
a6c69e2
create user if no user exists
pKorsholm Jan 30, 2024
1e29439
update creation type for auth-users
pKorsholm Jan 30, 2024
e7e5fde
create-customers route
pKorsholm Jan 30, 2024
7d90532
update provider types
pKorsholm Jan 30, 2024
753e59c
create initial middlewares
pKorsholm Jan 30, 2024
671404f
touch up
pKorsholm Jan 30, 2024
94cc17f
Merge branch 'develop' into feat/add-authentication-endpoints-for-v2
pKorsholm Jan 30, 2024
a8132f4
rm legacy provider
pKorsholm Jan 30, 2024
4f5753d
update middlewares
pKorsholm Jan 30, 2024
dc6e669
feat: poc middleware
srindom Jan 30, 2024
68cd24c
Merge branch 'feat/customer-store' of github.com:medusajs/medusa into…
srindom Jan 30, 2024
decb5db
fix: cleanup
srindom Jan 30, 2024
2afcae8
fix: cleanup
srindom Jan 30, 2024
1fda3f2
feat: /store/customers endpoints
srindom Jan 30, 2024
3e87fea
Merge remote-tracking branch 'origin/develop' into feat/customer-store
srindom Jan 30, 2024
4d61cdd
Merge branch 'develop' into feat/add-authentication-endpoints-for-v2
pKorsholm Jan 31, 2024
06f2adc
Merge branch 'feat/customer-store' into feat/add-authentication-endpo…
pKorsholm Jan 31, 2024
91d46ce
new middleware changes
pKorsholm Jan 31, 2024
42b3a6d
update auth middleware
pKorsholm Jan 31, 2024
ee398a6
Merge branch 'feat/customer-store' into feat/add-authentication-endpo…
pKorsholm Jan 31, 2024
d43cdfb
rm unused files
pKorsholm Jan 31, 2024
eeffa8e
revert changes
pKorsholm Jan 31, 2024
e5715b1
revert changes
pKorsholm Jan 31, 2024
36a6d83
import authentication middleware correctly
pKorsholm Jan 31, 2024
fe6bb3d
rm route
pKorsholm Jan 31, 2024
d0d64ea
cleanup for pr
pKorsholm Jan 31, 2024
bf2ba61
Merge branch 'develop' into feat/add-authentication-endpoints-for-v2
pKorsholm Feb 1, 2024
bab06a2
revert changes
pKorsholm Feb 1, 2024
c837d7a
update request type
pKorsholm Feb 1, 2024
1ecebf4
fix auth integration tests
pKorsholm Feb 1, 2024
be95dce
fix integration tests
pKorsholm Feb 1, 2024
ec43f6f
Feat(auth): Add user scope field (#6288)
pKorsholm Feb 1, 2024
5acba3b
pr feedback
pKorsholm Feb 1, 2024
0937f31
Update packages/medusa/src/api-v2/store/customers/route.ts
pKorsholm Feb 1, 2024
237a802
Update packages/auth/src/models/auth-user.ts
pKorsholm Feb 1, 2024
2f9c25b
throw if unauthorized
pKorsholm Feb 1, 2024
8c41abb
pr feedback
pKorsholm Feb 1, 2024
63adf0b
Merge branch 'develop' into feat/add-authentication-endpoints-for-v2
pKorsholm Feb 2, 2024
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
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ describe("AuthModuleService - AuthProvider", () => {
const { success, error } = await service.authenticate(
"emailpass",
{
scope: "non-existing",
authScope: "non-existing",
} as any
)

Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { AuthenticationInput, IAuthModuleService } from "@medusajs/types"
import { MedusaModule, Modules } from "@medusajs/modules-sdk"

import { IAuthModuleService } from "@medusajs/types"
import { MikroOrmWrapper } from "../../../utils"
import Scrypt from "scrypt-kdf"
import { SqlEntityManager } from "@mikro-orm/postgresql"
Expand Down Expand Up @@ -73,8 +73,8 @@ describe("AuthModuleService - AuthProvider", () => {
email: "test@test.com",
password: password,
},
scope: "store",
})
authScope: "store",
} as any)

expect(res).toEqual({
success: true,
Expand All @@ -92,8 +92,8 @@ describe("AuthModuleService - AuthProvider", () => {

const res = await service.authenticate("emailpass", {
body: { email: "test@test.com" },
scope: "store",
})
authScope: "store",
} as any)

expect(res).toEqual({
success: false,
Expand All @@ -106,8 +106,8 @@ describe("AuthModuleService - AuthProvider", () => {

const res = await service.authenticate("emailpass", {
body: { password: "supersecret" },
scope: "store",
})
authScope: "store",
} as any)

expect(res).toEqual({
success: false,
Expand Down Expand Up @@ -139,8 +139,8 @@ describe("AuthModuleService - AuthProvider", () => {
email: "test@test.com",
password: "password",
},
scope: "store",
})
authScope: "store",
} as any)

expect(res).toEqual({
success: false,
Expand Down
2 changes: 1 addition & 1 deletion packages/auth/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@
"@mikro-orm/migrations": "5.9.7",
"@mikro-orm/postgresql": "5.9.7",
"awilix": "^8.0.0",
"dotenv": "^16.1.4",
"dotenv": "16.3.1",
"jsonwebtoken": "^9.0.2",
"knex": "2.4.2",
"scrypt-kdf": "^2.0.1",
Expand Down
49 changes: 43 additions & 6 deletions packages/auth/src/providers/email-password.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { AbstractAuthModuleProvider, isString } from "@medusajs/utils"
import { AbstractAuthModuleProvider, MedusaError, isString } from "@medusajs/utils"
import { AuthenticationInput, AuthenticationResponse } from "@medusajs/types"

import { AuthUserService } from "@services"
Expand All @@ -16,6 +16,14 @@ class EmailPasswordProvider extends AbstractAuthModuleProvider {
this.authUserSerivce_ = authUserService
}

private getHashConfig(scope: string) {
const scopeConfig = this.scopes_[scope].hashConfig as
| Scrypt.ScryptParams
riqwan marked this conversation as resolved.
Show resolved Hide resolved
| undefined

return scopeConfig ?? { logN: 15, r: 8, p: 1 }
}
pKorsholm marked this conversation as resolved.
Show resolved Hide resolved

async authenticate(
userData: AuthenticationInput
): Promise<AuthenticationResponse> {
Expand All @@ -34,11 +42,40 @@ class EmailPasswordProvider extends AbstractAuthModuleProvider {
error: "Email should be a string",
}
}

const authUser = await this.authUserSerivce_.retrieveByProviderAndEntityId(
email,
EmailPasswordProvider.PROVIDER
)
let authUser

try {
authUser = await this.authUserSerivce_.retrieveByProviderAndEntityId(
email,
EmailPasswordProvider.PROVIDER
)
} catch (error) {
if (error.type === MedusaError.Types.NOT_FOUND) {
const password_hash = await Scrypt.kdf(
pKorsholm marked this conversation as resolved.
Show resolved Hide resolved
password,
this.getHashConfig(userData.authScope)
)

const [createdAuthUser] = await this.authUserSerivce_.create([
{
entity_id: email,
provider: EmailPasswordProvider.PROVIDER,
app_metadata: {
scope: userData.authScope,
},
provider_metadata: {
password: password_hash.toString("base64"),
},
},
])

return {
success: true,
authUser: JSON.parse(JSON.stringify(createdAuthUser)),
}
}
return { success: false, error: error.message }
}

const password_hash = authUser.provider_metadata?.password

Expand Down
31 changes: 13 additions & 18 deletions packages/auth/src/providers/google.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ class GoogleProvider extends AbstractAuthModuleProvider {

const code = req.query?.code ?? req.body?.code

return await this.validateCallbackToken(code, req.scope, config)
return await this.validateCallbackToken(code, req.authScope, config)
}

// abstractable
Expand All @@ -97,14 +97,15 @@ class GoogleProvider extends AbstractAuthModuleProvider {
)
} catch (error) {
if (error.type === MedusaError.Types.NOT_FOUND) {
authUser = await this.authUserSerivce_.create([
const [createdAuthUser] = await this.authUserSerivce_.create([
{
entity_id,
provider_id: GoogleProvider.PROVIDER,
provider: GoogleProvider.PROVIDER,
user_metadata: jwtData!.payload,
app_metadata: { scope },
},
])
authUser = createdAuthUser
} else {
return { success: false, error: error.message }
}
Expand All @@ -120,12 +121,12 @@ class GoogleProvider extends AbstractAuthModuleProvider {
{ clientID, callbackURL, clientSecret }: ProviderConfig
) {
const client = this.getAuthorizationCodeHandler({ clientID, clientSecret })

const tokenParams = {
code,
redirect_uri: callbackURL,
}

try {
const accessToken = await client.getToken(tokenParams)

Expand All @@ -135,24 +136,18 @@ class GoogleProvider extends AbstractAuthModuleProvider {
}
}

private getConfigFromScope(config: AuthProviderScope): ProviderConfig {
const providerConfig: Partial<ProviderConfig> = {}
private getConfigFromScope(config: AuthProviderScope & Partial<ProviderConfig>): ProviderConfig {
const providerConfig: Partial<ProviderConfig> = { ...config }

if (config.clientId) {
providerConfig.clientID = config.clientId
} else {
if (!providerConfig.clientID) {
throw new Error("Google clientID is required")
}
}

if (config.clientSecret) {
providerConfig.clientSecret = config.clientSecret
} else {
if (!providerConfig.clientSecret) {
throw new Error("Google clientSecret is required")
}

if (config.callbackURL) {
providerConfig.callbackURL = config.callbackUrl
} else {
if (!providerConfig.callbackURL) {
throw new Error("Google callbackUrl is required")
}

Expand All @@ -173,7 +168,7 @@ class GoogleProvider extends AbstractAuthModuleProvider {
): Promise<ProviderConfig> {
await this.authProviderService_.retrieve(GoogleProvider.PROVIDER)

const scopeConfig = this.scopes_[req.scope]
const scopeConfig = this.scopes_[req.authScope]

const config = this.getConfigFromScope(scopeConfig)

Expand Down
4 changes: 2 additions & 2 deletions packages/auth/src/services/auth-module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -400,7 +400,7 @@ export default class AuthModuleService<

protected getRegisteredAuthenticationProvider(
provider: string,
{ scope }: AuthenticationInput
{ authScope }: AuthenticationInput
): AbstractAuthModuleProvider {
let containerProvider: AbstractAuthModuleProvider
try {
Expand All @@ -412,7 +412,7 @@ export default class AuthModuleService<
)
}

containerProvider.validateScope(scope)
containerProvider.validateScope(authScope)

return containerProvider
}
Expand Down
2 changes: 1 addition & 1 deletion packages/auth/src/types/services/auth-user.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ export type AuthUserDTO = {

export type CreateAuthUserDTO = {
entity_id: string
provider_id: string
provider: string
provider_metadata?: Record<string, unknown>
user_metadata?: Record<string, unknown>
app_metadata?: Record<string, unknown>
Expand Down
2 changes: 1 addition & 1 deletion packages/customer/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@
"@mikro-orm/migrations": "5.9.7",
"@mikro-orm/postgresql": "5.9.7",
"awilix": "^8.0.0",
"dotenv": "^16.1.4",
"dotenv": "16.3.1",
"knex": "2.4.2"
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { MedusaRequest, MedusaResponse } from "../../../../../types/routing"
import { AuthenticationInput, IAuthModuleService } from "@medusajs/types"
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { MedusaError } from "@medusajs/utils"

export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const { scope, authProvider } = req.params

const service: IAuthModuleService = req.scope.resolve(
ModuleRegistrationName.AUTH
)

const authData = { ...req } as unknown as AuthenticationInput
pKorsholm marked this conversation as resolved.
Show resolved Hide resolved
authData.authScope = scope

const authResult = await service.validateCallback(authProvider, authData)

const { success, error, authUser, location } = authResult
if (location) {
Copy link
Member

Choose a reason for hiding this comment

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

if there is a location (the redirect could be configured) then we don't check any success or failure?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

yes, we could validate to ensure that success is required if you think that's appropriate? 😄 just figured that in case of a redirect you would want that no matter what

Copy link
Member

Choose a reason for hiding this comment

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

in general you have a success redirect and a failure redirect that eventually could have custom query parameters, let see that later

res.redirect(location)
Copy link
Contributor

Choose a reason for hiding this comment

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

question: shouldn't the client be the one doing the redirect?

return
}

if (!success && error) {
throw new MedusaError(MedusaError.Types.UNAUTHORIZED, error)
} else if (success) {
req.session.authUser = authUser
req.session.scope = authUser.app_metadata.scope

res.status(200).json({ authUser })
}
}

export const POST = async (req: MedusaRequest, res: MedusaResponse) => {
await GET(req, res)
}
36 changes: 36 additions & 0 deletions packages/medusa/src/api-v2/auth/[scope]/[authProvider]/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { MedusaRequest, MedusaResponse } from "../../../../types/routing"
import { AuthenticationInput, IAuthModuleService } from "@medusajs/types"
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { MedusaError } from "@medusajs/utils"

export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const { scope, authProvider } = req.params
riqwan marked this conversation as resolved.
Show resolved Hide resolved

const service: IAuthModuleService = req.scope.resolve(
ModuleRegistrationName.AUTH
)

const authData = { ...req } as unknown as AuthenticationInput
authData.authScope = scope

const authResult = await service.authenticate(authProvider, authData)

const { success, error, authUser, location } = authResult
if (location) {
res.redirect(location)
return
}

if (!success && error) {
throw new MedusaError(MedusaError.Types.UNAUTHORIZED, error)
} else if (success) {
req.session.auth_user = authUser
req.session.scope = authUser.app_metadata.scope

res.status(200).json({ authUser })
}
}

export const POST = async (req: MedusaRequest, res: MedusaResponse) => {
await GET(req, res)
}
5 changes: 3 additions & 2 deletions packages/medusa/src/api-v2/store/customers/me/route.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,9 @@
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { MedusaRequest, MedusaResponse } from "../../../../types/routing"

import { ModuleRegistrationName } from "@medusajs/modules-sdk"

export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const id = req.auth_user!.app_metadata.customer_id
const id = req.session.auth_user!.app_metadata.customer_id

const customerModule = req.scope.resolve(ModuleRegistrationName.CUSTOMER)

Expand Down
8 changes: 5 additions & 3 deletions packages/medusa/src/api-v2/store/customers/middlewares.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
import * as QueryConfig from "./query-config"

import { StoreGetCustomersMeParams, StorePostCustomersReq } from "./validators"
import { transformBody, transformQuery } from "../../../api/middlewares"

import { MiddlewareRoute } from "../../../loaders/helpers/routing/types"
import { StorePostCustomersReq, StoreGetCustomersMeParams } from "./validators"
import authenticate from "../../../utils/authenticate-middleware"
import * as QueryConfig from "./query-config"
import { authenticate } from "../../../utils/authenticate-middleware"

export const storeCustomerRoutesMiddlewares: MiddlewareRoute[] = [
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ export const defaultStoreCustomersFields: (keyof CustomerDTO)[] = [
]

export const retrieveTransformQueryConfig = {
defaultFields: defaultStoreCustomersFields,
defaultFields: defaultStoreCustomersFields as string[],
defaultRelations: defaultStoreCustomersRelations,
allowedRelations: allowedStoreCustomersRelations,
isList: false,
Expand Down
Loading
Loading