/* * Copyright (c) 2018-2020, Andreas Kling * Copyright (c) 2021, networkException * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace Core { class ConfigFile : public RefCounted { public: enum class AllowWriting { Yes, No, }; static ErrorOr> open_for_lib(ByteString const& lib_name, AllowWriting = AllowWriting::No); static ErrorOr> open_for_app(ByteString const& app_name, AllowWriting = AllowWriting::No); static ErrorOr> open_for_system(ByteString const& app_name, AllowWriting = AllowWriting::No); static ErrorOr> open(ByteString const& filename, AllowWriting = AllowWriting::No); static ErrorOr> open(ByteString const& filename, int fd); static ErrorOr> open(ByteString const& filename, NonnullOwnPtr); ~ConfigFile(); bool has_group(ByteString const&) const; bool has_key(ByteString const& group, ByteString const& key) const; Vector groups() const; Vector keys(ByteString const& group) const; size_t num_groups() const { return m_groups.size(); } ByteString read_entry(ByteString const& group, ByteString const& key, ByteString const& default_value = {}) const { return read_entry_optional(group, key).value_or(default_value); } Optional read_entry_optional(ByteString const& group, ByteString const& key) const; bool read_bool_entry(ByteString const& group, ByteString const& key, bool default_value = false) const; template T read_num_entry(ByteString const& group, ByteString const& key, T default_value = 0) const { if (!has_key(group, key)) return default_value; return read_entry(group, key, "").to_number().value_or(default_value); } void write_entry(ByteString const& group, ByteString const& key, ByteString const& value); void write_bool_entry(ByteString const& group, ByteString const& key, bool value); template void write_num_entry(ByteString const& group, ByteString const& key, T value) { write_entry(group, key, ByteString::number(value)); } void dump() const; bool is_dirty() const { return m_dirty; } ErrorOr sync(); void add_group(ByteString const& group); void remove_group(ByteString const& group); void remove_entry(ByteString const& group, ByteString const& key); ByteString const& filename() const { return m_filename; } private: ConfigFile(ByteString const& filename, OwnPtr open_file); ErrorOr reparse(); ByteString m_filename; OwnPtr m_file; HashMap> m_groups; bool m_dirty { false }; }; }