mirror of
https://git.selfprivacy.org/kherel/selfprivacy.org.app.git
synced 2024-09-18 17:37:50 +00:00
811 lines
25 KiB
Dart
811 lines
25 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:device_info_plus/device_info_plus.dart';
|
|
import 'package:dio/dio.dart';
|
|
import 'package:easy_localization/easy_localization.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:hive/hive.dart';
|
|
import 'package:pub_semver/pub_semver.dart';
|
|
import 'package:selfprivacy/config/get_it_config.dart';
|
|
import 'package:selfprivacy/config/hive_config.dart';
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/api_controller.dart';
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/api_factory_settings.dart';
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/dns_providers/dns_provider.dart';
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/dns_providers/dns_provider_api_settings.dart';
|
|
import 'package:selfprivacy/logic/api_maps/graphql_maps/server_api/server_api.dart';
|
|
import 'package:selfprivacy/logic/api_maps/rest_maps/server_providers/server_provider.dart';
|
|
import 'package:selfprivacy/logic/api_maps/staging_options.dart';
|
|
import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart';
|
|
import 'package:selfprivacy/logic/models/hive/backblaze_credential.dart';
|
|
import 'package:selfprivacy/logic/models/hive/server_details.dart';
|
|
import 'package:selfprivacy/logic/models/hive/server_domain.dart';
|
|
import 'package:selfprivacy/logic/models/hive/user.dart';
|
|
import 'package:selfprivacy/logic/models/json/device_token.dart';
|
|
import 'package:selfprivacy/logic/models/json/dns_records.dart';
|
|
import 'package:selfprivacy/logic/models/server_basic_info.dart';
|
|
import 'package:selfprivacy/logic/models/server_type.dart';
|
|
import 'package:selfprivacy/ui/helpers/modals.dart';
|
|
import 'package:selfprivacy/utils/network_utils.dart';
|
|
|
|
class IpNotFoundException implements Exception {
|
|
IpNotFoundException(this.message);
|
|
final String message;
|
|
}
|
|
|
|
class ServerAuthorizationException implements Exception {
|
|
ServerAuthorizationException(this.message);
|
|
final String message;
|
|
}
|
|
|
|
class ServerInstallationRepository {
|
|
Box box = Hive.box(BNames.serverInstallationBox);
|
|
Box<User> usersBox = Hive.box(BNames.usersBox);
|
|
|
|
Future<ServerInstallationState> load() async {
|
|
final String? providerApiToken = getIt<ApiConfigModel>().serverProviderKey;
|
|
final String? location = getIt<ApiConfigModel>().serverLocation;
|
|
final String? dnsApiToken = getIt<ApiConfigModel>().dnsProviderKey;
|
|
final String? serverTypeIdentificator = getIt<ApiConfigModel>().serverType;
|
|
final ServerDomain? serverDomain = getIt<ApiConfigModel>().serverDomain;
|
|
final ServerProvider? serverProvider =
|
|
getIt<ApiConfigModel>().serverProvider;
|
|
final BackblazeCredential? backblazeCredential =
|
|
getIt<ApiConfigModel>().backblazeCredential;
|
|
final ServerHostingDetails? serverDetails =
|
|
getIt<ApiConfigModel>().serverDetails;
|
|
final DnsProvider? dnsProvider = getIt<ApiConfigModel>().dnsProvider;
|
|
|
|
if (serverProvider != null ||
|
|
(serverDetails != null &&
|
|
serverDetails.provider != ServerProvider.unknown)) {
|
|
ApiController.initServerProviderApiFactory(
|
|
ServerProviderApiFactorySettings(
|
|
provider: serverProvider ?? serverDetails!.provider,
|
|
location: location,
|
|
),
|
|
);
|
|
|
|
// All current providers support volumes
|
|
// so it's safe to hardcode for now
|
|
ApiController.initVolumeProviderApiFactory(
|
|
ServerProviderApiFactorySettings(
|
|
provider: serverProvider ?? serverDetails!.provider,
|
|
location: location,
|
|
),
|
|
);
|
|
}
|
|
|
|
if (ApiController.currentDnsProviderApiFactory == null) {
|
|
if (dnsProvider != null ||
|
|
(serverDomain != null &&
|
|
serverDomain.provider != DnsProvider.unknown)) {
|
|
ApiController.initDnsProviderApiFactory(
|
|
DnsProviderApiFactorySettings(
|
|
provider: dnsProvider ?? serverDomain!.provider,
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
if (box.get(BNames.hasFinalChecked, defaultValue: false)) {
|
|
StagingOptions.verifyCertificate = true;
|
|
return ServerInstallationFinished(
|
|
providerApiToken: providerApiToken!,
|
|
serverTypeIdentificator: serverTypeIdentificator ?? '',
|
|
dnsApiToken: dnsApiToken!,
|
|
serverDomain: serverDomain!,
|
|
backblazeCredential: backblazeCredential!,
|
|
serverDetails: serverDetails!,
|
|
rootUser: box.get(BNames.rootUser),
|
|
isServerStarted: box.get(BNames.isServerStarted, defaultValue: false),
|
|
isServerResetedFirstTime:
|
|
box.get(BNames.isServerResetedFirstTime, defaultValue: false),
|
|
isServerResetedSecondTime:
|
|
box.get(BNames.isServerResetedSecondTime, defaultValue: false),
|
|
);
|
|
}
|
|
|
|
if (box.get(BNames.isRecoveringServer, defaultValue: false) &&
|
|
serverDomain != null) {
|
|
return ServerInstallationRecovery(
|
|
providerApiToken: providerApiToken,
|
|
dnsApiToken: dnsApiToken,
|
|
serverDomain: serverDomain,
|
|
backblazeCredential: backblazeCredential,
|
|
serverDetails: serverDetails,
|
|
rootUser: box.get(BNames.rootUser),
|
|
currentStep: _getCurrentRecoveryStep(
|
|
providerApiToken,
|
|
dnsApiToken,
|
|
serverDomain,
|
|
serverDetails,
|
|
),
|
|
recoveryCapabilities: await getRecoveryCapabilities(serverDomain),
|
|
);
|
|
}
|
|
|
|
return ServerInstallationNotFinished(
|
|
providerApiToken: providerApiToken,
|
|
dnsApiToken: dnsApiToken,
|
|
serverDomain: serverDomain,
|
|
backblazeCredential: backblazeCredential,
|
|
serverDetails: serverDetails,
|
|
rootUser: box.get(BNames.rootUser),
|
|
isServerStarted: box.get(BNames.isServerStarted, defaultValue: false),
|
|
isServerResetedFirstTime:
|
|
box.get(BNames.isServerResetedFirstTime, defaultValue: false),
|
|
isServerResetedSecondTime:
|
|
box.get(BNames.isServerResetedSecondTime, defaultValue: false),
|
|
isLoading: box.get(BNames.isLoading, defaultValue: false),
|
|
dnsMatches: null,
|
|
);
|
|
}
|
|
|
|
RecoveryStep _getCurrentRecoveryStep(
|
|
final String? serverProviderToken,
|
|
final String? cloudflareToken,
|
|
final ServerDomain serverDomain,
|
|
final ServerHostingDetails? serverDetails,
|
|
) {
|
|
if (serverDetails != null) {
|
|
if (serverProviderToken != null) {
|
|
if (serverDetails.provider != ServerProvider.unknown) {
|
|
if (serverDomain.provider != DnsProvider.unknown) {
|
|
return RecoveryStep.backblazeToken;
|
|
}
|
|
return RecoveryStep.dnsProviderToken;
|
|
}
|
|
return RecoveryStep.serverSelection;
|
|
}
|
|
return RecoveryStep.serverProviderToken;
|
|
}
|
|
return RecoveryStep.selecting;
|
|
}
|
|
|
|
void clearAppConfig() {
|
|
box.clear();
|
|
usersBox.clear();
|
|
}
|
|
|
|
Future<ServerHostingDetails> startServer(
|
|
final ServerHostingDetails server,
|
|
) async {
|
|
ServerHostingDetails serverDetails;
|
|
|
|
serverDetails = await ApiController.currentServerProviderApiFactory!
|
|
.getServerProvider()
|
|
.powerOn();
|
|
|
|
return serverDetails;
|
|
}
|
|
|
|
Future<String?> getDomainId(final String token, final String domain) async {
|
|
final DnsProviderApi dnsProviderApi =
|
|
ApiController.currentDnsProviderApiFactory!.getDnsProvider(
|
|
settings: DnsProviderApiSettings(
|
|
isWithToken: false,
|
|
customToken: token,
|
|
),
|
|
);
|
|
|
|
/// TODO: nvm it's because only Cloudflare uses Zone
|
|
/// for other providers we need to implement a different kind of
|
|
/// functionality here... but it's on refactoring, let it be here for now.
|
|
final APIGenericResult<bool> apiResponse =
|
|
await dnsProviderApi.isApiTokenValid(token);
|
|
|
|
String? domainId;
|
|
if (apiResponse.success && apiResponse.data) {
|
|
domainId = await dnsProviderApi.getZoneId(domain);
|
|
}
|
|
return domainId;
|
|
}
|
|
|
|
Future<Map<String, bool>> isDnsAddressesMatch(
|
|
final String? domainName,
|
|
final String? ip4,
|
|
final Map<String, bool> skippedMatches,
|
|
) async {
|
|
final Map<String, bool> matches = <String, bool>{};
|
|
await InternetAddress.lookup(domainName!).then(
|
|
(final records) {
|
|
for (final record in records) {
|
|
if (skippedMatches[record.host] ?? false) {
|
|
matches[record.host] = true;
|
|
continue;
|
|
}
|
|
if (record.address == ip4!) {
|
|
matches[record.host] = true;
|
|
}
|
|
}
|
|
},
|
|
);
|
|
|
|
return matches;
|
|
}
|
|
|
|
Future<void> createServer(
|
|
final User rootUser,
|
|
final String domainName,
|
|
final String dnsApiToken,
|
|
final BackblazeCredential backblazeCredential, {
|
|
required final void Function() onCancel,
|
|
required final Future<void> Function(ServerHostingDetails serverDetails)
|
|
onSuccess,
|
|
}) async {
|
|
final ServerProviderApi api =
|
|
ApiController.currentServerProviderApiFactory!.getServerProvider();
|
|
|
|
void showInstallationErrorPopUp() {
|
|
showPopUpAlert(
|
|
alertTitle: 'modals.unexpected_error'.tr(),
|
|
description: 'modals.try_again'.tr(),
|
|
actionButtonTitle: 'modals.yes'.tr(),
|
|
actionButtonOnPressed: () async {
|
|
ServerHostingDetails? serverDetails;
|
|
try {
|
|
final APIGenericResult createResult = await api.createServer(
|
|
dnsProvider: getIt<ApiConfigModel>().dnsProvider!,
|
|
dnsApiToken: dnsApiToken,
|
|
rootUser: rootUser,
|
|
domainName: domainName,
|
|
serverType: getIt<ApiConfigModel>().serverType!,
|
|
);
|
|
serverDetails = createResult.data;
|
|
} catch (e) {
|
|
print(e);
|
|
}
|
|
|
|
if (serverDetails == null) {
|
|
print('Server is not initialized!');
|
|
return;
|
|
}
|
|
await saveServerDetails(serverDetails);
|
|
onSuccess(serverDetails);
|
|
},
|
|
cancelButtonOnPressed: onCancel,
|
|
);
|
|
}
|
|
|
|
try {
|
|
final APIGenericResult<ServerHostingDetails?> createServerResult =
|
|
await api.createServer(
|
|
dnsProvider: getIt<ApiConfigModel>().dnsProvider!,
|
|
dnsApiToken: dnsApiToken,
|
|
rootUser: rootUser,
|
|
domainName: domainName,
|
|
serverType: getIt<ApiConfigModel>().serverType!,
|
|
);
|
|
|
|
if (createServerResult.data == null) {
|
|
const String e = 'Server is not initialized!';
|
|
print(e);
|
|
}
|
|
|
|
if (createServerResult.message == 'uniqueness_error') {
|
|
showPopUpAlert(
|
|
alertTitle: 'modals.already_exists'.tr(),
|
|
description: 'modals.destroy_server'.tr(),
|
|
actionButtonTitle: 'modals.yes'.tr(),
|
|
actionButtonOnPressed: () async {
|
|
await api.deleteServer(
|
|
domainName: domainName,
|
|
);
|
|
|
|
ServerHostingDetails? serverDetails;
|
|
try {
|
|
final APIGenericResult createResult = await api.createServer(
|
|
dnsProvider: getIt<ApiConfigModel>().dnsProvider!,
|
|
dnsApiToken: dnsApiToken,
|
|
rootUser: rootUser,
|
|
domainName: domainName,
|
|
serverType: getIt<ApiConfigModel>().serverType!,
|
|
);
|
|
serverDetails = createResult.data;
|
|
} catch (e) {
|
|
print(e);
|
|
}
|
|
|
|
if (serverDetails == null) {
|
|
print('Server is not initialized!');
|
|
return;
|
|
}
|
|
await saveServerDetails(serverDetails);
|
|
onSuccess(serverDetails);
|
|
},
|
|
cancelButtonOnPressed: onCancel,
|
|
);
|
|
return;
|
|
}
|
|
|
|
saveServerDetails(createServerResult.data!);
|
|
onSuccess(createServerResult.data!);
|
|
} catch (e) {
|
|
print(e);
|
|
showInstallationErrorPopUp();
|
|
}
|
|
}
|
|
|
|
Future<bool> createDnsRecords(
|
|
final ServerHostingDetails serverDetails,
|
|
final ServerDomain domain, {
|
|
required final void Function() onCancel,
|
|
}) async {
|
|
final DnsProviderApi dnsProviderApi =
|
|
ApiController.currentDnsProviderApiFactory!.getDnsProvider();
|
|
final ServerProviderApi serverApi =
|
|
ApiController.currentServerProviderApiFactory!.getServerProvider();
|
|
|
|
void showDomainErrorPopUp(final String error) {
|
|
showPopUpAlert(
|
|
alertTitle: error,
|
|
description: 'modals.delete_server_volume'.tr(),
|
|
cancelButtonOnPressed: onCancel,
|
|
actionButtonTitle: 'basis.delete'.tr(),
|
|
actionButtonOnPressed: () async {
|
|
await serverApi.deleteServer(
|
|
domainName: domain.domainName,
|
|
);
|
|
onCancel();
|
|
},
|
|
);
|
|
}
|
|
|
|
final APIGenericResult removingResult =
|
|
await dnsProviderApi.removeSimilarRecords(
|
|
ip4: serverDetails.ip4,
|
|
domain: domain,
|
|
);
|
|
|
|
if (!removingResult.success) {
|
|
showDomainErrorPopUp('domain.error'.tr());
|
|
return false;
|
|
}
|
|
|
|
bool createdSuccessfully = false;
|
|
String errorMessage = 'domain.error'.tr();
|
|
try {
|
|
final APIGenericResult createResult =
|
|
await dnsProviderApi.createMultipleDnsRecords(
|
|
ip4: serverDetails.ip4,
|
|
domain: domain,
|
|
);
|
|
createdSuccessfully = createResult.success;
|
|
} on DioError catch (e) {
|
|
if (e.response!.data['errors'][0]['code'] == 1038) {
|
|
errorMessage = 'modals.you_cant_use_this_api'.tr();
|
|
}
|
|
}
|
|
|
|
if (!createdSuccessfully) {
|
|
showDomainErrorPopUp(errorMessage);
|
|
return false;
|
|
}
|
|
|
|
final APIGenericResult createReverseResult =
|
|
await serverApi.createReverseDns(
|
|
serverDetails: serverDetails,
|
|
domain: domain,
|
|
);
|
|
|
|
if (!createReverseResult.success) {
|
|
showDomainErrorPopUp(errorMessage);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
Future<void> createDkimRecord(final ServerDomain cloudFlareDomain) async {
|
|
final DnsProviderApi dnsProviderApi =
|
|
ApiController.currentDnsProviderApiFactory!.getDnsProvider();
|
|
final ServerApi api = ServerApi();
|
|
|
|
late DnsRecord record;
|
|
try {
|
|
record = extractDkimRecord(await api.getDnsRecords())!;
|
|
} catch (e) {
|
|
print(e);
|
|
rethrow;
|
|
}
|
|
|
|
await dnsProviderApi.setDnsRecord(record, cloudFlareDomain);
|
|
}
|
|
|
|
Future<bool> isHttpServerWorking() async {
|
|
final ServerApi api = ServerApi();
|
|
return api.isHttpServerWorking();
|
|
}
|
|
|
|
Future<ServerHostingDetails> restart() async =>
|
|
ApiController.currentServerProviderApiFactory!
|
|
.getServerProvider()
|
|
.restart();
|
|
|
|
Future<ServerHostingDetails> powerOn() async =>
|
|
ApiController.currentServerProviderApiFactory!
|
|
.getServerProvider()
|
|
.powerOn();
|
|
|
|
Future<ServerRecoveryCapabilities> getRecoveryCapabilities(
|
|
final ServerDomain serverDomain,
|
|
) async {
|
|
final ServerApi serverApi = ServerApi(
|
|
isWithToken: false,
|
|
overrideDomain: serverDomain.domainName,
|
|
);
|
|
final String? serverApiVersion = await serverApi.getApiVersion();
|
|
if (serverApiVersion == null) {
|
|
return ServerRecoveryCapabilities.none;
|
|
}
|
|
try {
|
|
final Version parsedVersion = Version.parse(serverApiVersion);
|
|
if (!VersionConstraint.parse('>=1.2.0').allows(parsedVersion)) {
|
|
return ServerRecoveryCapabilities.legacy;
|
|
}
|
|
return ServerRecoveryCapabilities.loginTokens;
|
|
} on FormatException {
|
|
return ServerRecoveryCapabilities.none;
|
|
}
|
|
}
|
|
|
|
Future<String> getServerIpFromDomain(final ServerDomain serverDomain) async {
|
|
String? domain;
|
|
await InternetAddress.lookup(serverDomain.domainName).then(
|
|
(final records) {
|
|
for (final record in records) {
|
|
domain = record.address;
|
|
}
|
|
},
|
|
);
|
|
if (domain == null || domain!.isEmpty) {
|
|
throw IpNotFoundException('No IP found for domain $serverDomain');
|
|
}
|
|
return domain!;
|
|
}
|
|
|
|
Future<String> getDeviceName() async {
|
|
final DeviceInfoPlugin deviceInfo = DeviceInfoPlugin();
|
|
if (kIsWeb) {
|
|
return deviceInfo.webBrowserInfo.then(
|
|
(final WebBrowserInfo value) =>
|
|
'${value.browserName} ${value.platform}',
|
|
);
|
|
} else {
|
|
if (Platform.isAndroid) {
|
|
return deviceInfo.androidInfo.then(
|
|
(final AndroidDeviceInfo value) =>
|
|
'${value.model} ${value.version.release}',
|
|
);
|
|
} else if (Platform.isIOS) {
|
|
return deviceInfo.iosInfo.then(
|
|
(final IosDeviceInfo value) =>
|
|
'${value.utsname.machine} ${value.systemName} ${value.systemVersion}',
|
|
);
|
|
} else if (Platform.isLinux) {
|
|
return deviceInfo.linuxInfo
|
|
.then((final LinuxDeviceInfo value) => value.prettyName);
|
|
} else if (Platform.isMacOS) {
|
|
return deviceInfo.macOsInfo.then(
|
|
(final MacOsDeviceInfo value) =>
|
|
'${value.hostName} ${value.computerName}',
|
|
);
|
|
} else if (Platform.isWindows) {
|
|
return deviceInfo.windowsInfo
|
|
.then((final WindowsDeviceInfo value) => value.computerName);
|
|
}
|
|
}
|
|
return 'Unidentified';
|
|
}
|
|
|
|
Future<ServerHostingDetails> authorizeByNewDeviceKey(
|
|
final ServerDomain serverDomain,
|
|
final String newDeviceKey,
|
|
final ServerRecoveryCapabilities recoveryCapabilities,
|
|
) async {
|
|
final ServerApi serverApi = ServerApi(
|
|
isWithToken: false,
|
|
overrideDomain: serverDomain.domainName,
|
|
);
|
|
final String serverIp = await getServerIpFromDomain(serverDomain);
|
|
final APIGenericResult<String> result = await serverApi.authorizeDevice(
|
|
DeviceToken(device: await getDeviceName(), token: newDeviceKey),
|
|
);
|
|
|
|
if (result.success) {
|
|
return ServerHostingDetails(
|
|
apiToken: result.data,
|
|
volume: ServerVolume(
|
|
id: 0,
|
|
name: '',
|
|
sizeByte: 0,
|
|
serverId: 0,
|
|
linuxDevice: '',
|
|
),
|
|
provider: ServerProvider.unknown,
|
|
id: 0,
|
|
ip4: serverIp,
|
|
startTime: null,
|
|
createTime: null,
|
|
);
|
|
}
|
|
|
|
throw ServerAuthorizationException(
|
|
result.message ?? result.data,
|
|
);
|
|
}
|
|
|
|
Future<ServerHostingDetails> authorizeByRecoveryKey(
|
|
final ServerDomain serverDomain,
|
|
final String recoveryKey,
|
|
final ServerRecoveryCapabilities recoveryCapabilities,
|
|
) async {
|
|
final ServerApi serverApi = ServerApi(
|
|
isWithToken: false,
|
|
overrideDomain: serverDomain.domainName,
|
|
);
|
|
final String serverIp = await getServerIpFromDomain(serverDomain);
|
|
final APIGenericResult<String> result = await serverApi.useRecoveryToken(
|
|
DeviceToken(device: await getDeviceName(), token: recoveryKey),
|
|
);
|
|
|
|
if (result.success) {
|
|
return ServerHostingDetails(
|
|
apiToken: result.data,
|
|
volume: ServerVolume(
|
|
id: 0,
|
|
name: '',
|
|
sizeByte: 0,
|
|
serverId: 0,
|
|
linuxDevice: '',
|
|
),
|
|
provider: ServerProvider.unknown,
|
|
id: 0,
|
|
ip4: serverIp,
|
|
startTime: null,
|
|
createTime: null,
|
|
);
|
|
}
|
|
|
|
throw ServerAuthorizationException(
|
|
result.message ?? result.data,
|
|
);
|
|
}
|
|
|
|
Future<ServerHostingDetails> authorizeByApiToken(
|
|
final ServerDomain serverDomain,
|
|
final String apiToken,
|
|
final ServerRecoveryCapabilities recoveryCapabilities,
|
|
) async {
|
|
final ServerApi serverApi = ServerApi(
|
|
isWithToken: true,
|
|
overrideDomain: serverDomain.domainName,
|
|
customToken: apiToken,
|
|
);
|
|
final String serverIp = await getServerIpFromDomain(serverDomain);
|
|
if (recoveryCapabilities == ServerRecoveryCapabilities.legacy) {
|
|
if (await serverApi.isHttpServerWorking()) {
|
|
return ServerHostingDetails(
|
|
apiToken: apiToken,
|
|
volume: ServerVolume(
|
|
id: 0,
|
|
name: '',
|
|
serverId: 0,
|
|
sizeByte: 0,
|
|
linuxDevice: '',
|
|
),
|
|
provider: ServerProvider.unknown,
|
|
id: 0,
|
|
ip4: serverIp,
|
|
startTime: null,
|
|
createTime: null,
|
|
);
|
|
} else {
|
|
throw ServerAuthorizationException(
|
|
"Couldn't connect to server with this token",
|
|
);
|
|
}
|
|
}
|
|
final APIGenericResult<String> deviceAuthKey =
|
|
await serverApi.createDeviceToken();
|
|
final APIGenericResult<String> result = await serverApi.authorizeDevice(
|
|
DeviceToken(device: await getDeviceName(), token: deviceAuthKey.data),
|
|
);
|
|
|
|
if (result.success) {
|
|
return ServerHostingDetails(
|
|
apiToken: result.data,
|
|
volume: ServerVolume(
|
|
id: 0,
|
|
name: '',
|
|
sizeByte: 0,
|
|
serverId: 0,
|
|
linuxDevice: '',
|
|
),
|
|
provider: ServerProvider.unknown,
|
|
id: 0,
|
|
ip4: serverIp,
|
|
startTime: null,
|
|
createTime: null,
|
|
);
|
|
}
|
|
|
|
throw ServerAuthorizationException(
|
|
result.message ?? result.data,
|
|
);
|
|
}
|
|
|
|
Future<User> getMainUser() async {
|
|
final ServerApi serverApi = ServerApi();
|
|
const User fallbackUser = User(
|
|
isFoundOnServer: false,
|
|
type: UserType.primary,
|
|
note: "Couldn't find main user on server, API is outdated",
|
|
login: 'UNKNOWN',
|
|
sshKeys: [],
|
|
);
|
|
|
|
final String? serverApiVersion = await serverApi.getApiVersion();
|
|
final users = await serverApi.getAllUsers();
|
|
if (serverApiVersion == null || users.isEmpty) {
|
|
return fallbackUser;
|
|
}
|
|
try {
|
|
final Version parsedVersion = Version.parse(serverApiVersion);
|
|
if (!VersionConstraint.parse('>=1.2.5').allows(parsedVersion)) {
|
|
return fallbackUser;
|
|
}
|
|
return users.firstWhere(
|
|
(final User user) => user.type == UserType.primary,
|
|
);
|
|
} on FormatException {
|
|
return fallbackUser;
|
|
}
|
|
}
|
|
|
|
Future<List<ServerBasicInfo>> getServersOnProviderAccount() async =>
|
|
ApiController.currentServerProviderApiFactory!
|
|
.getServerProvider()
|
|
.getServers();
|
|
|
|
Future<void> saveServerDetails(
|
|
final ServerHostingDetails serverDetails,
|
|
) async {
|
|
await getIt<ApiConfigModel>().storeServerDetails(serverDetails);
|
|
}
|
|
|
|
Future<void> deleteServerDetails() async {
|
|
await box.delete(BNames.serverDetails);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
|
|
Future<void> saveServerProviderType(final ServerProvider type) async {
|
|
await getIt<ApiConfigModel>().storeServerProviderType(type);
|
|
}
|
|
|
|
Future<void> saveServerProviderKey(final String key) async {
|
|
await getIt<ApiConfigModel>().storeServerProviderKey(key);
|
|
}
|
|
|
|
Future<void> saveServerType(final ServerType serverType) async {
|
|
await getIt<ApiConfigModel>().storeServerTypeIdentifier(
|
|
serverType.identifier,
|
|
);
|
|
await getIt<ApiConfigModel>().storeServerLocation(
|
|
serverType.location.identifier,
|
|
);
|
|
}
|
|
|
|
Future<void> deleteServerProviderKey() async {
|
|
await box.delete(BNames.hetznerKey);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
|
|
Future<void> saveDnsProviderType(final DnsProvider type) async {
|
|
await getIt<ApiConfigModel>().storeDnsProviderType(type);
|
|
}
|
|
|
|
Future<void> saveBackblazeKey(
|
|
final BackblazeCredential backblazeCredential,
|
|
) async {
|
|
await getIt<ApiConfigModel>().storeBackblazeCredential(backblazeCredential);
|
|
}
|
|
|
|
Future<void> deleteBackblazeKey() async {
|
|
await box.delete(BNames.backblazeCredential);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
|
|
Future<void> saveDnsProviderKey(final String key) async {
|
|
await getIt<ApiConfigModel>().storeDnsProviderKey(key);
|
|
}
|
|
|
|
Future<void> deleteDnsProviderKey() async {
|
|
await box.delete(BNames.cloudFlareKey);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
|
|
Future<void> saveDomain(final ServerDomain serverDomain) async {
|
|
await getIt<ApiConfigModel>().storeServerDomain(serverDomain);
|
|
}
|
|
|
|
Future<void> deleteDomain() async {
|
|
await box.delete(BNames.serverDomain);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
|
|
Future<void> saveIsServerStarted(final bool value) async {
|
|
await box.put(BNames.isServerStarted, value);
|
|
}
|
|
|
|
Future<void> saveIsServerResetedFirstTime(final bool value) async {
|
|
await box.put(BNames.isServerResetedFirstTime, value);
|
|
}
|
|
|
|
Future<void> saveIsServerResetedSecondTime(final bool value) async {
|
|
await box.put(BNames.isServerResetedSecondTime, value);
|
|
}
|
|
|
|
Future<void> saveRootUser(final User rootUser) async {
|
|
await box.put(BNames.rootUser, rootUser);
|
|
}
|
|
|
|
Future<void> saveIsRecoveringServer(final bool value) async {
|
|
await box.put(BNames.isRecoveringServer, value);
|
|
}
|
|
|
|
Future<void> saveHasFinalChecked(final bool value) async {
|
|
await box.put(BNames.hasFinalChecked, value);
|
|
}
|
|
|
|
Future<bool> deleteServer(final ServerDomain serverDomain) async {
|
|
final APIGenericResult<bool> deletionResult = await ApiController
|
|
.currentServerProviderApiFactory!
|
|
.getServerProvider()
|
|
.deleteServer(
|
|
domainName: serverDomain.domainName,
|
|
);
|
|
|
|
if (!deletionResult.success) {
|
|
getIt<NavigationService>()
|
|
.showSnackBar('modals.server_validators_error'.tr());
|
|
return false;
|
|
}
|
|
|
|
if (!deletionResult.data) {
|
|
getIt<NavigationService>()
|
|
.showSnackBar('modals.server_deletion_error'.tr());
|
|
return false;
|
|
}
|
|
|
|
await box.put(BNames.hasFinalChecked, false);
|
|
await box.put(BNames.isServerStarted, false);
|
|
await box.put(BNames.isServerResetedFirstTime, false);
|
|
await box.put(BNames.isServerResetedSecondTime, false);
|
|
await box.put(BNames.isLoading, false);
|
|
await box.put(BNames.serverDetails, null);
|
|
|
|
final APIGenericResult<void> removalResult = await ApiController
|
|
.currentDnsProviderApiFactory!
|
|
.getDnsProvider()
|
|
.removeSimilarRecords(domain: serverDomain);
|
|
|
|
if (!removalResult.success) {
|
|
getIt<NavigationService>().showSnackBar('modals.dns_removal_error'.tr());
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Future<void> deleteServerRelatedRecords() async {
|
|
await box.deleteAll([
|
|
BNames.serverDetails,
|
|
BNames.isServerStarted,
|
|
BNames.isServerResetedFirstTime,
|
|
BNames.isServerResetedSecondTime,
|
|
BNames.hasFinalChecked,
|
|
BNames.isLoading,
|
|
]);
|
|
getIt<ApiConfigModel>().init();
|
|
}
|
|
}
|