Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a "status" API to DiagnosticContext, overhaul Downloads console output #1565

Merged
merged 28 commits into from
Jan 15, 2025
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
df8d332
Add an API to MessageSink which can accept entire lines including col…
BillyONeal Dec 19, 2024
face29d
Add status reporting API to DiagnosticContext.
BillyONeal Dec 19, 2024
0215853
Get rid of the "manager" antipattern for downloads.
BillyONeal Dec 19, 2024
bc3b87c
DiagnosticContext-ize system.process.
BillyONeal Dec 19, 2024
7a2f55c
Give tool hashing a bespoke message that better describes the situati…
BillyONeal Dec 20, 2024
8b409a3
Extract adding curl headers to ensure that the vcpkg user agent heade…
BillyONeal Jan 4, 2025
20a03bd
Add MessageLine::to_string.
BillyONeal Jan 4, 2025
d46bf00
Apply DiagnosticContext to WinHTTP parts, and make the handles non-as…
BillyONeal Jan 4, 2025
2a8adf5
DiagnosticContext-ize file hashing.
BillyONeal Jan 6, 2025
8090744
Overhaul and DiagnosticContext-ize downloads.
BillyONeal Jan 8, 2025
3beeb27
Fix line endings on new file to be LFs.
BillyONeal Jan 8, 2025
f3d2dd9
Fix macOS typo.
BillyONeal Jan 9, 2025
9ccc1b7
Fix ASAN failure.
BillyONeal Jan 9, 2025
2b1b07f
Url -> SanitizedUrl
BillyONeal Jan 9, 2025
659f955
Audit function parameter names, separate names for cached and uncache…
BillyONeal Jan 9, 2025
1d9cf47
upload_asset_cache_file + put_file_to_mirror => store_to_asset_cache
BillyONeal Jan 9, 2025
4313d69
Merge remote-tracking branch 'origin/main' into message-sink-line
BillyONeal Jan 9, 2025
dc1e485
Fix curl output matching on Ubuntu 22.04.
BillyONeal Jan 9, 2025
335f06c
Ensure pwsh ends with a trailing newline.
BillyONeal Jan 9, 2025
f7be0c5
Support curl output on Ubuntu 20.04.
BillyONeal Jan 9, 2025
c9b43e3
CR nitpicks.
BillyONeal Jan 9, 2025
0fb50e8
Actually save all the files :(
BillyONeal Jan 9, 2025
9b9a1eb
Add some shinier error message handling.
BillyONeal Jan 14, 2025
20dd331
Add azurl SHA mismatch test case.
BillyONeal Jan 14, 2025
c6d3282
Add upstream URL to asset sha mismatches.
BillyONeal Jan 14, 2025
279c0d4
Add ubuntu curl timeout form.
BillyONeal Jan 14, 2025
62b5ff2
Treat SHA mismatch as an ordinary download failure rather than a secu…
BillyONeal Jan 14, 2025
b966872
Add design goals comment.
BillyONeal Jan 15, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions azure-pipelines/e2e-assets/asset-caching/bad-hash-script.ps1
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
Param([string]$File)
Write-Host "Creating file with the wrong hash"
Set-Content -Path $File -Value "This is a file with the wrong hash" -Encoding Ascii -NoNewline
3 changes: 2 additions & 1 deletion azure-pipelines/e2e-assets/asset-caching/failing-script.ps1
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
throw "Script download error"
Write-Host "Script download error"
exit 1
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Write-Host "Not creating a file"
292 changes: 198 additions & 94 deletions azure-pipelines/end-to-end-tests-dir/asset-caching.ps1

Large diffs are not rendered by default.

18 changes: 10 additions & 8 deletions include/vcpkg/base/api-stable-format.h
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#pragma once

#include <vcpkg/base/expected.h>
#include <vcpkg/base/diagnostics.h>
#include <vcpkg/base/optional.h>
#include <vcpkg/base/stringview.h>

#include <string>
Expand All @@ -10,21 +11,22 @@ namespace vcpkg
namespace details
{
template<class F>
void api_stable_format_cb(void* f, std::string& s, StringView sv)
bool api_stable_format_cb(void* f, std::string& s, StringView sv)
{
(*(F*)(f))(s, sv);
return (*(F*)(f))(s, sv);
}

ExpectedL<std::string> api_stable_format_impl(StringView fmtstr,
void (*cb)(void*, std::string&, StringView),
void* data);
Optional<std::string> api_stable_format_impl(DiagnosticContext& context,
StringView fmtstr,
bool (*cb)(void*, std::string&, StringView),
void* data);
}

// This function exists in order to provide an API-stable formatting function similar to `std::format()` that does
// not depend on the feature set of fmt or the C++ standard library and thus can be contractual for user interfaces.
template<class F>
ExpectedL<std::string> api_stable_format(StringView fmtstr, F&& handler)
Optional<std::string> api_stable_format(DiagnosticContext& context, StringView fmtstr, F&& handler)
{
return details::api_stable_format_impl(fmtstr, &details::api_stable_format_cb<F>, &handler);
return details::api_stable_format_impl(context, fmtstr, &details::api_stable_format_cb<F>, &handler);
}
}
136 changes: 133 additions & 3 deletions include/vcpkg/base/diagnostics.h
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#pragma once

