1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-07 20:36:48 +01:00

feat(.well-known): add .well-known/immich to reference API endpoint

This commit is contained in:
Connery Noble 2023-01-11 17:30:01 -08:00
parent b9b2b559a1
commit 6962df6340
14 changed files with 119 additions and 23 deletions

View file

@ -76,3 +76,21 @@ PUBLIC_LOGIN_PAGE_MESSAGE=
IMMICH_WEB_URL=http://immich-web:3000 IMMICH_WEB_URL=http://immich-web:3000
IMMICH_SERVER_URL=http://immich-server:3001 IMMICH_SERVER_URL=http://immich-server:3001
IMMICH_MACHINE_LEARNING_URL=http://immich-machine-learning:3003 IMMICH_MACHINE_LEARNING_URL=http://immich-machine-learning:3003
####################################################################################
# Public Facing API Server URL - Optional
#
# This should point to the publicly accessible URL of the API (default is "/api")
# Use a full URL here if you want to serve the API via a dedicated domain.
####################################################################################
IMMICH_SERVER_PUBLIC_URL=/api
####################################################################################
# API Cors Allowed Origins - Optional
#
# This is required if the API is served from a different domain than the website.
# For example, if IMMICH_SERVER_PUBLIC_URL is set to a full URL.
####################################################################################
ALLOW_CORS_ORIGIN=http://localhost:2283

View file

@ -14,6 +14,7 @@ services:
- ${UPLOAD_LOCATION}:/usr/src/app/upload - ${UPLOAD_LOCATION}:/usr/src/app/upload
- /usr/src/app/node_modules - /usr/src/app/node_modules
ports: ports:
- 3001:3001
- 9230:9230 - 9230:9230
env_file: env_file:
- .env - .env
@ -75,6 +76,7 @@ services:
environment: environment:
# Rename these values for svelte public interface # Rename these values for svelte public interface
- PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL} - PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL}
- PUBLIC_IMMICH_SERVER_PUBLIC_URL=${IMMICH_SERVER_PUBLIC_URL}
ports: ports:
- 3000:3000 - 3000:3000
- 24678:24678 - 24678:24678

View file

@ -54,6 +54,7 @@ services:
environment: environment:
# Rename these values for svelte public interface # Rename these values for svelte public interface
- PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL} - PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL}
- PUBLIC_IMMICH_SERVER_PUBLIC_URL=${IMMICH_SERVER_PUBLIC_URL}
restart: always restart: always
redis: redis:

View file

@ -54,6 +54,7 @@ services:
environment: environment:
# Rename these values for svelte public interface # Rename these values for svelte public interface
- PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL} - PUBLIC_IMMICH_SERVER_URL=${IMMICH_SERVER_URL}
- PUBLIC_IMMICH_SERVER_PUBLIC_URL=${IMMICH_SERVER_PUBLIC_URL}
restart: always restart: always
redis: redis:

View file

