summaryrefslogtreecommitdiffstats
path: root/toolkit/components/aboutthirdparty/MsiDatabase.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--toolkit/components/aboutthirdparty/MsiDatabase.h94
1 files changed, 94 insertions, 0 deletions
diff --git a/toolkit/components/aboutthirdparty/MsiDatabase.h b/toolkit/components/aboutthirdparty/MsiDatabase.h
new file mode 100644
index 0000000000..91c1370b8d
--- /dev/null
+++ b/toolkit/components/aboutthirdparty/MsiDatabase.h
@@ -0,0 +1,94 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef __MsiDatabase_h__
+#define __MsiDatabase_h__
+
+#include "mozilla/Maybe.h"
+#include "mozilla/UniquePtr.h"
+#include "nsWindowsHelpers.h"
+
+#include <windows.h>
+#include <msi.h>
+#include <msiquery.h>
+
+namespace mozilla {
+
+class MsiDatabase final {
+ static UniquePtr<wchar_t[]> GetRecordString(MSIHANDLE aRecord,
+ UINT aFieldIndex);
+
+ nsAutoMsiHandle mDatabase;
+
+ MsiDatabase() = default;
+ explicit MsiDatabase(const wchar_t* aDatabasePath);
+
+ public:
+ // A callback function passed to ExecuteSingleColumnQuery uses this type
+ // to control the enumeration loop.
+ enum class CallbackResult { Continue, Stop };
+
+ static Maybe<MsiDatabase> FromProductId(const wchar_t* aProductId);
+
+ MsiDatabase(const MsiDatabase&) = delete;
+ MsiDatabase& operator=(const MsiDatabase&) = delete;
+ MsiDatabase(MsiDatabase&& aOther) : mDatabase(aOther.mDatabase.disown()) {}
+ MsiDatabase& operator=(MsiDatabase&& aOther) {
+ if (this != &aOther) {
+ mDatabase.own(aOther.mDatabase.disown());
+ }
+ return *this;
+ }
+
+ explicit operator bool() const;
+
+ template <typename CallbackT>
+ bool ExecuteSingleColumnQuery(const wchar_t* aQuery,
+ const CallbackT& aCallback) const {
+ MSIHANDLE handle;
+ UINT ret = ::MsiDatabaseOpenViewW(mDatabase, aQuery, &handle);
+ if (ret != ERROR_SUCCESS) {
+ return false;
+ }
+
+ nsAutoMsiHandle view(handle);
+
+ ret = ::MsiViewExecute(view, 0);
+ if (ret != ERROR_SUCCESS) {
+ return false;
+ }
+
+ for (;;) {
+ ret = ::MsiViewFetch(view, &handle);
+ if (ret == ERROR_NO_MORE_ITEMS) {
+ break;
+ } else if (ret != ERROR_SUCCESS) {
+ return false;
+ }
+
+ nsAutoMsiHandle record(handle);
+ UniquePtr<wchar_t[]> guidStr = GetRecordString(record, 1);
+ if (!guidStr) {
+ continue;
+ }
+
+ CallbackResult result = aCallback(guidStr.get());
+ if (result == CallbackResult::Continue) {
+ continue;
+ } else if (result == CallbackResult::Stop) {
+ break;
+ } else {
+ MOZ_ASSERT_UNREACHABLE("Unexpected CallbackResult.");
+ }
+ }
+
+ return true;
+ }
+};
+
+} // namespace mozilla
+
+#endif // __MsiDatabase_h__