This code is probably terrible but its ok

This commit is contained in:
Mars 2024-05-31 04:01:31 -04:00
parent 8db3dae6f1
commit e8fb8ec19f
Signed by: pupbrained
GPG key ID: 0FF5B8826803F895
10 changed files with 201 additions and 219 deletions

View file

@ -1,6 +1,4 @@
--- ---
Language: Cpp Language: Cpp
BasedOnStyle: Chromium BasedOnStyle: Chromium
IndentCaseBlocks: true SpaceBeforeCpp11BracedList: true
IndentCaseLabels: true

View file

@ -1,2 +1,3 @@
CompileFlags: CompileFlags:
Add: [-std=c++2b] Add: [-std=c++2b, -Wno-unused-function]
Remove: -Wc++98-compat

View file

@ -1,37 +0,0 @@
cmake_minimum_required(VERSION 3.28.1)
set(CMAKE_CXX_STANDARD 26)
set(CMAKE_EXPORT_COMPILE_COMMANDS 1)
set(CMAKE_CXX_SCAN_FOR_MODULES 1)
project(draconis++)
set(sources)
list(APPEND sources src/main.cpp)
if (${CMAKE_SYSTEM_NAME} MATCHES "Darwin")
set(MACOSX TRUE)
endif ()
if (MACOSX)
list(APPEND sources src/os/macos.cpp)
elseif (LINUX)
list(APPEND sources src/os/linux.cpp)
endif ()
add_executable(${PROJECT_NAME})
target_sources(${PROJECT_NAME} PUBLIC src/main.cpp)
target_sources(${PROJECT_NAME} PUBLIC FILE_SET all_my_modules TYPE CXX_MODULES FILES src/os/linux.cpp)
find_package(PkgConfig REQUIRED)
find_package(cpr REQUIRED)
find_package(Boost REQUIRED)
if (MACOSX)
pkg_check_modules(${PROJECT_NAME} REQUIRED IMPORTED_TARGET fmt tomlplusplus)
else ()
pkg_check_modules(${PROJECT_NAME} REQUIRED IMPORTED_TARGET fmt playerctl tomlplusplus)
endif ()
target_link_libraries(${PROJECT_NAME} PRIVATE PkgConfig::${PROJECT_NAME} cpr::cpr Boost::boost)

6
flake.lock generated
View file

