Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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__