#include <vcpkg/base/expected.h>
#include <vcpkg/base/message_sinks.h>
#include <vcpkg/base/messages.h>
#include <vcpkg/base/optional.h>

Expand Down Expand Up @@ -66,11 +67,22 @@ namespace vcpkg
std::string to_string() const;
void to_string(std::string& target) const;

MessageLine to_message_line() const;

LocalizedString to_json_reader_string(const std::string& path, const LocalizedString& type) const;

DiagKind kind() const noexcept { return m_kind; }
// Returns this DiagnosticLine with kind == Error reduced to Warning.
DiagnosticLine reduce_to_warning() const&;
DiagnosticLine reduce_to_warning() &&;

private:
DiagnosticLine(DiagKind kind,
const Optional<std::string>& origin,
TextRowCol position,
const LocalizedString& message);
DiagnosticLine(DiagKind kind, Optional<std::string>&& origin, TextRowCol position, LocalizedString&& message);

DiagKind m_kind;
Optional<std::string> m_origin;
TextRowCol m_position;
Expand All @@ -79,8 +91,11 @@ namespace vcpkg

struct DiagnosticContext
{
// The `report` family are used to report errors or warnings that may result in a function failing
// to do what it is intended to do. Data sent to the `report` family is expected to not be printed
// to the console if a caller decides to handle an error.
virtual void report(const DiagnosticLine& line) = 0;
virtual void report(DiagnosticLine&& line) { report(line); }
virtual void report(DiagnosticLine&& line);

void report_error(const LocalizedString& message) { report(DiagnosticLine{DiagKind::Error, message}); }
void report_error(LocalizedString&& message) { report(DiagnosticLine{DiagKind::Error, std::move(message)}); }
Expand All @@ -92,15 +107,64 @@ namespace vcpkg
this->report_error(std::move(message));
}

template<VCPKG_DECL_MSG_TEMPLATE>
void report_error_with_log(StringView log_content, VCPKG_DECL_MSG_ARGS)
{
LocalizedString message;
msg::format_to(message, VCPKG_EXPAND_MSG_ARGS);
message.append_raw('\n');
message.append_raw(log_content);
this->report_error(std::move(message));
}

void report_system_error(StringLiteral system_api_name, int error_value);

// The `status` family are used to report status or progress information that callers are expected
// to show on the console, even if it would decide to handle errors or warnings itself.
// Examples:
// * "Downloading file..."
// * "Building package 1 of 47..."
//
// Some implementations of DiagnosticContext may buffer these messages *anyway* if that makes sense,
// for example, if the work is happening on a background thread.
virtual void statusln(const LocalizedString& message) = 0;
virtual void statusln(LocalizedString&& message) = 0;
virtual void statusln(const MessageLine& message) = 0;
virtual void statusln(MessageLine&& message) = 0;

protected:
~DiagnosticContext() = default;
};

struct PrintingDiagnosticContext final : DiagnosticContext
{
PrintingDiagnosticContext(MessageSink& sink) : sink(sink) { }

virtual void report(const DiagnosticLine& line) override;

virtual void statusln(const LocalizedString& message) override;
virtual void statusln(LocalizedString&& message) override;
virtual void statusln(const MessageLine& message) override;
virtual void statusln(MessageLine&& message) override;

private:
MessageSink& sink;
};