@ -54,21 +54,15 @@ class AuthenticationNotifier extends StateNotifier<AuthenticationState> {
Future<bool> login( Future<bool> login(
String email, String email,
String password, String password,
String serverEndpoint, String serverUrl,
bool isSavedLoginInfo, bool isSavedLoginInfo,
) async { ) async {
// Store server endpoint to Hive and test endpoint
if (serverEndpoint[serverEndpoint.length - 1] == "/") {
var validUrl = serverEndpoint.substring(0, serverEndpoint.length - 1);
Hive.box(userInfoBox).put(serverEndpointKey, validUrl);
} else {
Hive.box(userInfoBox).put(serverEndpointKey, serverEndpoint);
}
// Check Server URL validity
try { try {
_apiService.setEndpoint(Hive.box(userInfoBox).get(serverEndpointKey)); // Resolve API server endpoint from user provided serverUrl
final serverEndpoint = await _apiService.resolveEndpoint(serverUrl);
_apiService.setEndpoint(serverEndpoint);
await _apiService.serverInfoApi.pingServer(); await _apiService.serverInfoApi.pingServer();
Hive.box(userInfoBox).put(serverEndpointKey, serverEndpoint);
} catch (e) { } catch (e) {
debugPrint('Invalid Server Endpoint Url $e'); debugPrint('Invalid Server Endpoint Url $e');
return false; return false;
@ -90,7 +84,7 @@ class AuthenticationNotifier extends StateNotifier<AuthenticationState> {
return setSuccessLoginInfo( return setSuccessLoginInfo(
accessToken: loginResponse.accessToken, accessToken: loginResponse.accessToken,
serverUrl: serverEndpoint, serverUrl: serverUrl,
isSavedLoginInfo: isSavedLoginInfo, isSavedLoginInfo: isSavedLoginInfo,
); );
} catch (e) { } catch (e) {
@ -174,7 +168,6 @@ class AuthenticationNotifier extends StateNotifier<AuthenticationState> {
var deviceInfo = await _deviceInfoService.getDeviceInfo(); var deviceInfo = await _deviceInfoService.getDeviceInfo();
userInfoHiveBox.put(deviceIdKey, deviceInfo["deviceId"]); userInfoHiveBox.put(deviceIdKey, deviceInfo["deviceId"]);
userInfoHiveBox.put(accessTokenKey, accessToken); userInfoHiveBox.put(accessTokenKey, accessToken);
userInfoHiveBox.put(serverEndpointKey, serverUrl);
state = state.copyWith( state = state.copyWith(
isAuthenticated: true, isAuthenticated: true,

View file

@ -11,8 +11,10 @@ class OAuthService {
OAuthService(this._apiService); OAuthService(this._apiService);
Future<OAuthConfigResponseDto?> getOAuthServerConfig( Future<OAuthConfigResponseDto?> getOAuthServerConfig(
String serverEndpoint, String serverUrl,
) async { ) async {
// Resolve API server endpoint from user provided serverUrl
final serverEndpoint = await _apiService.resolveEndpoint(serverUrl);
_apiService.setEndpoint(serverEndpoint); _apiService.setEndpoint(serverEndpoint);
return await _apiService.oAuthApi.generateConfig( return await _apiService.oAuthApi.generateConfig(

View file

@ -38,13 +38,17 @@ class LoginForm extends HookConsumerWidget {
var urlText = serverEndpointController.text.trim(); var urlText = serverEndpointController.text.trim();
try { try {
var endpointUrl = Uri.tryParse(urlText); var serverUrl = Uri.tryParse(urlText);
if (endpointUrl != null) { if (serverUrl != null) {
isLoading.value = true; isLoading.value = true;
apiService.setEndpoint(endpointUrl.toString()); final serverEndpoint =
await apiService.resolveEndpoint(serverUrl.toString());
apiService.setEndpoint(serverEndpoint);
Hive.box(userInfoBox).put(serverEndpointKey, serverEndpoint);
var loginConfig = await apiService.oAuthApi.generateConfig( var loginConfig = await apiService.oAuthApi.generateConfig(
OAuthConfigDto(redirectUri: endpointUrl.toString()), OAuthConfigDto(redirectUri: serverEndpoint),
); );
if (loginConfig != null) { if (loginConfig != null) {

View file

@ -58,14 +58,15 @@ class WebsocketNotifier extends StateNotifier<WebsocketState> {
if (authenticationState.isAuthenticated) { if (authenticationState.isAuthenticated) {
var accessToken = Hive.box(userInfoBox).get(accessTokenKey); var accessToken = Hive.box(userInfoBox).get(accessTokenKey);
var endpoint = Hive.box(userInfoBox).get(serverEndpointKey);
try { try {
var endpoint = Uri.parse(Hive.box(userInfoBox).get(serverEndpointKey));
debugPrint("Attempting to connect to websocket"); debugPrint("Attempting to connect to websocket");
// Configure socket transports must be specified // Configure socket transports must be specified
Socket socket = io( Socket socket = io(
endpoint.toString().replaceAll('/api', ''), endpoint.origin,
OptionBuilder() OptionBuilder()
.setPath('/api/socket.io') .setPath("${endpoint.path}/socket.io")
.setTransports(['websocket']) .setTransports(['websocket'])
.enableReconnection() .enableReconnection()
.enableForceNew() .enableForceNew()

View file

@ -1,4 +1,8 @@
import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:openapi/api.dart'; import 'package:openapi/api.dart';
import 'package:http/http.dart';
class ApiService { class ApiService {
late ApiClient _apiClient; late ApiClient _apiClient;
@ -22,6 +26,47 @@ class ApiService {
deviceInfoApi = DeviceInfoApi(_apiClient); deviceInfoApi = DeviceInfoApi(_apiClient);
} }
resolveEndpoint(String serverUrl) async {
// Sanitize URL to only include origin+path
final url = Uri.parse(serverUrl);
final baseUrl = "${url.origin}${url.path}";
// Remove trailing slash, if exists
final endpoint = baseUrl[baseUrl.length - 1] == "/"
? baseUrl.substring(0, baseUrl.length - 1)
: baseUrl;
// Check for .well-known definition, otherwise assume endpoint is full API address
final apiEndpoint = await getWellKnownEndpoint(endpoint) ?? endpoint;
return apiEndpoint;
}
getWellKnownEndpoint(String baseUrl) async {
final Client client = Client();
try {
final res = await client.get(
Uri.parse("$baseUrl/.well-known/immich"),
headers: {"Accept": "application/json"},
);
if (res.statusCode == 200) {
final data = jsonDecode(res.body);
final endpoint = data['api']['endpoint'] as String;
if (endpoint.startsWith('/')) {
// Full URL is relative to base
return "$baseUrl$endpoint";
}
return endpoint;
}
} catch (e) {
debugPrint("Could not locate .well-known at $baseUrl: $e");
}
return null;
}
setAccessToken(String accessToken) { setAccessToken(String accessToken) {
_apiClient.addDefaultHeader('Authorization', 'Bearer $accessToken'); _apiClient.addDefaultHeader('Authorization', 'Bearer $accessToken');
} }

View file

@ -22,8 +22,11 @@ class SplashScreenPage extends HookConsumerWidget {
void performLoggingIn() async { void performLoggingIn() async {
try { try {
if (loginInfo != null) { if (loginInfo != null) {
// Make sure API service is initialized // Resolve API server endpoint from user provided serverUrl
apiService.setEndpoint(loginInfo.serverUrl); final serverEndpoint =
await apiService.resolveEndpoint(loginInfo.serverUrl);
apiService.setEndpoint(serverEndpoint);
Hive.box(userInfoBox).put(serverEndpointKey, serverEndpoint);
var isSuccess = await ref var isSuccess = await ref
.read(authenticationProvider.notifier) .read(authenticationProvider.notifier)

View file

@ -25,6 +25,10 @@ async function bootstrap() {
app.use(json({ limit: '10mb' })); app.use(json({ limit: '10mb' }));
if (process.env.NODE_ENV === 'development') { if (process.env.NODE_ENV === 'development') {
app.enableCors(); app.enableCors();
} else if (process.env.ALLOW_CORS_ORIGIN) {
app.enableCors({
origin: process.env.ALLOW_CORS_ORIGIN
});
} }
const redisIoAdapter = new RedisIoAdapter(app); const redisIoAdapter = new RedisIoAdapter(app);

View file

@ -28,5 +28,6 @@ export const immichAppConfig: ConfigModuleOptions = {
DISABLE_REVERSE_GEOCODING: Joi.boolean().optional().valid(true, false).default(false), DISABLE_REVERSE_GEOCODING: Joi.boolean().optional().valid(true, false).default(false),
REVERSE_GEOCODING_PRECISION: Joi.number().optional().valid(0, 1, 2, 3).default(3), REVERSE_GEOCODING_PRECISION: Joi.number().optional().valid(0, 1, 2, 3).default(3),
LOG_LEVEL: Joi.string().optional().valid('simple', 'verbose', 'debug', 'log', 'warn', 'error').default('log'), LOG_LEVEL: Joi.string().optional().valid('simple', 'verbose', 'debug', 'log', 'warn', 'error').default('log'),
ALLOW_CORS_ORIGIN: Joi.string().optional(),
}), }),
}; };

View file

@ -54,9 +54,17 @@ class ImmichApi {
public setBaseUrl(baseUrl: string) { public setBaseUrl(baseUrl: string) {
this.config.basePath = baseUrl; this.config.basePath = baseUrl;
} }
public getBaseUrl() {
return this.config.basePath;
}
} }
// Browser side (public) API client
export const api = new ImmichApi(); export const api = new ImmichApi();
api.setBaseUrl(env.PUBLIC_IMMICH_SERVER_PUBLIC_URL || '/api');
// Server side API client
export const serverApi = new ImmichApi(); export const serverApi = new ImmichApi();
const immich_server_url = env.PUBLIC_IMMICH_SERVER_URL || 'http://immich-server:3001'; const immich_server_url = env.PUBLIC_IMMICH_SERVER_URL || 'http://immich-server:3001';
serverApi.setBaseUrl(immich_server_url); serverApi.setBaseUrl(immich_server_url);

View file

@ -0,0 +1,13 @@
import { json } from '@sveltejs/kit';
import { api } from '@api';
const endpoint = api.getBaseUrl();
export const prerender = true;
export const GET = async () => {
return json({
api: {
endpoint
}
});
};