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:
parent
b9b2b559a1
commit
6962df6340
14 changed files with 119 additions and 23 deletions
|
@ -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
|
|
@ -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
|
||||||
|
|
|
@ -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:
|
||||||
|
|
|
@ -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:
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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');
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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(),
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
|
@ -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);
|
||||||
|
|
13
web/src/routes/.well-known/immich/+server.ts
Normal file
13
web/src/routes/.well-known/immich/+server.ts
Normal 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
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
Loading…
Reference in a new issue