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(server): client version check #9205

Open
wants to merge 7 commits into
base: canary
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
2 changes: 2 additions & 0 deletions packages/backend/server/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@
"prisma": "^5.22.0",
"reflect-metadata": "^0.2.2",
"rxjs": "^7.8.1",
"semver": "^7.6.3",
"ses": "^1.10.0",
"socket.io": "^4.8.1",
"stripe": "^17.4.0",
Expand All @@ -104,6 +105,7 @@
"@types/node": "^20.17.10",
"@types/nodemailer": "^6.4.17",
"@types/on-headers": "^1.0.3",
"@types/semver": "^7.5.8",
"@types/sinon": "^17.0.3",
"@types/supertest": "^6.0.2",
"ava": "^6.2.0",
Expand Down
2 changes: 2 additions & 0 deletions packages/backend/server/src/app.module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ import { SelfhostModule } from './core/selfhost';
import { StorageModule } from './core/storage';
import { SyncModule } from './core/sync';
import { UserModule } from './core/user';
import { VersionModule } from './core/version';
import { WorkspaceModule } from './core/workspaces';
import { REGISTERED_PLUGINS } from './plugins';
import { ENABLED_PLUGINS } from './plugins/registry';
Expand Down Expand Up @@ -167,6 +168,7 @@ export function buildAppModule() {
.useIf(
config => config.flavor.graphql,
ScheduleModule.forRoot(),
VersionModule,
GqlModule,
StorageModule,
ServerConfigModule,
Expand Down
11 changes: 11 additions & 0 deletions packages/backend/server/src/base/error/def.ts
Original file line number Diff line number Diff line change
Expand Up @@ -593,4 +593,15 @@ export const USER_FRIENDLY_ERRORS = {
type: 'bad_request',
message: 'Captcha verification failed.',
},
// version errors
unsupported_client_version: {
type: 'action_forbidden',
args: {
clientVersion: 'string',
recommendedVersion: 'string',
action: 'string',
},
message: ({ clientVersion, recommendedVersion, action }) =>
`Unsupported client version: ${clientVersion}, please ${action} to ${recommendedVersion}.`,
},
} satisfies Record<string, UserFriendlyErrorOptions>;
17 changes: 15 additions & 2 deletions packages/backend/server/src/base/error/errors.gen.ts
Original file line number Diff line number Diff line change
Expand Up @@ -591,6 +591,18 @@ export class CaptchaVerificationFailed extends UserFriendlyError {
super('bad_request', 'captcha_verification_failed', message);
}
}
@ObjectType()
class UnsupportedClientVersionDataType {
@Field() clientVersion!: string
@Field() recommendedVersion!: string
@Field() action!: string
}

