2022-07-11 10:14:53 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2022, Andrew Kaster <akaster@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
#include <AK/DeprecatedString.h>
|
2022-07-11 10:14:53 +03:00
|
|
|
#include <AK/LexicalPath.h>
|
|
|
|
#include <AK/Platform.h>
|
|
|
|
#include <AK/ScopeGuard.h>
|
|
|
|
#include <LibArchive/Tar.h>
|
|
|
|
#include <LibArchive/TarStream.h>
|
2023-07-19 19:27:16 +03:00
|
|
|
#include <LibCompress/Gzip.h>
|
2022-07-11 10:14:53 +03:00
|
|
|
#include <LibCore/Directory.h>
|
|
|
|
#include <LibCore/System.h>
|
2023-07-19 19:27:16 +03:00
|
|
|
#include <LibFileSystem/FileSystem.h>
|
2022-07-11 10:14:53 +03:00
|
|
|
#include <LibMain/Main.h>
|
|
|
|
|
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QJniObject>
|
|
|
|
#include <QSslSocket>
|
|
|
|
|
|
|
|
#ifndef AK_OS_ANDROID
|
|
|
|
# error This file is for Android only, check CMake config!
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// HACK ALERT, we need to include LibMain manually here because the Qt build system doesn't include LibMain.a in the actual executable,
|
|
|
|
// nor include it in libladybird_<arch>.so
|
|
|
|
#include <LibMain/Main.cpp> // NOLINT(bugprone-suspicious-include)
|
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
extern DeprecatedString s_serenity_resource_root;
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
void android_platform_init();
|
|
|
|
static void extract_ladybird_resources();
|
2023-07-19 19:27:16 +03:00
|
|
|
static ErrorOr<void> extract_tar_archive(String archive_file, DeprecatedString output_directory);
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
void android_platform_init()
|
|
|
|
{
|
|
|
|
qDebug() << "Device supports OpenSSL: " << QSslSocket::supportsSsl();
|
|
|
|
|
|
|
|
QJniObject res = QJniObject::callStaticMethod<jstring>("org/serenityos/ladybird/TransferAssets",
|
|
|
|
"transferAssets",
|
|
|
|
"(Landroid/content/Context;)Ljava/lang/String;",
|
|
|
|
QNativeInterface::QAndroidApplication::context());
|
|
|
|
s_serenity_resource_root = res.toString().toUtf8().data();
|
|
|
|
|
|
|
|
extract_ladybird_resources();
|
|
|
|
}
|
|
|
|
|
|
|
|
void extract_ladybird_resources()
|
|
|
|
{
|
|
|
|
qDebug() << "serenity resource root is " << s_serenity_resource_root.characters();
|
2022-12-04 21:43:54 +03:00
|
|
|
auto file_or_error = Core::System::open(DeprecatedString::formatted("{}/res/icons/16x16/app-browser.png", s_serenity_resource_root), O_RDONLY);
|
2022-07-11 10:14:53 +03:00
|
|
|
if (file_or_error.is_error()) {
|
|
|
|
qDebug() << "Unable to open test file file as expected, extracting asssets...";
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
MUST(extract_tar_archive(MUST(String::formatted("{}/ladybird-assets.tar", s_serenity_resource_root)), s_serenity_resource_root));
|
2022-07-11 10:14:53 +03:00
|
|
|
} else {
|
|
|
|
qDebug() << "Opened app-browser.png test file, good to go!";
|
|
|
|
qDebug() << "Hopefully no developer changed the asset files and expected them to be re-extracted!";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
ErrorOr<void> extract_tar_archive(String archive_file, DeprecatedString output_directory)
|
2022-07-11 10:14:53 +03:00
|
|
|
{
|
|
|
|
constexpr size_t buffer_size = 4096;
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
auto file = TRY(Core::InputBufferedFile::create(TRY(Core::File::open(archive_file, Core::File::OpenMode::Read))));
|
2022-07-11 10:14:53 +03:00
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
DeprecatedString old_pwd = TRY(Core::System::getcwd());
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
TRY(Core::System::chdir(output_directory));
|
|
|
|
ScopeGuard go_back = [&old_pwd] { MUST(Core::System::chdir(old_pwd)); };
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
auto tar_stream = TRY(Archive::TarInputStream::construct(make<Compress::GzipCompressor>(move(file))));
|
2022-07-11 10:14:53 +03:00
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
HashMap<DeprecatedString, DeprecatedString> global_overrides;
|
|
|
|
HashMap<DeprecatedString, DeprecatedString> local_overrides;
|
2022-07-11 10:14:53 +03:00
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
auto get_override = [&](StringView key) -> Optional<DeprecatedString> {
|
|
|
|
Optional<DeprecatedString> maybe_local = local_overrides.get(key);
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
if (maybe_local.has_value())
|
|
|
|
return maybe_local;
|
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
Optional<DeprecatedString> maybe_global = global_overrides.get(key);
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
if (maybe_global.has_value())
|
|
|
|
return maybe_global;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
};
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
while (!tar_stream->finished()) {
|
|
|
|
Archive::TarFileHeader const& header = tar_stream->header();
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
// Handle meta-entries earlier to avoid consuming the file content stream.
|
|
|
|
if (header.content_is_like_extended_header()) {
|
|
|
|
switch (header.type_flag()) {
|
|
|
|
case Archive::TarFileType::GlobalExtendedHeader: {
|
2023-07-19 19:27:16 +03:00
|
|
|
TRY(tar_stream->for_each_extended_header([&](StringView key, StringView value) {
|
2022-07-11 10:14:53 +03:00
|
|
|
if (value.length() == 0)
|
|
|
|
global_overrides.remove(key);
|
|
|
|
else
|
|
|
|
global_overrides.set(key, value);
|
|
|
|
}));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Archive::TarFileType::ExtendedHeader: {
|
2023-07-19 19:27:16 +03:00
|
|
|
TRY(tar_stream->for_each_extended_header([&](StringView key, StringView value) {
|
2022-07-11 10:14:53 +03:00
|
|
|
local_overrides.set(key, value);
|
|
|
|
}));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
warnln("Unknown extended header type '{}' of {}", (char)header.type_flag(), header.filename());
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
TRY(tar_stream->advance());
|
2022-07-11 10:14:53 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
Archive::TarFileStream file_stream = tar_stream->file_contents();
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
// Handle other header types that don't just have an effect on extraction.
|
|
|
|
switch (header.type_flag()) {
|
|
|
|
case Archive::TarFileType::LongName: {
|
|
|
|
StringBuilder long_name;
|
|
|
|
|
|
|
|
Array<u8, buffer_size> buffer;
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
while (!file_stream.is_eof()) {
|
|
|
|
auto slice = TRY(file_stream.read_some(buffer));
|
|
|
|
long_name.append(reinterpret_cast<char*>(slice.data()), slice.size());
|
|
|
|
}
|
2022-07-11 10:14:53 +03:00
|
|
|
|
2022-12-04 21:43:54 +03:00
|
|
|
local_overrides.set("path", long_name.to_deprecated_string());
|
2023-07-19 19:27:16 +03:00
|
|
|
TRY(tar_stream->advance());
|
2022-07-11 10:14:53 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// None of the relevant headers, so continue as normal.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
LexicalPath path = LexicalPath(header.filename());
|
|
|
|
if (!header.prefix().is_empty())
|
|
|
|
path = path.prepend(header.prefix());
|
2022-12-04 21:43:54 +03:00
|
|
|
DeprecatedString filename = get_override("path"sv).value_or(path.string());
|
2022-07-11 10:14:53 +03:00
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
DeprecatedString absolute_path = TRY(FileSystem::absolute_path(filename)).to_deprecated_string();
|
2022-07-11 10:14:53 +03:00
|
|
|
auto parent_path = LexicalPath(absolute_path).parent();
|
2023-07-19 19:27:16 +03:00
|
|
|
auto header_mode = TRY(header.mode());
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
switch (header.type_flag()) {
|
|
|
|
case Archive::TarFileType::NormalFile:
|
|
|
|
case Archive::TarFileType::AlternateNormalFile: {
|
|
|
|
MUST(Core::Directory::create(parent_path, Core::Directory::CreateDirectories::Yes));
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
int fd = TRY(Core::System::open(absolute_path, O_CREAT | O_WRONLY, header_mode));
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
Array<u8, buffer_size> buffer;
|
2023-07-19 19:27:16 +03:00
|
|
|
while (!file_stream.is_eof()) {
|
|
|
|
auto slice = TRY(file_stream.read_some(buffer));
|
|
|
|
TRY(Core::System::write(fd, slice));
|
|
|
|
}
|
2022-07-11 10:14:53 +03:00
|
|
|
|
|
|
|
TRY(Core::System::close(fd));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Archive::TarFileType::SymLink: {
|
|
|
|
MUST(Core::Directory::create(parent_path, Core::Directory::CreateDirectories::Yes));
|
|
|
|
|
|
|
|
TRY(Core::System::symlink(header.link_name(), absolute_path));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Archive::TarFileType::Directory: {
|
|
|
|
MUST(Core::Directory::create(parent_path, Core::Directory::CreateDirectories::Yes));
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
auto result_or_error = Core::System::mkdir(absolute_path, header_mode);
|
2022-07-11 10:14:53 +03:00
|
|
|
if (result_or_error.is_error() && result_or_error.error().code() != EEXIST)
|
2023-07-19 19:27:16 +03:00
|
|
|
return result_or_error.release_error();
|
2022-07-11 10:14:53 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// FIXME: Implement other file types
|
|
|
|
warnln("file type '{}' of {} is not yet supported", (char)header.type_flag(), header.filename());
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Non-global headers should be cleared after every file.
|
|
|
|
local_overrides.clear();
|
|
|
|
|
2023-07-19 19:27:16 +03:00
|
|
|
TRY(tar_stream->advance());
|
|
|
|
}
|
2022-07-11 10:14:53 +03:00
|
|
|
return {};
|
|
|
|
}
|