mirror of
https://github.com/ImranR98/Obtainium.git
synced 2025-07-16 14:46:44 +02:00
498 lines
14 KiB
Dart
498 lines
14 KiB
Dart
// Defines App sources and provides functions used to interact with them
|
|
// AppSource is an abstract class with a concrete implementation for each source
|
|
|
|
import 'dart:convert';
|
|
|
|
import 'package:html/dom.dart';
|
|
import 'package:http/http.dart';
|
|
import 'package:html/parser.dart';
|
|
|
|
class AppNames {
|
|
late String author;
|
|
late String name;
|
|
|
|
AppNames(this.author, this.name);
|
|
}
|
|
|
|
class APKDetails {
|
|
late String version;
|
|
late List<String> apkUrls;
|
|
|
|
APKDetails(this.version, this.apkUrls);
|
|
}
|
|
|
|
class App {
|
|
late String id;
|
|
late String url;
|
|
late String author;
|
|
late String name;
|
|
String? installedVersion;
|
|
late String latestVersion;
|
|
List<String> apkUrls = [];
|
|
late int preferredApkIndex;
|
|
App(this.id, this.url, this.author, this.name, this.installedVersion,
|
|
this.latestVersion, this.apkUrls, this.preferredApkIndex);
|
|
|
|
@override
|
|
String toString() {
|
|
return 'ID: $id URL: $url INSTALLED: $installedVersion LATEST: $latestVersion APK: $apkUrls';
|
|
}
|
|
|
|
factory App.fromJson(Map<String, dynamic> json) => App(
|
|
json['id'] as String,
|
|
json['url'] as String,
|
|
json['author'] as String,
|
|
json['name'] as String,
|
|
json['installedVersion'] == null
|
|
? null
|
|
: json['installedVersion'] as String,
|
|
json['latestVersion'] as String,
|
|
List<String>.from(jsonDecode(json['apkUrls'])),
|
|
json['preferredApkIndex'] == null
|
|
? 0
|
|
: json['preferredApkIndex'] as int,
|
|
);
|
|
|
|
Map<String, dynamic> toJson() => {
|
|
'id': id,
|
|
'url': url,
|
|
'author': author,
|
|
'name': name,
|
|
'installedVersion': installedVersion,
|
|
'latestVersion': latestVersion,
|
|
'apkUrls': jsonEncode(apkUrls),
|
|
'preferredApkIndex': preferredApkIndex
|
|
};
|
|
}
|
|
|
|
escapeRegEx(String s) {
|
|
return s.replaceAllMapped(RegExp(r'[.*+?^${}()|[\]\\]'), (x) {
|
|
return "\\${x[0]}";
|
|
});
|
|
}
|
|
|
|
const String couldNotFindReleases = 'Unable to fetch release info';
|
|
const String couldNotFindLatestVersion =
|
|
'Could not determine latest release version';
|
|
const String notValidURL = 'Not a valid URL';
|
|
const String noAPKFound = 'No APK found';
|
|
|
|
List<String> getLinksFromParsedHTML(
|
|
Document dom, RegExp hrefPattern, String prependToLinks) =>
|
|
dom
|
|
.querySelectorAll('a')
|
|
.where((element) {
|
|
if (element.attributes['href'] == null) return false;
|
|
return hrefPattern.hasMatch(element.attributes['href']!);
|
|
})
|
|
.map((e) => '$prependToLinks${e.attributes['href']!}')
|
|
.toList();
|
|
|
|
abstract class AppSource {
|
|
late String host;
|
|
String standardizeURL(String url);
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl);
|
|
AppNames getAppNames(String standardUrl);
|
|
}
|
|
|
|
class GitHub implements AppSource {
|
|
@override
|
|
late String host = 'github.com';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+');
|
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
|
if (match == null) {
|
|
throw notValidURL;
|
|
}
|
|
return url.substring(0, match.end);
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res = await get(Uri.parse(
|
|
'https://api.$host/repos${standardUrl.substring('https://$host'.length)}/releases'));
|
|
if (res.statusCode == 200) {
|
|
var releases = jsonDecode(res.body) as List<dynamic>;
|
|
// Right now, the latest non-prerelease version is picked
|
|
// If none exists, the latest prerelease version is picked
|
|
// In the future, the user could be given a choice
|
|
var nonPrereleaseReleases =
|
|
releases.where((element) => element['prerelease'] != true).toList();
|
|
var latestRelease = nonPrereleaseReleases.isNotEmpty
|
|
? nonPrereleaseReleases[0]
|
|
: releases.isNotEmpty
|
|
? releases[0]
|
|
: null;
|
|
if (latestRelease == null) {
|
|
throw couldNotFindReleases;
|
|
}
|
|
List<dynamic>? assets = latestRelease['assets'];
|
|
List<String>? apkUrlList = assets
|
|
?.map((e) {
|
|
return e['browser_download_url'] != null
|
|
? e['browser_download_url'] as String
|
|
: '';
|
|
})
|
|
.where((element) => element.toLowerCase().endsWith('.apk'))
|
|
.toList();
|
|
if (apkUrlList == null || apkUrlList.isEmpty) {
|
|
throw noAPKFound;
|
|
}
|
|
String? version = latestRelease['tag_name'];
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(version, apkUrlList);
|
|
} else {
|
|
if (res.headers['x-ratelimit-remaining'] == '0') {
|
|
throw 'Rate limit reached - try again in ${(int.parse(res.headers['x-ratelimit-reset'] ?? '1800000000') / 60000000).toString()} minutes';
|
|
}
|
|
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) {
|
|
String temp = standardUrl.substring(standardUrl.indexOf('://') + 3);
|
|
List<String> names = temp.substring(temp.indexOf('/') + 1).split('/');
|
|
return AppNames(names[0], names[1]);
|
|
}
|
|
}
|
|
|
|
class GitLab implements AppSource {
|
|
@override
|
|
late String host = 'gitlab.com';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+');
|
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
|
if (match == null) {
|
|
throw notValidURL;
|
|
}
|
|
return url.substring(0, match.end);
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom'));
|
|
if (res.statusCode == 200) {
|
|
var standardUri = Uri.parse(standardUrl);
|
|
var parsedHtml = parse(res.body);
|
|
var entry = parsedHtml.querySelector('entry');
|
|
var entryContent =
|
|
parse(parseFragment(entry?.querySelector('content')!.innerHtml).text);
|
|
var apkUrlList = [
|
|
...getLinksFromParsedHTML(
|
|
entryContent,
|
|
RegExp(
|
|
'^${escapeRegEx(standardUri.path)}/uploads/[^/]+/[^/]+\\.apk\$',
|
|
caseSensitive: false),
|
|
standardUri.origin),
|
|
// GitLab releases may contain links to externally hosted APKs
|
|
...getLinksFromParsedHTML(entryContent,
|
|
RegExp('/[^/]+\\.apk\$', caseSensitive: false), '')
|
|
.where((element) => Uri.parse(element).host != '')
|
|
.toList()
|
|
];
|
|
if (apkUrlList.isEmpty) {
|
|
throw noAPKFound;
|
|
}
|
|
|
|
var entryId = entry?.querySelector('id')?.innerHtml;
|
|
var version =
|
|
entryId == null ? null : Uri.parse(entryId).pathSegments.last;
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(version, apkUrlList);
|
|
} else {
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) {
|
|
// Same as GitHub
|
|
return GitHub().getAppNames(standardUrl);
|
|
}
|
|
}
|
|
|
|
class Signal implements AppSource {
|
|
@override
|
|
late String host = 'signal.org';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
return 'https://$host';
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res =
|
|
await get(Uri.parse('https://updates.$host/android/latest.json'));
|
|
if (res.statusCode == 200) {
|
|
var json = jsonDecode(res.body);
|
|
String? apkUrl = json['url'];
|
|
if (apkUrl == null) {
|
|
throw noAPKFound;
|
|
}
|
|
String? version = json['versionName'];
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(version, [apkUrl]);
|
|
} else {
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) => AppNames('Signal', 'Signal');
|
|
}
|
|
|
|
class FDroid implements AppSource {
|
|
@override
|
|
late String host = 'f-droid.org';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/packages/[^/]+');
|
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
|
if (match == null) {
|
|
throw notValidURL;
|
|
}
|
|
return url.substring(0, match.end);
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res = await get(Uri.parse(standardUrl));
|
|
if (res.statusCode == 200) {
|
|
var latestReleaseDiv =
|
|
parse(res.body).querySelector('#latest.package-version');
|
|
var apkUrl = latestReleaseDiv
|
|
?.querySelector('.package-version-download a')
|
|
?.attributes['href'];
|
|
if (apkUrl == null) {
|
|
throw noAPKFound;
|
|
}
|
|
var version = latestReleaseDiv
|
|
?.querySelector('.package-version-header b')
|
|
?.innerHtml
|
|
.split(' ')
|
|
.last;
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(version, [apkUrl]);
|
|
} else {
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) {
|
|
return AppNames('F-Droid', Uri.parse(standardUrl).pathSegments.last);
|
|
}
|
|
}
|
|
|
|
class Mullvad implements AppSource {
|
|
@override
|
|
late String host = 'mullvad.net';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
RegExp standardUrlRegEx = RegExp('^https?://$host');
|
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
|
if (match == null) {
|
|
throw notValidURL;
|
|
}
|
|
return url.substring(0, match.end);
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res = await get(Uri.parse('$standardUrl/en/download/android'));
|
|
if (res.statusCode == 200) {
|
|
var version = parse(res.body)
|
|
.querySelector('p.subtitle.is-6')
|
|
?.querySelector('a')
|
|
?.attributes['href']
|
|
?.split('/')
|
|
.last;
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(
|
|
version, ['https://mullvad.net/download/app/apk/latest']);
|
|
} else {
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) {
|
|
return AppNames('Mullvad-VPN', 'Mullvad-VPN');
|
|
}
|
|
}
|
|
|
|
class IzzyOnDroid implements AppSource {
|
|
@override
|
|
late String host = 'android.izzysoft.de';
|
|
|
|
@override
|
|
String standardizeURL(String url) {
|
|
RegExp standardUrlRegEx = RegExp('^https?://$host/repo/apk/[^/]+');
|
|
RegExpMatch? match = standardUrlRegEx.firstMatch(url.toLowerCase());
|
|
if (match == null) {
|
|
throw notValidURL;
|
|
}
|
|
return url.substring(0, match.end);
|
|
}
|
|
|
|
@override
|
|
Future<APKDetails> getLatestAPKDetails(String standardUrl) async {
|
|
Response res = await get(Uri.parse(standardUrl));
|
|
if (res.statusCode == 200) {
|
|
var parsedHtml = parse(res.body);
|
|
var multipleVersionApkUrls = parsedHtml
|
|
.querySelectorAll('a')
|
|
.where((element) =>
|
|
element.attributes['href']?.toLowerCase().endsWith('.apk') ??
|
|
false)
|
|
.map((e) => 'https://$host${e.attributes['href'] ?? ''}')
|
|
.toList();
|
|
if (multipleVersionApkUrls.isEmpty) {
|
|
throw noAPKFound;
|
|
}
|
|
var version = parsedHtml
|
|
.querySelector('#keydata')
|
|
?.querySelectorAll('b')
|
|
.where(
|
|
(element) => element.innerHtml.toLowerCase().contains('version'))
|
|
.toList()[0]
|
|
.parentNode
|
|
?.parentNode
|
|
?.children[1]
|
|
.innerHtml;
|
|
if (version == null) {
|
|
throw couldNotFindLatestVersion;
|
|
}
|
|
return APKDetails(version, [multipleVersionApkUrls[0]]);
|
|
} else {
|
|
throw couldNotFindReleases;
|
|
}
|
|
}
|
|
|
|
@override
|
|
AppNames getAppNames(String standardUrl) {
|
|
return AppNames('IzzyOnDroid', Uri.parse(standardUrl).pathSegments.last);
|
|
}
|
|
}
|
|
|
|
class SourceProvider {
|
|
List<AppSource> sources = [
|
|
GitHub(),
|
|
GitLab(),
|
|
FDroid(),
|
|
IzzyOnDroid(),
|
|
Mullvad(),
|
|
Signal()
|
|
];
|
|
|
|
List<MassAppSource> massSources = [GitHubStars()];
|
|
|
|
// Add more source classes here so they are available via the service
|
|
AppSource getSource(String url) {
|
|
AppSource? source;
|
|
for (var s in sources) {
|
|
if (url.toLowerCase().contains('://${s.host}')) {
|
|
source = s;
|
|
break;
|
|
}
|
|
}
|
|
if (source == null) {
|
|
throw 'URL does not match a known source';
|
|
}
|
|
return source;
|
|
}
|
|
|
|
Future<App> getApp(String url) async {
|
|
if (url.toLowerCase().indexOf('http://') != 0 &&
|
|
url.toLowerCase().indexOf('https://') != 0) {
|
|
url = 'https://$url';
|
|
}
|
|
if (url.toLowerCase().indexOf('https://www.') == 0) {
|
|
url = 'https://${url.substring(12)}';
|
|
}
|
|
AppSource source = getSource(url);
|
|
String standardUrl = source.standardizeURL(url);
|
|
AppNames names = source.getAppNames(standardUrl);
|
|
APKDetails apk = await source.getLatestAPKDetails(standardUrl);
|
|
return App(
|
|
'${names.author.toLowerCase()}_${names.name.toLowerCase()}_${source.host}',
|
|
standardUrl,
|
|
names.author[0].toUpperCase() + names.author.substring(1),
|
|
names.name[0].toUpperCase() + names.name.substring(1),
|
|
null,
|
|
apk.version,
|
|
apk.apkUrls,
|
|
apk.apkUrls.length - 1);
|
|
}
|
|
|
|
/// Returns a length 2 list, where the first element is a list of Apps and
|
|
/// the second is a Map<String, dynamic> of URLs and errors
|
|
Future<List<dynamic>> getApps(List<String> urls) async {
|
|
List<App> apps = [];
|
|
Map<String, dynamic> errors = {};
|
|
for (var url in urls) {
|
|
try {
|
|
apps.add(await getApp(url));
|
|
} catch (e) {
|
|
errors.addAll(<String, dynamic>{url: e});
|
|
}
|
|
}
|
|
return [apps, errors];
|
|
}
|
|
|
|
List<String> getSourceHosts() => sources.map((e) => e.host).toList();
|
|
}
|
|
|
|
abstract class MassAppSource {
|
|
late String name;
|
|
late List<String> requiredArgs;
|
|
Future<List<String>> getUrls(List<String> args);
|
|
}
|
|
|
|
class GitHubStars implements MassAppSource {
|
|
@override
|
|
late String name = 'GitHub Starred Repos';
|
|
|
|
@override
|
|
late List<String> requiredArgs = ['Username'];
|
|
|
|
@override
|
|
Future<List<String>> getUrls(List<String> args) async {
|
|
if (args.length != requiredArgs.length) {
|
|
throw 'Wrong number of arguments provided';
|
|
}
|
|
Response res =
|
|
await get(Uri.parse('https://api.github.com/users/${args[0]}/starred'));
|
|
if (res.statusCode == 200) {
|
|
return (jsonDecode(res.body) as List<dynamic>)
|
|
.map((e) => e['html_url'] as String)
|
|
.toList();
|
|
} else {
|
|
if (res.headers['x-ratelimit-remaining'] == '0') {
|
|
throw 'Rate limit reached - try again in ${(int.parse(res.headers['x-ratelimit-reset'] ?? '1800000000') / 60000000).toString()} minutes';
|
|
}
|
|
|
|
throw 'Unable to find user\'s starred repos';
|
|
}
|
|
}
|
|
}
|