forked from baron/baron-sso
261 lines
8.2 KiB
Dart
261 lines
8.2 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
import 'package:userfront/core/constants/error_whitelist.dart';
|
|
import 'package:userfront/features/auth/presentation/error_screen.dart';
|
|
import 'package:userfront/i18n.dart';
|
|
|
|
Future<void> _pumpErrorScreen(
|
|
WidgetTester tester, {
|
|
String? errorCode,
|
|
String? description,
|
|
bool? isProdOverride,
|
|
Future<Map<String, dynamic>> Function()? sessionProfileLoader,
|
|
Map<String, dynamic>? tenantAccessDetails,
|
|
}) async {
|
|
await tester.pumpWidget(
|
|
MaterialApp(
|
|
home: ErrorScreen(
|
|
errorCode: errorCode,
|
|
description: description,
|
|
isProdOverride: isProdOverride,
|
|
sessionProfileLoader: sessionProfileLoader,
|
|
tenantAccessDetails: tenantAccessDetails,
|
|
),
|
|
),
|
|
);
|
|
await tester.pump();
|
|
}
|
|
|
|
void main() {
|
|
setUpAll(() {
|
|
TestWidgetsFlutterBinding.ensureInitialized();
|
|
final dispatcher = TestWidgetsFlutterBinding.instance.platformDispatcher;
|
|
dispatcher.localeTestValue = const Locale('ko');
|
|
dispatcher.localesTestValue = const [Locale('ko')];
|
|
});
|
|
|
|
tearDownAll(() {
|
|
final dispatcher = TestWidgetsFlutterBinding.instance.platformDispatcher;
|
|
dispatcher.clearLocaleTestValue();
|
|
dispatcher.clearLocalesTestValue();
|
|
});
|
|
|
|
testWidgets('개발환경은 원문 메시지를 노출한다', (WidgetTester tester) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'custom_error',
|
|
description: '원문 메시지',
|
|
isProdOverride: false,
|
|
);
|
|
|
|
final title = tr(
|
|
'msg.userfront.error.title_with_code',
|
|
fallback: '오류: {{code}}',
|
|
params: {'code': 'custom_error'},
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'custom_error'},
|
|
);
|
|
|
|
expect(find.text(title), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsOneWidget);
|
|
expect(find.text(type), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('프로덕션은 whitelist 메시지를 노출한다', (WidgetTester tester) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'settings_disabled',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
);
|
|
|
|
final title = tr(
|
|
'msg.userfront.error.title',
|
|
fallback: '인증 과정에서 오류가 발생했습니다',
|
|
);
|
|
final detail = tr(
|
|
'msg.userfront.error.whitelist.settings_disabled',
|
|
fallback: tr(internalErrorWhitelistMessageKeys['settings_disabled']!),
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'settings_disabled'},
|
|
);
|
|
|
|
expect(find.text(title), findsOneWidget);
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsNothing);
|
|
expect(find.text(type), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('프로덕션은 ORY 코드를 bypass 처리한다', (WidgetTester tester) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'access_denied',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
);
|
|
|
|
final title = tr(
|
|
'msg.userfront.error.title',
|
|
fallback: '인증 과정에서 오류가 발생했습니다',
|
|
);
|
|
final detail = tr(
|
|
'msg.userfront.error.ory.access_denied',
|
|
fallback: '사용자가 동의를 거부했습니다.',
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'access_denied'},
|
|
);
|
|
|
|
expect(find.text(title), findsOneWidget);
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsNothing);
|
|
expect(find.text(type), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('프로덕션은 비허용 에러를 unknown_error로 처리한다', (WidgetTester tester) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'weird_error',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
);
|
|
|
|
final title = tr(
|
|
'msg.userfront.error.title',
|
|
fallback: '인증 과정에서 오류가 발생했습니다',
|
|
);
|
|
final detail = tr(
|
|
'msg.userfront.error.detail_contact',
|
|
fallback: '에러가 계속되면 관리자에게 문의해주세요',
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'unknown_error'},
|
|
);
|
|
|
|
expect(find.text(title), findsOneWidget);
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsNothing);
|
|
expect(find.text(type), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('프로덕션은 not_found 코드를 whitelist 메시지로 노출한다 (404 매핑)', (
|
|
WidgetTester tester,
|
|
) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'not_found',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
);
|
|
|
|
final detail = tr(
|
|
'msg.userfront.error.whitelist.not_found',
|
|
fallback: tr(internalErrorWhitelistMessageKeys['not_found']!),
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'not_found'},
|
|
);
|
|
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text(type), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsNothing);
|
|
});
|
|
|
|
testWidgets('프로덕션은 rate_limited 코드를 whitelist 메시지로 노출한다 (429 매핑)', (
|
|
WidgetTester tester,
|
|
) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'rate_limited',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
);
|
|
|
|
final detail = tr(
|
|
'msg.userfront.error.whitelist.rate_limited',
|
|
fallback: tr(internalErrorWhitelistMessageKeys['rate_limited']!),
|
|
);
|
|
final type = tr(
|
|
'msg.userfront.error.type',
|
|
fallback: '오류 종류: {{type}}',
|
|
params: {'type': 'rate_limited'},
|
|
);
|
|
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text(type), findsOneWidget);
|
|
expect(find.text('원문 메시지'), findsNothing);
|
|
});
|
|
|
|
testWidgets('tenant_not_allowed는 전용 차단 정보를 노출한다', (
|
|
WidgetTester tester,
|
|
) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'tenant_not_allowed',
|
|
description: '원문 메시지',
|
|
isProdOverride: true,
|
|
sessionProfileLoader: () async {
|
|
return {
|
|
'email': 'employee@example.com',
|
|
'tenant': {'name': 'Baron HQ', 'slug': 'baron-hq'},
|
|
};
|
|
},
|
|
);
|
|
|
|
const title = 'Application access is restricted';
|
|
const detail =
|
|
'The current signed-in account cannot access this application.';
|
|
const account = 'Account';
|
|
const primaryTenant = 'Primary affiliated tenant';
|
|
const affiliatedTenants = 'All affiliated tenants';
|
|
expect(find.text(title), findsOneWidget);
|
|
expect(find.text(detail), findsOneWidget);
|
|
expect(find.text(account), findsOneWidget);
|
|
expect(find.text('employee@example.com'), findsOneWidget);
|
|
expect(find.text(primaryTenant), findsOneWidget);
|
|
expect(find.text(affiliatedTenants), findsOneWidget);
|
|
expect(find.text('Baron HQ'), findsNWidgets(2));
|
|
expect(find.byType(ElevatedButton), findsOneWidget);
|
|
expect(find.byType(OutlinedButton), findsOneWidget);
|
|
});
|
|
|
|
testWidgets('tenant_not_allowed는 details를 우선 사용해 계정과 테넌트 정보를 노출한다', (
|
|
WidgetTester tester,
|
|
) async {
|
|
await _pumpErrorScreen(
|
|
tester,
|
|
errorCode: 'tenant_not_allowed',
|
|
isProdOverride: true,
|
|
tenantAccessDetails: {
|
|
'account': {'email': 'dyddus1210@gmail.com'},
|
|
'current_tenant': {'name': 'test1 company', 'slug': 'test1-company'},
|
|
'affiliated_tenants': [
|
|
{'name': 'test1 company', 'slug': 'test1-company'},
|
|
{'name': 'test2 company', 'slug': 'test-company'},
|
|
],
|
|
'allowed_tenants': [
|
|
{'name': 'test4', 'slug': 'test4'},
|
|
],
|
|
},
|
|
);
|
|
|
|
expect(find.text('dyddus1210@gmail.com'), findsOneWidget);
|
|
expect(find.text('test1 company'), findsOneWidget);
|
|
expect(find.text('test1 company, test2 company'), findsOneWidget);
|
|
expect(find.text('test4'), findsOneWidget);
|
|
expect(find.text('알 수 없음'), findsNothing);
|
|
});
|
|
}
|