Backend (FastAPI + SQLAlchemy + SQLite): - JWT auth with access/refresh tokens, bcrypt password hashing - User model with member/organizer/admin roles, auto-approve members - Championship, Registration, ParticipantList, Notification models - Alembic async migrations, seed data with test users - Registration endpoint returns tokens for members, pending for organizers - /registrations/my returns championship title/date/location via eager loading - Admin endpoints: list users, approve/reject organizers Mobile (React Native + Expo + TypeScript): - Zustand auth store, Axios client with token refresh interceptor - Role-based registration (Member vs Organizer) with contextual form labels - Tab navigation with Ionicons, safe area headers, admin tab for admin role - Championships list with status badges, detail screen with registration progress - My Registrations with championship title, progress bar, and tap-to-navigate - Admin panel with pending/all filter, approve/reject with confirmation - Profile screen with role badge, Ionicons info rows, sign out - Password visibility toggle (Ionicons), keyboard flow hints (returnKeyType) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
34 lines
1.1 KiB
TypeScript
34 lines
1.1 KiB
TypeScript
import { apiClient } from './client';
|
|
import type { TokenPair, User } from '../types';
|
|
|
|
export const authApi = {
|
|
register: (data: {
|
|
email: string;
|
|
password: string;
|
|
full_name: string;
|
|
phone?: string;
|
|
requested_role: 'member' | 'organizer';
|
|
organization_name?: string;
|
|
instagram_handle?: string;
|
|
}) =>
|
|
apiClient
|
|
.post<{ user: User; access_token?: string; refresh_token?: string }>('/auth/register', data)
|
|
.then((r) => r.data),
|
|
|
|
login: (data: { email: string; password: string }) =>
|
|
apiClient.post<TokenPair>('/auth/login', data).then((r) => r.data),
|
|
|
|
refresh: (refresh_token: string) =>
|
|
apiClient
|
|
.post<{ access_token: string; refresh_token: string }>('/auth/refresh', { refresh_token })
|
|
.then((r) => r.data),
|
|
|
|
logout: (refresh_token: string) =>
|
|
apiClient.post('/auth/logout', { refresh_token }),
|
|
|
|
me: () => apiClient.get<User>('/auth/me').then((r) => r.data),
|
|
|
|
updateMe: (data: { full_name?: string; phone?: string; expo_push_token?: string }) =>
|
|
apiClient.patch<User>('/auth/me', data).then((r) => r.data),
|
|
};
|