File: LogResult.cpp

package info (click to toggle)
nextcloud-desktop 4.0.1-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 40,404 kB
  • sloc: cpp: 118,401; objc: 752; python: 606; sh: 395; ansic: 391; ruby: 174; makefile: 44; javascript: 32; xml: 6
file content (124 lines) | stat: -rw-r--r-- 3,951 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/*
 * SPDX-FileCopyrightText: 2020 Nextcloud GmbH and Nextcloud contributors
 * SPDX-License-Identifier: GPL-2.0-or-later OR CPOL-1.02
 * 
 * Parts of this file are based on:
 * https://www.codeproject.com/articles/570751/devmsi-an-example-cplusplus-msi-wix-deferred-custo
 * 
 * Licensed under the The Code Project Open License (CPOL):
 * https://www.codeproject.com/info/cpol10.aspx
 */

#include "NCMsiHelper.h"

//
// This code modified from MSDN article 256348
// "How to obtain error message descriptions using the FormatMessage API"
// Currently found at http://support.microsoft.com/kb/256348/en-us

#define ERRMSGBUFFERSIZE 256

/**
 * Use FormatMessage() to look an error code and log the error text.
 *
 * @param dwErrorMsgId The error code to be investigated.
 */
void LogError(DWORD dwErrorMsgId)
{
    HLOCAL pBuffer = nullptr;   // Buffer to hold the textual error description.
    DWORD ret = 0;              // Temp space to hold a return value.
    HINSTANCE hInst = nullptr;  // Instance handle for DLL.
    bool doLookup = true;
    DWORD dwMessageId = dwErrorMsgId;
    LPCSTR pMessage = "Error %d";
    DWORD flags = FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS;

    if (HRESULT_FACILITY(dwErrorMsgId) == FACILITY_MSMQ) {
        hInst = LoadLibrary(TEXT("MQUTIL.DLL"));
        flags |= FORMAT_MESSAGE_FROM_HMODULE;
        doLookup = (nullptr != hInst);
    } else if (dwErrorMsgId >= NERR_BASE && dwErrorMsgId <= MAX_NERR) {
        hInst = LoadLibrary(TEXT("NETMSG.DLL"));
        flags |= FORMAT_MESSAGE_FROM_HMODULE;
        doLookup = (nullptr != hInst);
    } else if (HRESULT_FACILITY(dwErrorMsgId) == FACILITY_WIN32) {
        // A "GetLastError" error, drop the HRESULT_FACILITY
        dwMessageId &= 0x0000FFFF;
        flags |= FORMAT_MESSAGE_FROM_SYSTEM;
    }

    if (doLookup) {
        ret = FormatMessageA( 
            flags,
            hInst, // Handle to the DLL.
            dwMessageId, // Message identifier.
            MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language.
            reinterpret_cast<LPSTR>(&pBuffer), // Buffer that will hold the text string.
            ERRMSGBUFFERSIZE, // Allocate at least this many chars for pBuffer.
            nullptr // No insert values.
        );
    }

    if (0 < ret && nullptr != pBuffer) {
        pMessage = static_cast<LPSTR>(pBuffer);
    }

    // Display the string.
    if (WcaIsInitialized()) {
        WcaLogError(dwErrorMsgId, pMessage, dwMessageId);
    } else { 
        // Log to stdout/stderr
        fprintf_s(stderr, pMessage, dwMessageId);
        if ('\n' != pMessage[strlen(pMessage) - 1]) {
            fprintf_s(stderr, "\n");
        }
    }

    // Free the buffer.
    LocalFree(pBuffer);
}

void LogResult(
    __in HRESULT hr,
    __in_z __format_string PCSTR fmt, ...
    )
{
    // This code taken from MSDN vsprintf example found currently at
    // http://msdn.microsoft.com/en-us/library/28d5ce15(v=vs.71).aspx
    // ...and then modified... because it doesn't seem to work!
    va_list args;

    va_start(args, fmt);
#pragma warning(push)
#pragma warning(disable : 4996)
    const auto len = _vsnprintf(nullptr, 0, fmt, args) + 1;
#pragma warning(pop)
    auto buffer = static_cast<char*>(malloc(len * sizeof(char)));

#ifdef _DEBUG
    ::ZeroMemory(buffer, len);
#endif // _DEBUG
    _vsnprintf_s(buffer, len, len - 1, fmt, args);

    // (MSDN code complete)

    // Now that the buffer holds the formatted string, send it to
    // the appropriate output.
    if (WcaIsInitialized()) {
        if (FAILED(hr)) {
            WcaLogError(hr, buffer);
            LogError(hr);
        } else {
            WcaLog(LOGMSG_STANDARD, buffer);
        }
    } else { // Log to stdout/stderr
        if (FAILED(hr)) {
            fprintf_s(stderr, "%s\n", buffer);
            LogError(hr);
        } else {
            fprintf_s(stdout, "%s\n", buffer);
        }
    }

    free(buffer);
}