// Stores all diagnostics into a vector, while passing through status lines to an underlying MessageSink.
struct BufferedDiagnosticContext final : DiagnosticContext
{
BufferedDiagnosticContext(MessageSink& status_sink) : status_sink(status_sink) { }

virtual void report(const DiagnosticLine& line) override;
virtual void report(DiagnosticLine&& line) override;

virtual void statusln(const LocalizedString& message) override;
virtual void statusln(LocalizedString&& message) override;
virtual void statusln(const MessageLine& message) override;
virtual void statusln(MessageLine&& message) override;

MessageSink& status_sink;
std::vector<DiagnosticLine> lines;

// Prints all diagnostics to the supplied sink.
Expand All @@ -111,9 +175,75 @@ namespace vcpkg
void to_string(std::string& target) const;

bool any_errors() const noexcept;
bool empty() const noexcept;
};

// Stores all diagnostics and status messages into a vector. This is generally used for background thread or similar
// scenarios where even status messages can't be immediately printed.
struct FullyBufferedDiagnosticContext final : DiagnosticContext
{
virtual void report(const DiagnosticLine& line) override;
virtual void report(DiagnosticLine&& line) override;

virtual void statusln(const LocalizedString& message) override;
virtual void statusln(LocalizedString&& message) override;
virtual void statusln(const MessageLine& message) override;
virtual void statusln(MessageLine&& message) override;

std::vector<MessageLine> lines;

// Prints all diagnostics to the supplied sink.
void print_to(MessageSink& sink) const;
// Converts this message into a string
// Prefer print() if possible because it applies color
std::string to_string() const;
void to_string(std::string& target) const;

bool empty() const noexcept;
};

// DiagnosticContext for attempted operations that may be recovered.
// Stores all diagnostics and passes through all status messages. Afterwards, call commit() to report all
// diagnostics to the outer DiagnosticContext, or handle() to forget them.
struct AttemptDiagnosticContext final : DiagnosticContext
{
AttemptDiagnosticContext(DiagnosticContext& inner_context) : inner_context(inner_context) { }

virtual void report(const DiagnosticLine& line) override;
virtual void report(DiagnosticLine&& line) override;

virtual void statusln(const LocalizedString& message) override;
virtual void statusln(LocalizedString&& message) override;
virtual void statusln(const MessageLine& message) override;
virtual void statusln(MessageLine&& message) override;

void commit();
void handle();

~AttemptDiagnosticContext();

DiagnosticContext& inner_context;
std::vector<DiagnosticLine> lines;
};

// Wraps another DiagnosticContext and reduces the severity of any reported diagnostics to warning from error.
struct WarningDiagnosticContext final : DiagnosticContext
{
WarningDiagnosticContext(DiagnosticContext& inner_context) : inner_context(inner_context) { }

virtual void report(const DiagnosticLine& line) override;
virtual void report(DiagnosticLine&& line) override;

virtual void statusln(const LocalizedString& message) override;
virtual void statusln(LocalizedString&& message) override;
virtual void statusln(const MessageLine& message) override;
virtual void statusln(MessageLine&& message) override;

DiagnosticContext& inner_context;
};

extern DiagnosticContext& console_diagnostic_context;
extern DiagnosticContext& status_only_diagnostic_context;
extern DiagnosticContext& null_diagnostic_context;

// The following overloads are implementing
Expand Down Expand Up @@ -191,7 +321,7 @@ namespace vcpkg
{
using Unwrapper = AdaptContextUnwrapOptional<std::invoke_result_t<Fn, BufferedDiagnosticContext&, Args...>>;
using ReturnType = ExpectedL<typename Unwrapper::type>;
BufferedDiagnosticContext bdc;
BufferedDiagnosticContext bdc{out_sink};
decltype(auto) maybe_result = functor(bdc, std::forward<Args>(args)...);
if (auto result = maybe_result.get())
{
Expand Down Expand Up @@ -261,7 +391,7 @@ namespace vcpkg
{
using ReturnType = ExpectedL<
typename AdaptContextDetectUniquePtr<std::invoke_result_t<Fn, BufferedDiagnosticContext&, Args...>>::type>;
BufferedDiagnosticContext bdc;
BufferedDiagnosticContext bdc{out_sink};
decltype(auto) maybe_result = functor(bdc, std::forward<Args>(args)...);
if (maybe_result)
{
Expand Down
Loading
Loading