@ -2,11 +2,11 @@
"nodes": { "nodes": {
"nixpkgs": { "nixpkgs": {
"locked": { "locked": {
"lastModified": 1716715802, "lastModified": 1717112898,
"narHash": "sha256-usk0vE7VlxPX8jOavrtpOqphdfqEQpf9lgedlY/r66c=", "narHash": "sha256-7R2ZvOnvd9h8fDd65p0JnB7wXfUvreox3xFdYWd1BnY=",
"owner": "NixOS", "owner": "NixOS",
"repo": "nixpkgs", "repo": "nixpkgs",
"rev": "e2dd4e18cc1c7314e24154331bae07df76eb582f", "rev": "6132b0f6e344ce2fe34fc051b72fb46e34f668e0",
"type": "github" "type": "github"
}, },
"original": { "original": {

112
flake.nix
View file

@ -16,6 +16,7 @@
system: let system: let
pkgs = import nixpkgs { pkgs = import nixpkgs {
inherit system; inherit system;
overlays = [ overlays = [
(self: super: { (self: super: {
ccacheWrapper = super.ccacheWrapper.override { ccacheWrapper = super.ccacheWrapper.override {
@ -45,111 +46,80 @@
]; ];
}; };
llvm = pkgs.llvmPackages_latest; stdenv = pkgs.llvmPackages_18.stdenv;
in
stdenv = pkgs.ccacheStdenv.override { with pkgs; {
stdenv = packages = rec {
if pkgs.hostPlatform.isDarwin draconis-cpp = stdenv.mkDerivation {
then llvm.libcxxStdenv
else pkgs.stdenvAdapters.useMoldLinker pkgs.clangStdenv;
};
darwinPkgs = nixpkgs.lib.optionals pkgs.stdenv.isDarwin (with pkgs.darwin; [
apple_sdk.frameworks.AppKit
apple_sdk.frameworks.Carbon
apple_sdk.frameworks.Cocoa
apple_sdk.frameworks.CoreFoundation
apple_sdk.frameworks.IOKit
apple_sdk.frameworks.WebKit
apple_sdk.frameworks.Security
apple_sdk.frameworks.DisplayServices
]);
in {
packages = rec {
draconis-cpp = with pkgs;
stdenv.mkDerivation {
name = "draconis++"; name = "draconis++";
version = "0.1.0";
src = self; src = self;
nativeBuildInputs = [ nativeBuildInputs = [
cmake meson
ninja ninja
pkg-config pkg-config
]; ];
propagatedBuildInputs = propagatedBuildInputs =
[ [
boost185
glib
]
++ (
if pkgs.hostPlatform.isLinux
then [playerctl]
else []
);
buildInputs =
[
fmt
libcpr libcpr
tomlplusplus tomlplusplus
] ]
++ darwinPkgs; ++ (lib.optionals pkgs.hostPlatform.isLinux [
glib
playerctl
]);
buildInputs = [
boost185
fmt
];
configurePhase = ''
meson setup build
'';
buildPhase = '' buildPhase = ''
cmake -GNinja . meson compile -C build
ninja
''; '';
installPhase = '' installPhase = ''
install -Dm755 ./draconis++ $out/bin/draconis++ mkdir -p $out/bin
mv build/draconis++ $out/bin/draconis++
''; '';
}; };
default = draconis-cpp; default = draconis-cpp;
}; };
devShell = with pkgs; formatter = alejandra;
mkShell.override {inherit stdenv;} {
packages = with pkgs; devShell = mkShell.override {inherit stdenv;} {
packages =
[ [
# builder alejandra
cmake bear
clang-tools
meson
ninja ninja
pkg-config pkg-config
# debugger
lldb
# fix headers not found
clang-tools_18
# LSP and compiler
llvm.libstdcxxClang
# other tools
cppcheck
llvm.libllvm
# stdlib for cpp
llvm.libcxx
# libraries
boost185 boost185
fmt fmt
glib glib
libcpr libcpr
tomlplusplus tomlplusplus
] ]
++ ( ++ (lib.optionals pkgs.hostPlatform.isLinux [playerctl]);
if stdenv.isDarwin
then [] buildInputs = [
else [playerctl] libcpr
) tomlplusplus
++ darwinPkgs; ];
name = "C++"; name = "C++";
}; };
} }
); );
} }

59
meson.build Normal file
View file

@ -0,0 +1,59 @@
project(
'draconis++', 'cpp',
default_options: [
'cpp_std=c++26',
'default_library=static',
'buildtype=debugoptimized',
'warning_level=everything',
]
)
cc = meson.get_compiler('cpp')
add_project_arguments(
cc.get_supported_arguments([
'-Wno-c++98-compat',
'-Wno-c++98-compat-pedantic',
'-Wno-padded',
'-Wno-pre-c++20-compat-pedantic',
'-Wno-switch-default',
'-Wno-missing-prototypes'
]),
language: 'cpp'
)
source_file_names = ['src/main.cpp', 'src/os/os.h']
if host_machine.system() == 'linux'
source_file_names += ['src/os/linux.cpp']
endif
if host_machine.system() == 'darwin'
source_file_names += ['src/os/macos.cpp']
endif
sources = []
foreach file : source_file_names
sources += files(file)
endforeach
boost_dep = dependency('boost')
cpr_dep = cc.find_library('cpr')
curl_dep = cc.find_library('curl')
fmt_dep = dependency('fmt')
tomlplusplus_dep = cc.find_library('tomlplusplus')
playerctl_dep = dependency('playerctl')
executable(
'draconis++',
sources,
dependencies: [
boost_dep,
cpr_dep,
curl_dep,
fmt_dep,
tomlplusplus_dep,
playerctl_dep
]
)

View file

@ -4,9 +4,9 @@
#include <fmt/core.h> #include <fmt/core.h>
#include <fmt/format.h> #include <fmt/format.h>
#include <boost/json/src.hpp> #include <boost/json/src.hpp>
#include <ctime>
#include <toml++/toml.hpp> #include <toml++/toml.hpp>
#include "os/os.h"
import OS;
using std::string; using std::string;
@ -24,35 +24,41 @@ struct LatLon {
}; };
struct Location { struct Location {
enum { coords, city } type; enum {
coords,
city,
} type;
union { union {
LatLon coords;
const char* city; const char* city;
LatLon coords;
} data; } data;
}; };
void use_location(const Location& l) {
switch (l.type) {
case Location::city:
{
char* c = const_cast<char*>(l.data.city);
puts(c);
free(c);
break;
}
case Location::coords:
{
const auto [lat, lon] = l.data.coords;
printf("%.f, %.f", lon, lat);
break;
}
}
}
struct Weather { struct Weather {
Location location; Location location;
string api_key; string api_key;
string units; string units;
Weather() = default;
Weather(const char* _city, string _api_key, string _units) {
this->api_key = _api_key;
this->units = _units;
this->location = Location {
Location::city,
{.city = _city},
};
}
Weather(LatLon _coords, string _api_key, string _units) {
this->api_key = _api_key;
this->units = _units;
this->location = Location {
Location::coords,
{.coords = _coords},
};
}
}; };
struct Config { struct Config {
@ -60,44 +66,28 @@ struct Config {
NowPlaying now_playing; NowPlaying now_playing;
Weather weather; Weather weather;
explicit Config(auto toml) { Config(toml::table toml) {
general = General{.name = toml["general"]["name"].value_or(getlogin())}; general = General {toml["general"]["name"].value_or(getlogin())};
now_playing = now_playing = NowPlaying {toml["now_playing"]["enable"].value_or(false)};
NowPlaying{.enable = toml["now_playing"]["enable"].value_or(false)};
const char* location = toml["weather"]["location"].value_or(""); const auto location = toml["weather"]["location"];
const string api_key = toml["weather"]["api_key"].value_or("");
const string units = toml["weather"]["units"].value_or("metric");
const size_t len = strlen(location); if (location.is_string())
char* loc = new char[len + 1]; weather = Weather(location.value_or(""), api_key, units);
strncpy(loc, location, len); else
loc[len] = '\0'; weather = Weather(
LatLon {location["lat"].value_or(0.0), location["lon"].value_or(0.0)},
if (toml["weather"]["location"].is_string()) { api_key, units);
weather = Weather{
.location = Location{.type = Location::city, .data = {.city = loc}},
.api_key = toml["weather"]["api_key"].value_or(""),
.units = toml["weather"]["units"].value_or("metric"),
};
} else {
weather = Weather{
.location =
Location{
.type = Location::coords,
.data = {.coords =
LatLon{
toml["weather"]["location"]["lat"].value_or(
0.0),
toml["weather"]["location"]["lon"].value_or(
0.0)}}},
.api_key = toml["weather"]["api_key"].value_or(""),
.units = toml["weather"]["units"].value_or("metric"),
};
}
} }
}; };
static Config CONFIG = Config(toml::parse_file("./config.toml")); static const Config& CONFIG() {
static const Config& CONFIG = *new Config(toml::parse_file("./config.toml"));
return CONFIG;
}
struct BytesToGiB { struct BytesToGiB {
uint64_t value; uint64_t value;
@ -108,7 +98,7 @@ struct fmt::formatter<BytesToGiB> : formatter<double> {
template <typename FormatContext> template <typename FormatContext>
auto format(const BytesToGiB b, FormatContext& ctx) { auto format(const BytesToGiB b, FormatContext& ctx) {
auto out = formatter<double>::format( auto out = formatter<double>::format(
b.value / pow(1024, 3), ctx); // NOLINT(*-narrowing-conversions); static_cast<double>(b.value) / pow(1024, 3), ctx);
*out++ = 'G'; *out++ = 'G';
*out++ = 'i'; *out++ = 'i';
*out++ = 'B'; *out++ = 'B';
@ -136,30 +126,30 @@ boost::json::object get_weather() {
using namespace boost::json; using namespace boost::json;
using namespace fmt; using namespace fmt;
if (CONFIG.weather.location.type == Location::city) { if (CONFIG().weather.location.type == Location::city) {
const char* location = const char* location = curl_easy_escape(
curl_easy_escape(nullptr, CONFIG.weather.location.data.city, nullptr, CONFIG().weather.location.data.city,
strlen(CONFIG.weather.location.data.city)); static_cast<int>(strlen(CONFIG().weather.location.data.city)));
const char* api_key = CONFIG.weather.api_key.c_str(); const char* api_key = CONFIG().weather.api_key.c_str();
const char* units = CONFIG.weather.units.c_str(); const char* units = CONFIG().weather.units.c_str();
const Response r = const Response r =
Get(Url{format("https://api.openweathermap.org/data/2.5/" Get(Url {format("https://api.openweathermap.org/data/2.5/"
"weather?q={}&appid={}&units={}", "weather?q={}&appid={}&units={}",
location, api_key, units)}); location, api_key, units)});
value json = parse(r.text); value json = parse(r.text);
return json.as_object(); return json.as_object();
} else { } else {
const auto [lat, lon] = CONFIG.weather.location.data.coords; const auto [lat, lon] = CONFIG().weather.location.data.coords;
const char* api_key = CONFIG.weather.api_key.c_str(); const char* api_key = CONFIG().weather.api_key.c_str();
const char* units = CONFIG.weather.units.c_str(); const char* units = CONFIG().weather.units.c_str();
const Response r = const Response r =
Get(Url{format("https://api.openweathermap.org/data/2.5/" Get(Url {format("https://api.openweathermap.org/data/2.5/"
"weather?lat={}&lon={}&appid={}&units={}", "weather?lat={}&lon={}&appid={}&units={}",
lat, lon, api_key, units)}); lat, lon, api_key, units)});
value json = parse(r.text); value json = parse(r.text);
@ -168,20 +158,28 @@ boost::json::object get_weather() {
} }
int main() { int main() {
const auto toml = toml::parse_file("./config.toml"); using boost::json::object;
using fmt::format;
using fmt::localtime;
using fmt::println;
using std::time;
using std::time_t;
using toml::parse_result;
if (CONFIG.now_playing.enable) const parse_result toml = toml::parse_file("./config.toml");
fmt::println("{}", get_nowplaying());
fmt::println("Hello {}!", CONFIG.general.name); if (CONFIG().now_playing.enable)
println("{}", get_nowplaying());
println("Hello {}!", CONFIG().general.name);
const uint64_t meminfo = get_meminfo(); const uint64_t meminfo = get_meminfo();
fmt::println("{:.2f}", BytesToGiB{meminfo}); println("{:.2f}", BytesToGiB {meminfo});
const std::time_t t = std::time(nullptr); const time_t t = time(nullptr);
string date = fmt::format("{:%d}", fmt::localtime(t)); string date = format("{:%d}", localtime(t));
switch (parse_date(date)) { switch (parse_date(date)) {
case Ones: case Ones:
@ -201,15 +199,14 @@ int main() {
break; break;
} }
fmt::println("{:%B} {}, {:%-I:%0M %p}", fmt::localtime(t), date, println("{:%B} {}, {:%-I:%0M %p}", localtime(t), date, localtime(t));
fmt::localtime(t));
boost::json::object json = get_weather(); object json = get_weather();
const char* town_name = const char* town_name =
json["name"].is_string() ? json["name"].as_string().c_str() : "Unknown"; json["name"].is_string() ? json["name"].as_string().c_str() : "Unknown";
fmt::println("{}", town_name); println("{}", town_name);
return 0; return 0;
} }

View file

@ -1,11 +1,7 @@
module;
#include <fmt/core.h> #include <fmt/core.h>
#include <playerctl/playerctl.h> #include <playerctl/playerctl.h>
#include <fstream> #include <fstream>
export module OS;
using std::string; using std::string;
uint64_t parse_line_as_number(const string& input) { uint64_t parse_line_as_number(const string& input) {
@ -13,15 +9,13 @@ uint64_t parse_line_as_number(const string& input) {
string::size_type start = 0; string::size_type start = 0;
// Skip leading non-numbers // Skip leading non-numbers
while (!isdigit(input[++start])) while (!isdigit(input[++start]));
;
// Start searching from the start of the number // Start searching from the start of the number
string::size_type end = start; string::size_type end = start;
// Increment to the end of the number // Increment to the end of the number
while (isdigit(input[++end])) while (isdigit(input[++end]));
;
// Return the substring containing the number // Return the substring containing the number
return std::stoul(input.substr(start, end - start)); return std::stoul(input.substr(start, end - start));
@ -31,16 +25,15 @@ uint64_t meminfo_parse(std::ifstream is) {
string line; string line;
// Skip every line before the one that starts with "MemTotal" // Skip every line before the one that starts with "MemTotal"
while (std::getline(is, line) && !line.starts_with("MemTotal")) while (std::getline(is, line) && !line.starts_with("MemTotal"));
;
// Parse the number from the line // Parse the number from the line
const auto num = parse_line_as_number(line); const uint64_t num = parse_line_as_number(line);
return num; return num;
} }
export uint64_t get_meminfo() { uint64_t get_meminfo() {
return meminfo_parse(std::ifstream("/proc/meminfo")) * 1024; return meminfo_parse(std::ifstream("/proc/meminfo")) * 1024;
} }
@ -60,7 +53,7 @@ PlayerctlPlayer* init_playerctl() {
return nullptr; return nullptr;
// Get the first player // Get the first player
const auto player_name = PlayerctlPlayerName* player_name =
static_cast<PlayerctlPlayerName*>(available_players->data); static_cast<PlayerctlPlayerName*>(available_players->data);
// Create the player // Create the player
@ -80,10 +73,8 @@ PlayerctlPlayer* init_playerctl() {
return current_player; return current_player;
} }
export string get_nowplaying() { string get_nowplaying() {
if (PlayerctlPlayer* current_player = init_playerctl()) { if (PlayerctlPlayer* current_player = init_playerctl())
return playerctl_player_get_title(current_player, nullptr); return playerctl_player_get_title(current_player, nullptr);
}
return "Could not get now playing info"; return "Could not get now playing info";
} }

View file

@ -1,10 +1,8 @@
module;
#include <sys/sysctl.h> #include <sys/sysctl.h>
#include <unistd.h>
#include <string>
export module OS; uint64_t get_meminfo() {
export uint64_t get_meminfo() {
uint64_t mem = 0; uint64_t mem = 0;
size_t size = sizeof(mem); size_t size = sizeof(mem);
@ -13,6 +11,6 @@ export uint64_t get_meminfo() {
return mem; return mem;
} }
export string get_nowplaying() { static std::string get_nowplaying() {
return ""; return "";
} }

5
src/os/os.h Normal file
View file

@ -0,0 +1,5 @@
#include <cstdint>
#include <string>
uint64_t get_meminfo();
std::string get_nowplaying();