-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* reverting old callback * to more recent version
- Loading branch information
1 parent
6e43b6a
commit fc549de
Showing
2 changed files
with
139 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
'use client' | ||
|
||
import {useSearchParams} from 'next/navigation' | ||
import {useEffect} from 'react' | ||
import {FullScreenCenter} from '@/components/FullScreenCenter' | ||
|
||
/** | ||
* Workaround for searchParams being empty on production. Will ahve to check | ||
* @see https://github.com/vercel/next.js/issues/43077#issuecomment-1383742153 | ||
*/ | ||
export const dynamic = 'force-dynamic' | ||
|
||
/** https://beta.nextjs.org/docs/api-reference/file-conventions/page#searchparams-optional */ | ||
export default function OAuthCallback() { | ||
const searchParams = useSearchParams() | ||
const code = searchParams.get('code') | ||
const state = searchParams.get('state') | ||
|
||
useEffect(() => { | ||
if ( | ||
code && | ||
state && | ||
Buffer.from(state, 'base64').toString('utf8').startsWith('conn_') | ||
) { | ||
// Just close the window - parent that opens this in a popup after redirect will read params directly | ||
window.close() | ||
} | ||
}, [code, state]) | ||
|
||
return ( | ||
<FullScreenCenter> | ||
<span className="mb-2">Processing authentication...</span> | ||
</FullScreenCenter> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,35 +1,118 @@ | ||
'use client' | ||
|
||
import {useSearchParams} from 'next/navigation' | ||
import {useEffect} from 'react' | ||
import '@openint/app-config/register.node' | ||
import {cookies} from 'next/headers' | ||
import {redirect} from 'next/navigation' | ||
import {kAccessToken} from '@openint/app-config/constants' | ||
import {envRequired} from '@openint/app-config/env' | ||
import type {Id} from '@openint/cdk' | ||
import {initNangoSDK, NangoConnect} from '@openint/cdk' | ||
import type {FrameMessage} from '@openint/connect' | ||
import {FullScreenCenter} from '@/components/FullScreenCenter' | ||
import {serverSideHelpersFromViewer} from '@/lib-server' | ||
import {serverComponentGetViewer} from '@/lib-server/server-component-helpers' | ||
import {kConnectSession, zConnectSession} from '../../shared' | ||
import {CallbackEffect} from './CallbackEffect' | ||
|
||
export const metadata = { | ||
title: 'OpenInt Oauth Callback', | ||
} | ||
|
||
/** | ||
* Workaround for searchParams being empty on production. Will ahve to check | ||
* @see https://github.com/vercel/next.js/issues/43077#issuecomment-1383742153 | ||
*/ | ||
@@ -23,104 +11,25 @@ export const metadata = { | ||
export const dynamic = 'force-dynamic' | ||
/** https://beta.nextjs.org/docs/api-reference/file-conventions/page#searchparams-optional */ | ||
export default function OAuthCallback() { | ||
const searchParams = useSearchParams() | ||
const code = searchParams.get('code') | ||
const state = searchParams.get('state') | ||
|
||
useEffect(() => { | ||
if ( | ||
code && | ||
state && | ||
Buffer.from(state, 'base64').toString('utf8').startsWith('conn_') | ||
) { | ||
// Just close the window - parent that opens this in a popup after redirect will read params directly | ||
window.close() | ||
export default async function ConnectCallback({ | ||
searchParams, | ||
}: { | ||
// Only accessible in PageComponent rather than layout component | ||
// @see https://github.com/vercel/next.js/issues/43704 | ||
searchParams: Record<string, string | string[] | undefined> | ||
}) { | ||
// TODO: Can we use cookies-next to read cookie in this environment? | ||
const cookie = cookies().get(kConnectSession) | ||
if (!cookie) { | ||
console.warn('No cookie found, redirecting to openint') | ||
// Temporary hack to redirect to the right place to accomodate for oauth url not fully changed yet | ||
const url = new URL('https://app.venice.is/connect/callback') | ||
for (const [key, value] of Object.entries(searchParams)) { | ||
url.searchParams.append(key, value as string) | ||
} | ||
return redirect(url.toString()) | ||
} | ||
const msg = await (async (): Promise<FrameMessage | null> => { | ||
try { | ||
const res = await NangoConnect.doOauthCallback(searchParams) | ||
if (!res) { | ||
// This means that we are using the @nango/frontend websocket client... | ||
return null | ||
} | ||
if (!cookie) { | ||
return { | ||
type: 'ERROR', | ||
data: {code: 'BAD_REQUEST', message: 'No session found'}, | ||
} | ||
} | ||
if (res.eventType !== 'AUTHORIZATION_SUCEEDED') { | ||
return { | ||
type: 'ERROR', | ||
data: {code: res.data.authErrorType, message: res.data.authErrorDesc}, | ||
} | ||
} | ||
const session = zConnectSession.parse(JSON.parse(cookie.value)) | ||
const viewer = await serverComponentGetViewer({ | ||
searchParams: {[kAccessToken]: session.token}, | ||
}) | ||
const connectionId = res.data.connectionId as Id['conn'] | ||
if (session.connectionId !== connectionId) { | ||
console.warn('Revoking due to unmatched connectionId') | ||
const nango = initNangoSDK({ | ||
headers: {authorization: `Bearer ${envRequired.NANGO_SECRET_KEY}`}, | ||
}) | ||
await nango.DELETE('/connection/{connectionId}', { | ||
params: { | ||
path: {connectionId: res.data.connectionId}, | ||
query: {provider_config_key: res.data.providerConfigKey}, | ||
}, | ||
}) | ||
return { | ||
type: 'ERROR', | ||
data: { | ||
code: 'FORBIDDEN', | ||
message: `Session connectionId (${session.connectionId}) not matching connected connectionId ${connectionId}`, | ||
}, | ||
} | ||
} | ||
const {caller} = serverSideHelpersFromViewer(viewer) | ||
await caller.postConnect([res.data, res.data.providerConfigKey, {}]) | ||
return { | ||
type: 'SUCCESS', | ||
data: {connectionId: res.data.connectionId as Id['conn']}, | ||
} | ||
} catch (err) { | ||
console.error('[oauth] Error during connect', err) | ||
return { | ||
type: 'ERROR', | ||
data: {code: 'INTERNAL_SERVER_ERROR', message: `${err}`}, | ||
} | ||
} | ||
}, [code, state]) | ||
})() | ||
console.log('[oauth] callback result', msg) | ||
|
||
// How do we do redirect here? | ||
return ( | ||
<FullScreenCenter> | ||
<span className="mb-2">Processing authentication...</span> | ||
{msg && ( | ||
<> | ||
<span className="mb-2">{msg.type} </span> | ||
<span className="mb-2"> | ||
{msg.type === 'ERROR' | ||
? `[${msg.data.code}] ${msg.data.message}` | ||
: msg.data.connectionId} | ||
</span> | ||
</> | ||
)} | ||
<CallbackEffect msg={msg} autoClose={!msg} /> | ||
</FullScreenCenter> | ||
) | ||
} |