export class UnsupportedClientVersion extends UserFriendlyError {
constructor(args: UnsupportedClientVersionDataType, message?: string | ((args: UnsupportedClientVersionDataType) => string)) {
super('action_forbidden', 'unsupported_client_version', message, args);
}
}
export enum ErrorNames {
INTERNAL_SERVER_ERROR,
TOO_MANY_REQUEST,
Expand Down Expand Up @@ -669,7 +681,8 @@ export enum ErrorNames {
MAILER_SERVICE_IS_NOT_CONFIGURED,
CANNOT_DELETE_ALL_ADMIN_ACCOUNT,
CANNOT_DELETE_OWN_ACCOUNT,
CAPTCHA_VERIFICATION_FAILED
CAPTCHA_VERIFICATION_FAILED,
UNSUPPORTED_CLIENT_VERSION
}
registerEnumType(ErrorNames, {
name: 'ErrorNames'
Expand All @@ -678,5 +691,5 @@ registerEnumType(ErrorNames, {
export const ErrorDataUnionType = createUnionType({
name: 'ErrorDataUnion',
types: () =>
[WrongSignInCredentialsDataType, UnknownOauthProviderDataType, MissingOauthQueryParameterDataType, InvalidEmailDataType, InvalidPasswordLengthDataType, SpaceNotFoundDataType, MemberNotFoundInSpaceDataType, NotInSpaceDataType, AlreadyInSpaceDataType, SpaceAccessDeniedDataType, SpaceOwnerNotFoundDataType, DocNotFoundDataType, DocAccessDeniedDataType, VersionRejectedDataType, InvalidHistoryTimestampDataType, DocHistoryNotFoundDataType, BlobNotFoundDataType, UnsupportedSubscriptionPlanDataType, SubscriptionAlreadyExistsDataType, SubscriptionNotExistsDataType, SameSubscriptionRecurringDataType, SubscriptionPlanNotFoundDataType, CopilotMessageNotFoundDataType, CopilotPromptNotFoundDataType, CopilotProviderSideErrorDataType, RuntimeConfigNotFoundDataType, InvalidRuntimeConfigTypeDataType] as const,
[WrongSignInCredentialsDataType, UnknownOauthProviderDataType, MissingOauthQueryParameterDataType, InvalidEmailDataType, InvalidPasswordLengthDataType, SpaceNotFoundDataType, MemberNotFoundInSpaceDataType, NotInSpaceDataType, AlreadyInSpaceDataType, SpaceAccessDeniedDataType, SpaceOwnerNotFoundDataType, DocNotFoundDataType, DocAccessDeniedDataType, VersionRejectedDataType, InvalidHistoryTimestampDataType, DocHistoryNotFoundDataType, BlobNotFoundDataType, UnsupportedSubscriptionPlanDataType, SubscriptionAlreadyExistsDataType, SubscriptionNotExistsDataType, SameSubscriptionRecurringDataType, SubscriptionPlanNotFoundDataType, CopilotMessageNotFoundDataType, CopilotPromptNotFoundDataType, CopilotProviderSideErrorDataType, RuntimeConfigNotFoundDataType, InvalidRuntimeConfigTypeDataType, UnsupportedClientVersionDataType] as const,
});
15 changes: 10 additions & 5 deletions packages/backend/server/src/base/guard/guard.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,14 +18,19 @@ export class BasicGuard implements CanActivate {

async canActivate(context: ExecutionContext) {
// get registered guard name
const providerName = this.reflector.get<string>(
const providerName = this.reflector.get<string[]>(
BasicGuardSymbol,
context.getHandler()
);

const provider = GUARD_PROVIDER[providerName as NamedGuards];
if (provider) {
return await provider.canActivate(context);
if (Array.isArray(providerName) && providerName.length > 0) {
for (const name of providerName) {
const provider = GUARD_PROVIDER[name as NamedGuards];
if (provider) {
const ret = await provider.canActivate(context);
if (!ret) return false;
}
}
}

return true;
Expand All @@ -46,5 +51,5 @@ export class BasicGuard implements CanActivate {
* }
* ```
*/
export const UseNamedGuard = (name: NamedGuards) =>
export const UseNamedGuard = (...name: NamedGuards[]) =>
applyDecorators(UseGuards(BasicGuard), SetMetadata(BasicGuardSymbol, name));
4 changes: 3 additions & 1 deletion packages/backend/server/src/core/auth/controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ export class AuthController {
}

@Public()
@UseNamedGuard('version')
@Post('/preflight')
async preflight(
@Body() params?: { email: string }
Expand Down Expand Up @@ -103,7 +104,7 @@ export class AuthController {
}

@Public()
@UseNamedGuard('captcha')
@UseNamedGuard('version', 'captcha')
@Post('/sign-in')
@Header('content-type', 'application/json')
async signIn(
Expand Down Expand Up @@ -236,6 +237,7 @@ export class AuthController {
}

@Public()
@UseNamedGuard('version')
@Post('/magic-link')
async magicLinkSignIn(
@Req() req: Request,
Expand Down
29 changes: 29 additions & 0 deletions packages/backend/server/src/core/version/config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import { defineRuntimeConfig, ModuleConfig } from '../../base/config';

export interface VersionConfig {
enable: boolean;
allowedVersion: string;
}

declare module '../../base/config' {
interface AppConfig {
version: ModuleConfig<never, VersionConfig>;
}
}

declare module '../../base/guard' {
interface RegisterGuardName {
version: 'version';
}
}

defineRuntimeConfig('version', {
enable: {
desc: 'Check version of the app',
default: false,
},
allowedVersion: {
desc: 'Allowed version range of the app that can access the server',
default: '>=0.0.1',
},
});
40 changes: 40 additions & 0 deletions packages/backend/server/src/core/version/guard.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import type {
CanActivate,
ExecutionContext,
OnModuleInit,
} from '@nestjs/common';
import { Injectable } from '@nestjs/common';

import {
getRequestResponseFromContext,
GuardProvider,
Runtime,
} from '../../base';
import { VersionService } from './service';

@Injectable()
export class VersionGuardProvider
extends GuardProvider
implements CanActivate, OnModuleInit
{
name = 'version' as const;

constructor(
private readonly runtime: Runtime,
private readonly version: VersionService
) {
super();
}

async canActivate(context: ExecutionContext) {
if (!(await this.runtime.fetch('version/enable'))) {
return true;
}

const { req } = getRequestResponseFromContext(context);

const version = req.headers['x-affine-version'];

return this.version.checkVersion(version);
}
}
13 changes: 13 additions & 0 deletions packages/backend/server/src/core/version/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import './config';

import { Module } from '@nestjs/common';

import { VersionGuardProvider } from './guard';
import { VersionService } from './service';

@Module({
providers: [VersionService, VersionGuardProvider],
})
export class VersionModule {}

export type { VersionConfig } from './config';
68 changes: 68 additions & 0 deletions packages/backend/server/src/core/version/service.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
import assert from 'node:assert';

import { Injectable, Logger } from '@nestjs/common';
import semver from 'semver';

import { Runtime, UnsupportedClientVersion } from '../../base';

@Injectable()
export class VersionService {
private readonly logger = new Logger(VersionService.name);

constructor(private readonly runtime: Runtime) {}

private async getRecommendedVersion(versionRange: string) {
try {
const range = new semver.Range(versionRange);
const versions = range.set
.flat()
.map(c => c.semver)
.toSorted((a, b) => semver.rcompare(a, b));
return versions[0]?.toString();
} catch {
return semver.valid(semver.coerce(versionRange));
}
}

async checkVersion(clientVersion?: any) {
Copy link
Member

Choose a reason for hiding this comment

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

we might need the channel control, cross channel connections should be forbidden

const allowedVersion = await this.runtime.fetch('version/allowedVersion');
const recommendedVersion = await this.getRecommendedVersion(allowedVersion);
if (!allowedVersion || !recommendedVersion) {
// ignore invalid allowed version config
return true;
}

const parsedClientVersion = semver.valid(clientVersion);
const action = semver.lt(parsedClientVersion || '0.0.0', recommendedVersion)
? 'upgrade'
: 'downgrade';
assert(
typeof clientVersion === 'string' && clientVersion.length > 0,
new UnsupportedClientVersion({
clientVersion: '[Not Provided]',
recommendedVersion,
action,
})
);

if (parsedClientVersion) {
if (!semver.satisfies(parsedClientVersion, allowedVersion)) {
throw new UnsupportedClientVersion({
clientVersion,
recommendedVersion,
action,
});
}
return true;
} else {
if (clientVersion) {
this.logger.warn(`Invalid client version: ${clientVersion}`);
}
throw new UnsupportedClientVersion({
clientVersion,
recommendedVersion,
action,
});
}
}
}
3 changes: 3 additions & 0 deletions packages/backend/server/src/plugins/oauth/controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import {
OauthAccountAlreadyConnected,
OauthStateExpired,
UnknownOauthProvider,
UseNamedGuard,
} from '../../base';
import { AuthService, Public } from '../../core/auth';
import { UserService } from '../../core/user';
Expand All @@ -35,6 +36,7 @@ export class OAuthController {
) {}

@Public()
@UseNamedGuard('version')
@Post('/preflight')
@HttpCode(HttpStatus.OK)
async preflight(
Expand Down Expand Up @@ -64,6 +66,7 @@ export class OAuthController {
}

@Public()
@UseNamedGuard('version')
@Post('/callback')
@HttpCode(HttpStatus.OK)
async callback(
Expand Down
9 changes: 8 additions & 1 deletion packages/backend/server/src/schema.gql
Original file line number Diff line number Diff line change
Expand Up @@ -209,7 +209,7 @@ type EditorType {
name: String!
}

union ErrorDataUnion = AlreadyInSpaceDataType | BlobNotFoundDataType | CopilotMessageNotFoundDataType | CopilotPromptNotFoundDataType | CopilotProviderSideErrorDataType | DocAccessDeniedDataType | DocHistoryNotFoundDataType | DocNotFoundDataType | InvalidEmailDataType | InvalidHistoryTimestampDataType | InvalidPasswordLengthDataType | InvalidRuntimeConfigTypeDataType | MemberNotFoundInSpaceDataType | MissingOauthQueryParameterDataType | NotInSpaceDataType | RuntimeConfigNotFoundDataType | SameSubscriptionRecurringDataType | SpaceAccessDeniedDataType | SpaceNotFoundDataType | SpaceOwnerNotFoundDataType | SubscriptionAlreadyExistsDataType | SubscriptionNotExistsDataType | SubscriptionPlanNotFoundDataType | UnknownOauthProviderDataType | UnsupportedSubscriptionPlanDataType | VersionRejectedDataType | WrongSignInCredentialsDataType
union ErrorDataUnion = AlreadyInSpaceDataType | BlobNotFoundDataType | CopilotMessageNotFoundDataType | CopilotPromptNotFoundDataType | CopilotProviderSideErrorDataType | DocAccessDeniedDataType | DocHistoryNotFoundDataType | DocNotFoundDataType | InvalidEmailDataType | InvalidHistoryTimestampDataType | InvalidPasswordLengthDataType | InvalidRuntimeConfigTypeDataType | MemberNotFoundInSpaceDataType | MissingOauthQueryParameterDataType | NotInSpaceDataType | RuntimeConfigNotFoundDataType | SameSubscriptionRecurringDataType | SpaceAccessDeniedDataType | SpaceNotFoundDataType | SpaceOwnerNotFoundDataType | SubscriptionAlreadyExistsDataType | SubscriptionNotExistsDataType | SubscriptionPlanNotFoundDataType | UnknownOauthProviderDataType | UnsupportedClientVersionDataType | UnsupportedSubscriptionPlanDataType | VersionRejectedDataType | WrongSignInCredentialsDataType

enum ErrorNames {
ACCESS_DENIED
Expand Down Expand Up @@ -282,6 +282,7 @@ enum ErrorNames {
TOO_MANY_REQUEST
UNKNOWN_OAUTH_PROVIDER
UNSPLASH_IS_NOT_CONFIGURED
UNSUPPORTED_CLIENT_VERSION
UNSUPPORTED_SUBSCRIPTION_PLAN
USER_AVATAR_NOT_FOUND
USER_NOT_FOUND
Expand Down Expand Up @@ -861,6 +862,12 @@ type UnknownOauthProviderDataType {
name: String!
}

type UnsupportedClientVersionDataType {
action: String!
clientVersion: String!
recommendedVersion: String!
}

type UnsupportedSubscriptionPlanDataType {
plan: String!
}
Expand Down
Loading
Loading