fix: make client credential optional in authcode + pkce flow (#4114)
* fix: make client credential optional * test: update test fixtures --------- Co-authored-by: jamesgeorge007 <25279263+jamesgeorge007@users.noreply.github.com>
This commit is contained in:
@@ -302,7 +302,7 @@ const supportedGrantTypes = [
|
|||||||
|
|
||||||
auth.value.grantTypeInfo = {
|
auth.value.grantTypeInfo = {
|
||||||
...auth.value.grantTypeInfo,
|
...auth.value.grantTypeInfo,
|
||||||
clientSecret: value,
|
clientSecret: value ?? "",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -30,7 +30,6 @@ const AuthCodeOauthFlowParamsSchema = AuthCodeGrantTypeParams.pick({
|
|||||||
params.authEndpoint.length >= 1 &&
|
params.authEndpoint.length >= 1 &&
|
||||||
params.tokenEndpoint.length >= 1 &&
|
params.tokenEndpoint.length >= 1 &&
|
||||||
params.clientID.length >= 1 &&
|
params.clientID.length >= 1 &&
|
||||||
params.clientSecret.length >= 1 &&
|
|
||||||
(!params.scopes || params.scopes.trim().length >= 1)
|
(!params.scopes || params.scopes.trim().length >= 1)
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
@@ -85,7 +84,7 @@ const initAuthCodeOauthFlow = async ({
|
|||||||
grant_type: "AUTHORIZATION_CODE"
|
grant_type: "AUTHORIZATION_CODE"
|
||||||
authEndpoint: string
|
authEndpoint: string
|
||||||
tokenEndpoint: string
|
tokenEndpoint: string
|
||||||
clientSecret: string
|
clientSecret?: string
|
||||||
clientID: string
|
clientID: string
|
||||||
isPKCE: boolean
|
isPKCE: boolean
|
||||||
codeVerifier?: string
|
codeVerifier?: string
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ export const REST_COLLECTIONS_MOCK: HoppCollection[] = [
|
|||||||
folders: [],
|
folders: [],
|
||||||
requests: [
|
requests: [
|
||||||
{
|
{
|
||||||
v: "4",
|
v: "5",
|
||||||
endpoint: "https://echo.hoppscotch.io",
|
endpoint: "https://echo.hoppscotch.io",
|
||||||
name: "Echo test",
|
name: "Echo test",
|
||||||
params: [],
|
params: [],
|
||||||
@@ -50,7 +50,7 @@ export const GQL_COLLECTIONS_MOCK: HoppCollection[] = [
|
|||||||
folders: [],
|
folders: [],
|
||||||
requests: [
|
requests: [
|
||||||
{
|
{
|
||||||
v: 4,
|
v: 5,
|
||||||
name: "Echo test",
|
name: "Echo test",
|
||||||
url: "https://echo.hoppscotch.io/graphql",
|
url: "https://echo.hoppscotch.io/graphql",
|
||||||
headers: [],
|
headers: [],
|
||||||
@@ -138,7 +138,7 @@ export const REST_HISTORY_MOCK: RESTHistoryEntry[] = [
|
|||||||
preRequestScript: "",
|
preRequestScript: "",
|
||||||
testScript: "",
|
testScript: "",
|
||||||
requestVariables: [],
|
requestVariables: [],
|
||||||
v: "4",
|
v: "5",
|
||||||
},
|
},
|
||||||
responseMeta: { duration: 807, statusCode: 200 },
|
responseMeta: { duration: 807, statusCode: 200 },
|
||||||
star: false,
|
star: false,
|
||||||
@@ -150,7 +150,7 @@ export const GQL_HISTORY_MOCK: GQLHistoryEntry[] = [
|
|||||||
{
|
{
|
||||||
v: 1,
|
v: 1,
|
||||||
request: {
|
request: {
|
||||||
v: 4,
|
v: 5,
|
||||||
name: "Untitled",
|
name: "Untitled",
|
||||||
url: "https://echo.hoppscotch.io/graphql",
|
url: "https://echo.hoppscotch.io/graphql",
|
||||||
query: "query Request { url }",
|
query: "query Request { url }",
|
||||||
@@ -171,7 +171,7 @@ export const GQL_TAB_STATE_MOCK: PersistableTabState<HoppGQLDocument> = {
|
|||||||
tabID: "5edbe8d4-65c9-4381-9354-5f1bf05d8ccc",
|
tabID: "5edbe8d4-65c9-4381-9354-5f1bf05d8ccc",
|
||||||
doc: {
|
doc: {
|
||||||
request: {
|
request: {
|
||||||
v: 4,
|
v: 5,
|
||||||
name: "Untitled",
|
name: "Untitled",
|
||||||
url: "https://echo.hoppscotch.io/graphql",
|
url: "https://echo.hoppscotch.io/graphql",
|
||||||
headers: [],
|
headers: [],
|
||||||
@@ -194,7 +194,7 @@ export const REST_TAB_STATE_MOCK: PersistableTabState<HoppRESTDocument> = {
|
|||||||
tabID: "e6e8d800-caa8-44a2-a6a6-b4765a3167aa",
|
tabID: "e6e8d800-caa8-44a2-a6a6-b4765a3167aa",
|
||||||
doc: {
|
doc: {
|
||||||
request: {
|
request: {
|
||||||
v: "4",
|
v: "5",
|
||||||
endpoint: "https://echo.hoppscotch.io",
|
endpoint: "https://echo.hoppscotch.io",
|
||||||
name: "Echo test",
|
name: "Echo test",
|
||||||
params: [],
|
params: [],
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import V1_VERSION from "./v/1"
|
|||||||
import V2_VERSION from "./v/2"
|
import V2_VERSION from "./v/2"
|
||||||
import V3_VERSION from "./v/3"
|
import V3_VERSION from "./v/3"
|
||||||
import V4_VERSION from "./v/4"
|
import V4_VERSION from "./v/4"
|
||||||
|
import V5_VERSION from "./v/5"
|
||||||
|
|
||||||
export { GQLHeader } from "./v/1"
|
export { GQLHeader } from "./v/1"
|
||||||
export {
|
export {
|
||||||
@@ -13,23 +14,24 @@ export {
|
|||||||
HoppGQLAuthInherit,
|
HoppGQLAuthInherit,
|
||||||
} from "./v/2"
|
} from "./v/2"
|
||||||
|
|
||||||
export { HoppGQLAuth } from "./v/4"
|
export { HoppGQLAuthOAuth2, HoppGQLAuth } from "./v/5"
|
||||||
export { HoppGQLAuthOAuth2 } from "./v/3"
|
|
||||||
export { HoppGQLAuthAPIKey } from "./v/4"
|
export { HoppGQLAuthAPIKey } from "./v/4"
|
||||||
|
|
||||||
export const GQL_REQ_SCHEMA_VERSION = 4
|
export const GQL_REQ_SCHEMA_VERSION = 5
|
||||||
|
|
||||||
const versionedObject = z.object({
|
const versionedObject = z.object({
|
||||||
v: z.number(),
|
v: z.number(),
|
||||||
})
|
})
|
||||||
|
|
||||||
export const HoppGQLRequest = createVersionedEntity({
|
export const HoppGQLRequest = createVersionedEntity({
|
||||||
latestVersion: 4,
|
latestVersion: 5,
|
||||||
versionMap: {
|
versionMap: {
|
||||||
1: V1_VERSION,
|
1: V1_VERSION,
|
||||||
2: V2_VERSION,
|
2: V2_VERSION,
|
||||||
3: V3_VERSION,
|
3: V3_VERSION,
|
||||||
4: V4_VERSION,
|
4: V4_VERSION,
|
||||||
|
5: V5_VERSION,
|
||||||
},
|
},
|
||||||
getVersion(x) {
|
getVersion(x) {
|
||||||
const result = versionedObject.safeParse(x)
|
const result = versionedObject.safeParse(x)
|
||||||
|
|||||||
47
packages/hoppscotch-data/src/graphql/v/5.ts
Normal file
47
packages/hoppscotch-data/src/graphql/v/5.ts
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import { z } from "zod"
|
||||||
|
|
||||||
|
import { defineVersion } from "verzod"
|
||||||
|
|
||||||
|
import { HoppRESTAuthOAuth2 } from "../../rest/v/5"
|
||||||
|
import {
|
||||||
|
HoppGQLAuthBasic,
|
||||||
|
HoppGQLAuthBearer,
|
||||||
|
HoppGQLAuthInherit,
|
||||||
|
HoppGQLAuthNone,
|
||||||
|
} from "./2"
|
||||||
|
import { HoppGQLAuthAPIKey, V4_SCHEMA } from "./4"
|
||||||
|
|
||||||
|
export { HoppRESTAuthOAuth2 as HoppGQLAuthOAuth2 } from "../../rest/v/5"
|
||||||
|
|
||||||
|
export const HoppGQLAuth = z
|
||||||
|
.discriminatedUnion("authType", [
|
||||||
|
HoppGQLAuthNone,
|
||||||
|
HoppGQLAuthInherit,
|
||||||
|
HoppGQLAuthBasic,
|
||||||
|
HoppGQLAuthBearer,
|
||||||
|
HoppGQLAuthAPIKey,
|
||||||
|
HoppRESTAuthOAuth2, // both rest and gql have the same auth type for oauth2
|
||||||
|
])
|
||||||
|
.and(
|
||||||
|
z.object({
|
||||||
|
authActive: z.boolean(),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
export type HoppGQLAuth = z.infer<typeof HoppGQLAuth>
|
||||||
|
|
||||||
|
export const V5_SCHEMA = V4_SCHEMA.extend({
|
||||||
|
v: z.literal(5),
|
||||||
|
auth: HoppGQLAuth,
|
||||||
|
})
|
||||||
|
|
||||||
|
export default defineVersion({
|
||||||
|
initial: false,
|
||||||
|
schema: V5_SCHEMA,
|
||||||
|
up(old: z.infer<typeof V4_SCHEMA>) {
|
||||||
|
return {
|
||||||
|
...old,
|
||||||
|
v: 5 as const,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
@@ -6,12 +6,13 @@ import V1_VERSION from "./v/1"
|
|||||||
import V2_VERSION from "./v/2"
|
import V2_VERSION from "./v/2"
|
||||||
import V3_VERSION from "./v/3"
|
import V3_VERSION from "./v/3"
|
||||||
import V4_VERSION from "./v/4"
|
import V4_VERSION from "./v/4"
|
||||||
|
import V5_VERSION from "./v/5"
|
||||||
import { createVersionedEntity, InferredEntity } from "verzod"
|
import { createVersionedEntity, InferredEntity } from "verzod"
|
||||||
import { lodashIsEqualEq, mapThenEq, undefinedEq } from "../utils/eq"
|
import { lodashIsEqualEq, mapThenEq, undefinedEq } from "../utils/eq"
|
||||||
|
|
||||||
import { HoppRESTReqBody, HoppRESTHeaders, HoppRESTParams } from "./v/1"
|
import { HoppRESTReqBody, HoppRESTHeaders, HoppRESTParams } from "./v/1"
|
||||||
|
|
||||||
import { HoppRESTAuth } from "./v/4"
|
import { HoppRESTAuth } from "./v/5"
|
||||||
import { HoppRESTRequestVariables } from "./v/2"
|
import { HoppRESTRequestVariables } from "./v/2"
|
||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
|
|
||||||
@@ -29,14 +30,18 @@ export {
|
|||||||
} from "./v/1"
|
} from "./v/1"
|
||||||
|
|
||||||
export {
|
export {
|
||||||
HoppRESTAuthOAuth2,
|
|
||||||
AuthCodeGrantTypeParams,
|
|
||||||
ClientCredentialsGrantTypeParams,
|
ClientCredentialsGrantTypeParams,
|
||||||
ImplicitOauthFlowParams,
|
ImplicitOauthFlowParams,
|
||||||
PasswordGrantTypeParams,
|
PasswordGrantTypeParams,
|
||||||
} from "./v/3"
|
} from "./v/3"
|
||||||
|
|
||||||
export { HoppRESTAuth, HoppRESTAuthAPIKey } from "./v/4"
|
export {
|
||||||
|
AuthCodeGrantTypeParams,
|
||||||
|
HoppRESTAuthOAuth2,
|
||||||
|
HoppRESTAuth,
|
||||||
|
} from "./v/5"
|
||||||
|
|
||||||
|
export { HoppRESTAuthAPIKey } from "./v/4"
|
||||||
|
|
||||||
export { HoppRESTRequestVariables } from "./v/2"
|
export { HoppRESTRequestVariables } from "./v/2"
|
||||||
|
|
||||||
@@ -46,13 +51,14 @@ const versionedObject = z.object({
|
|||||||
})
|
})
|
||||||
|
|
||||||
export const HoppRESTRequest = createVersionedEntity({
|
export const HoppRESTRequest = createVersionedEntity({
|
||||||
latestVersion: 4,
|
latestVersion: 5,
|
||||||
versionMap: {
|
versionMap: {
|
||||||
0: V0_VERSION,
|
0: V0_VERSION,
|
||||||
1: V1_VERSION,
|
1: V1_VERSION,
|
||||||
2: V2_VERSION,
|
2: V2_VERSION,
|
||||||
3: V3_VERSION,
|
3: V3_VERSION,
|
||||||
4: V4_VERSION,
|
4: V4_VERSION,
|
||||||
|
5: V5_VERSION,
|
||||||
},
|
},
|
||||||
getVersion(data) {
|
getVersion(data) {
|
||||||
// For V1 onwards we have the v string storing the number
|
// For V1 onwards we have the v string storing the number
|
||||||
@@ -94,7 +100,7 @@ const HoppRESTRequestEq = Eq.struct<HoppRESTRequest>({
|
|||||||
),
|
),
|
||||||
})
|
})
|
||||||
|
|
||||||
export const RESTReqSchemaVersion = "4"
|
export const RESTReqSchemaVersion = "5"
|
||||||
|
|
||||||
export type HoppRESTParam = HoppRESTRequest["params"][number]
|
export type HoppRESTParam = HoppRESTRequest["params"][number]
|
||||||
export type HoppRESTHeader = HoppRESTRequest["headers"][number]
|
export type HoppRESTHeader = HoppRESTRequest["headers"][number]
|
||||||
@@ -189,7 +195,7 @@ export function makeRESTRequest(
|
|||||||
|
|
||||||
export function getDefaultRESTRequest(): HoppRESTRequest {
|
export function getDefaultRESTRequest(): HoppRESTRequest {
|
||||||
return {
|
return {
|
||||||
v: "4",
|
v: "5",
|
||||||
endpoint: "https://echo.hoppscotch.io",
|
endpoint: "https://echo.hoppscotch.io",
|
||||||
name: "Untitled",
|
name: "Untitled",
|
||||||
params: [],
|
params: [],
|
||||||
|
|||||||
66
packages/hoppscotch-data/src/rest/v/5.ts
Normal file
66
packages/hoppscotch-data/src/rest/v/5.ts
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import { z } from "zod"
|
||||||
|
import { defineVersion } from "verzod"
|
||||||
|
import {
|
||||||
|
HoppRESTAuthBasic,
|
||||||
|
HoppRESTAuthBearer,
|
||||||
|
HoppRESTAuthInherit,
|
||||||
|
HoppRESTAuthNone,
|
||||||
|
} from "./1"
|
||||||
|
import { HoppRESTAuthAPIKey, V4_SCHEMA } from "./4"
|
||||||
|
import {
|
||||||
|
AuthCodeGrantTypeParams as AuthCodeGrantTypeParamsOld,
|
||||||
|
ClientCredentialsGrantTypeParams,
|
||||||
|
HoppRESTAuthOAuth2 as HoppRESTAuthOAuth2Old,
|
||||||
|
ImplicitOauthFlowParams,
|
||||||
|
PasswordGrantTypeParams,
|
||||||
|
} from "./3"
|
||||||
|
|
||||||
|
export const AuthCodeGrantTypeParams = AuthCodeGrantTypeParamsOld.extend({
|
||||||
|
clientSecret: z.string().optional(),
|
||||||
|
})
|
||||||
|
|
||||||
|
export const HoppRESTAuthOAuth2 = HoppRESTAuthOAuth2Old.extend({
|
||||||
|
grantTypeInfo: z.discriminatedUnion("grantType", [
|
||||||
|
AuthCodeGrantTypeParams,
|
||||||
|
ClientCredentialsGrantTypeParams,
|
||||||
|
PasswordGrantTypeParams,
|
||||||
|
ImplicitOauthFlowParams,
|
||||||
|
]),
|
||||||
|
})
|
||||||
|
|
||||||
|
export type HoppRESTAuthOAuth2 = z.infer<typeof HoppRESTAuthOAuth2>
|
||||||
|
|
||||||
|
export const HoppRESTAuth = z
|
||||||
|
.discriminatedUnion("authType", [
|
||||||
|
HoppRESTAuthNone,
|
||||||
|
HoppRESTAuthInherit,
|
||||||
|
HoppRESTAuthBasic,
|
||||||
|
HoppRESTAuthBearer,
|
||||||
|
HoppRESTAuthOAuth2,
|
||||||
|
HoppRESTAuthAPIKey,
|
||||||
|
])
|
||||||
|
.and(
|
||||||
|
z.object({
|
||||||
|
authActive: z.boolean(),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
export type HoppRESTAuth = z.infer<typeof HoppRESTAuth>
|
||||||
|
|
||||||
|
export const V5_SCHEMA = V4_SCHEMA.extend({
|
||||||
|
v: z.literal("5"),
|
||||||
|
auth: HoppRESTAuth,
|
||||||
|
})
|
||||||
|
|
||||||
|
export default defineVersion({
|
||||||
|
schema: V5_SCHEMA,
|
||||||
|
initial: false,
|
||||||
|
up(old: z.infer<typeof V4_SCHEMA>) {
|
||||||
|
// v5 is not a breaking change in terms of migrations
|
||||||
|
// we're just making clientSecret in authcode + pkce flow optional
|
||||||
|
return {
|
||||||
|
...old,
|
||||||
|
v: "5" as const,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user