1
1
Fork 0
mirror of https://github.com/NixOS/nix.git synced 2025-11-20 01:09:37 +01:00

Merge pull request #14426 from obsidiansystems/json-schema-build-result

JSON Impl and schema for BuildResult
This commit is contained in:
John Ericson 2025-11-06 18:40:35 +00:00 committed by GitHub
commit 147e183c68
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
16 changed files with 467 additions and 0 deletions

View file

@ -1,4 +1,6 @@
#include "nix/store/build-result.hh"
#include "nix/util/json-utils.hh"
#include <array>
namespace nix {
@ -11,4 +13,144 @@ std::strong_ordering BuildResult::Success::operator<=>(const BuildResult::Succes
bool BuildResult::Failure::operator==(const BuildResult::Failure &) const noexcept = default;
std::strong_ordering BuildResult::Failure::operator<=>(const BuildResult::Failure &) const noexcept = default;
static constexpr std::array<std::pair<BuildResult::Success::Status, std::string_view>, 4> successStatusStrings{{
#define ENUM_ENTRY(e) {BuildResult::Success::e, #e}
ENUM_ENTRY(Built),
ENUM_ENTRY(Substituted),
ENUM_ENTRY(AlreadyValid),
ENUM_ENTRY(ResolvesToAlreadyValid),
#undef ENUM_ENTRY
}};
static std::string_view successStatusToString(BuildResult::Success::Status status)
{
for (const auto & [enumVal, str] : successStatusStrings) {
if (enumVal == status)
return str;
}
throw Error("unknown success status: %d", static_cast<int>(status));
}
static BuildResult::Success::Status successStatusFromString(std::string_view str)
{
for (const auto & [enumVal, enumStr] : successStatusStrings) {
if (enumStr == str)
return enumVal;
}
throw Error("unknown built result success status '%s'", str);
}
static constexpr std::array<std::pair<BuildResult::Failure::Status, std::string_view>, 12> failureStatusStrings{{
#define ENUM_ENTRY(e) {BuildResult::Failure::e, #e}
ENUM_ENTRY(PermanentFailure),
ENUM_ENTRY(InputRejected),
ENUM_ENTRY(OutputRejected),
ENUM_ENTRY(TransientFailure),
ENUM_ENTRY(CachedFailure),
ENUM_ENTRY(TimedOut),
ENUM_ENTRY(MiscFailure),
ENUM_ENTRY(DependencyFailed),
ENUM_ENTRY(LogLimitExceeded),
ENUM_ENTRY(NotDeterministic),
ENUM_ENTRY(NoSubstituters),
ENUM_ENTRY(HashMismatch),
#undef ENUM_ENTRY
}};
static std::string_view failureStatusToString(BuildResult::Failure::Status status)
{
for (const auto & [enumVal, str] : failureStatusStrings) {
if (enumVal == status)
return str;
}
throw Error("unknown failure status: %d", static_cast<int>(status));
}
static BuildResult::Failure::Status failureStatusFromString(std::string_view str)
{
for (const auto & [enumVal, enumStr] : failureStatusStrings) {
if (enumStr == str)
return enumVal;
}
throw Error("unknown built result failure status '%s'", str);
}
} // namespace nix
namespace nlohmann {
using namespace nix;
void adl_serializer<BuildResult>::to_json(json & res, const BuildResult & br)
{
res = json::object();
// Common fields
res["timesBuilt"] = br.timesBuilt;
res["startTime"] = br.startTime;
res["stopTime"] = br.stopTime;
if (br.cpuUser.has_value()) {
res["cpuUser"] = br.cpuUser->count();
}
if (br.cpuSystem.has_value()) {
res["cpuSystem"] = br.cpuSystem->count();
}
// Handle success or failure variant
std::visit(
overloaded{
[&](const BuildResult::Success & success) {
res["success"] = true;
res["status"] = successStatusToString(success.status);
res["builtOutputs"] = success.builtOutputs;
},
[&](const BuildResult::Failure & failure) {
res["success"] = false;
res["status"] = failureStatusToString(failure.status);
res["errorMsg"] = failure.errorMsg;
res["isNonDeterministic"] = failure.isNonDeterministic;
},
},
br.inner);
}
BuildResult adl_serializer<BuildResult>::from_json(const json & _json)
{
auto & json = getObject(_json);
BuildResult br;
// Common fields
br.timesBuilt = getUnsigned(valueAt(json, "timesBuilt"));
br.startTime = getUnsigned(valueAt(json, "startTime"));
br.stopTime = getUnsigned(valueAt(json, "stopTime"));
if (auto cpuUser = optionalValueAt(json, "cpuUser")) {
br.cpuUser = std::chrono::microseconds(getUnsigned(*cpuUser));
}
if (auto cpuSystem = optionalValueAt(json, "cpuSystem")) {
br.cpuSystem = std::chrono::microseconds(getUnsigned(*cpuSystem));
}
// Determine success or failure based on success field
bool success = getBoolean(valueAt(json, "success"));
std::string statusStr = getString(valueAt(json, "status"));
if (success) {
BuildResult::Success s;
s.status = successStatusFromString(statusStr);
s.builtOutputs = valueAt(json, "builtOutputs");
br.inner = std::move(s);
} else {
BuildResult::Failure f;
f.status = failureStatusFromString(statusStr);
f.errorMsg = getString(valueAt(json, "errorMsg"));
f.isNonDeterministic = getBoolean(valueAt(json, "isNonDeterministic"));
br.inner = std::move(f);
}
return br;
}
} // namespace nlohmann

View file

@ -7,6 +7,7 @@
#include "nix/store/derived-path.hh"
#include "nix/store/realisation.hh"
#include "nix/util/json-impls.hh"
namespace nix {
@ -175,3 +176,5 @@ struct KeyedBuildResult : BuildResult
};
} // namespace nix
JSON_IMPL(nix::BuildResult)