gecko-dev/toolkit/components/parentalcontrols/nsParentalControlsServiceWi...

295 строки
8.9 KiB
C++

/* -*- Mode: C++; tab-width: 2; indent-tabs-mode:nil; c-basic-offset: 2 -*- */
/* 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/. */
#include "nsParentalControlsService.h"
#include "nsString.h"
#include "nsIArray.h"
#include "nsIWidget.h"
#include "nsIInterfaceRequestor.h"
#include "nsIInterfaceRequestorUtils.h"
#include "nsIFile.h"
#include "nsILocalFileWin.h"
#include "nsArrayUtils.h"
#include "nsIXULAppInfo.h"
#include "mozilla/UniquePtr.h"
#include "mozilla/WindowsVersion.h"
using namespace mozilla;
NS_IMPL_ISUPPORTS(nsParentalControlsService, nsIParentalControlsService)
nsParentalControlsService::nsParentalControlsService()
: mEnabled(false), mProvider(0), mPC(nullptr) {
HRESULT hr;
CoInitialize(nullptr);
hr = CoCreateInstance(__uuidof(WindowsParentalControls), nullptr,
CLSCTX_INPROC, IID_PPV_ARGS(&mPC));
if (FAILED(hr)) return;
RefPtr<IWPCSettings> wpcs;
if (FAILED(mPC->GetUserSettings(nullptr, getter_AddRefs(wpcs)))) {
// Not available on this os or not enabled for this user account or we're
// running as admin
mPC->Release();
mPC = nullptr;
return;
}
DWORD settings = 0;
wpcs->GetRestrictions(&settings);
// If we can't determine specifically whether Web Filtering is on/off (i.e.
// we're on Windows < 8), then assume it's on unless no restrictions are set.
bool enable = IsWin8OrLater() ? settings & WPCFLAG_WEB_FILTERED
: settings != WPCFLAG_NO_RESTRICTION;
if (enable) {
mEnabled = true;
}
}
nsParentalControlsService::~nsParentalControlsService() {
if (mPC) mPC->Release();
if (mProvider) {
EventUnregister(mProvider);
}
}
//------------------------------------------------------------------------
NS_IMETHODIMP
nsParentalControlsService::GetParentalControlsEnabled(bool *aResult) {
*aResult = false;
if (mEnabled) *aResult = true;
return NS_OK;
}
NS_IMETHODIMP
nsParentalControlsService::GetBlockFileDownloadsEnabled(bool *aResult) {
*aResult = false;
if (!mEnabled) return NS_ERROR_NOT_AVAILABLE;
RefPtr<IWPCWebSettings> wpcws;
if (SUCCEEDED(mPC->GetWebSettings(nullptr, getter_AddRefs(wpcws)))) {
DWORD settings = 0;
wpcws->GetSettings(&settings);
if (settings == WPCFLAG_WEB_SETTING_DOWNLOADSBLOCKED) *aResult = true;
}
return NS_OK;
}
NS_IMETHODIMP
nsParentalControlsService::GetLoggingEnabled(bool *aResult) {
*aResult = false;
if (!mEnabled) return NS_ERROR_NOT_AVAILABLE;
// Check the general purpose logging flag
RefPtr<IWPCSettings> wpcs;
if (SUCCEEDED(mPC->GetUserSettings(nullptr, getter_AddRefs(wpcs)))) {
BOOL enabled = FALSE;
wpcs->IsLoggingRequired(&enabled);
if (enabled) *aResult = true;
}
return NS_OK;
}
// Post a log event to the system
NS_IMETHODIMP
nsParentalControlsService::Log(int16_t aEntryType, bool blocked,
nsIURI *aSource, nsIFile *aTarget) {
if (!mEnabled) return NS_ERROR_NOT_AVAILABLE;
NS_ENSURE_ARG_POINTER(aSource);
// Confirm we should be logging
bool enabled;
GetLoggingEnabled(&enabled);
if (!enabled) return NS_ERROR_NOT_AVAILABLE;
// Register a Vista log event provider associated with the parental controls
// channel.
if (!mProvider) {
if (EventRegister(&WPCPROV, nullptr, nullptr, &mProvider) != ERROR_SUCCESS)
return NS_ERROR_OUT_OF_MEMORY;
}
switch (aEntryType) {
case ePCLog_URIVisit:
// Not needed, Vista's web content filter handles this for us
break;
case ePCLog_FileDownload:
LogFileDownload(blocked, aSource, aTarget);
break;
default:
break;
}
return NS_OK;
}
// Override a single URI
NS_IMETHODIMP
nsParentalControlsService::RequestURIOverride(
nsIURI *aTarget, nsIInterfaceRequestor *aWindowContext, bool *_retval) {
*_retval = false;
if (!mEnabled) return NS_ERROR_NOT_AVAILABLE;
NS_ENSURE_ARG_POINTER(aTarget);
nsAutoCString spec;
aTarget->GetSpec(spec);
if (spec.IsEmpty()) return NS_ERROR_INVALID_ARG;
HWND hWnd = nullptr;
// If we have a native window, use its handle instead
nsCOMPtr<nsIWidget> widget(do_GetInterface(aWindowContext));
if (widget) hWnd = (HWND)widget->GetNativeData(NS_NATIVE_WINDOW);
if (hWnd == nullptr) hWnd = GetDesktopWindow();
BOOL ret;
RefPtr<IWPCWebSettings> wpcws;
if (SUCCEEDED(mPC->GetWebSettings(nullptr, getter_AddRefs(wpcws)))) {
wpcws->RequestURLOverride(hWnd, NS_ConvertUTF8toUTF16(spec).get(), 0,
nullptr, &ret);
*_retval = ret;
}
return NS_OK;
}
// Override a web page
NS_IMETHODIMP
nsParentalControlsService::RequestURIOverrides(
nsIArray *aTargets, nsIInterfaceRequestor *aWindowContext, bool *_retval) {
*_retval = false;
if (!mEnabled) return NS_ERROR_NOT_AVAILABLE;
NS_ENSURE_ARG_POINTER(aTargets);
uint32_t arrayLength = 0;
aTargets->GetLength(&arrayLength);
if (!arrayLength) return NS_ERROR_INVALID_ARG;
if (arrayLength == 1) {
nsCOMPtr<nsIURI> uri = do_QueryElementAt(aTargets, 0);
if (!uri) return NS_ERROR_INVALID_ARG;
return RequestURIOverride(uri, aWindowContext, _retval);
}
HWND hWnd = nullptr;
// If we have a native window, use its handle instead
nsCOMPtr<nsIWidget> widget(do_GetInterface(aWindowContext));
if (widget) hWnd = (HWND)widget->GetNativeData(NS_NATIVE_WINDOW);
if (hWnd == nullptr) hWnd = GetDesktopWindow();
// The first entry should be the root uri
nsAutoCString rootSpec;
nsCOMPtr<nsIURI> rootURI = do_QueryElementAt(aTargets, 0);
if (!rootURI) return NS_ERROR_INVALID_ARG;
rootURI->GetSpec(rootSpec);
if (rootSpec.IsEmpty()) return NS_ERROR_INVALID_ARG;
// Allocate an array of sub uri
int32_t count = arrayLength - 1;
auto arrUrls = MakeUnique<LPCWSTR[]>(count);
uint32_t uriIdx = 0, idx;
for (idx = 1; idx < arrayLength; idx++) {
nsCOMPtr<nsIURI> uri = do_QueryElementAt(aTargets, idx);
if (!uri) continue;
nsAutoCString subURI;
if (NS_FAILED(uri->GetSpec(subURI))) continue;
arrUrls[uriIdx] = (LPCWSTR)UTF8ToNewUnicode(subURI); // allocation
if (!arrUrls[uriIdx]) continue;
uriIdx++;
}
if (!uriIdx) return NS_ERROR_INVALID_ARG;
BOOL ret;
RefPtr<IWPCWebSettings> wpcws;
if (SUCCEEDED(mPC->GetWebSettings(nullptr, getter_AddRefs(wpcws)))) {
wpcws->RequestURLOverride(hWnd, NS_ConvertUTF8toUTF16(rootSpec).get(),
uriIdx, (LPCWSTR *)arrUrls.get(), &ret);
*_retval = ret;
}
// Free up the allocated strings in our array
for (idx = 0; idx < uriIdx; idx++) free((void *)arrUrls[idx]);
return NS_OK;
}
//------------------------------------------------------------------------
// Sends a file download event to the Vista Event Log
void nsParentalControlsService::LogFileDownload(bool blocked, nsIURI *aSource,
nsIFile *aTarget) {
nsAutoCString curi;
// Note, EventDataDescCreate is a macro defined in the headers, not a function
aSource->GetSpec(curi);
nsAutoString uri = NS_ConvertUTF8toUTF16(curi);
// Get the name of the currently running process
nsCOMPtr<nsIXULAppInfo> appInfo =
do_GetService("@mozilla.org/xre/app-info;1");
nsAutoCString asciiAppName;
if (appInfo) appInfo->GetName(asciiAppName);
nsAutoString appName = NS_ConvertUTF8toUTF16(asciiAppName);
static const WCHAR fill[] = L"";
// See wpcevent.h and msdn for event formats
EVENT_DATA_DESCRIPTOR eventData[WPC_ARGS_FILEDOWNLOADEVENT_CARGS];
DWORD dwBlocked = blocked;
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_URL],
(const void *)uri.get(),
((ULONG)uri.Length() + 1) * sizeof(WCHAR));
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_APPNAME],
(const void *)appName.get(),
((ULONG)appName.Length() + 1) * sizeof(WCHAR));
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_VERSION],
(const void *)fill, sizeof(fill));
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_BLOCKED],
(const void *)&dwBlocked, sizeof(dwBlocked));
nsCOMPtr<nsILocalFileWin> local(do_QueryInterface(aTarget)); // May be null
if (local) {
nsAutoString path;
local->GetCanonicalPath(path);
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_PATH],
(const void *)path.get(),
((ULONG)path.Length() + 1) * sizeof(WCHAR));
} else {
EventDataDescCreate(&eventData[WPC_ARGS_FILEDOWNLOADEVENT_PATH],
(const void *)fill, sizeof(fill));
}
EventWrite(mProvider, &WPCEVENT_WEB_FILEDOWNLOAD, ARRAYSIZE(eventData),
eventData);
}
NS_IMETHODIMP
nsParentalControlsService::IsAllowed(int16_t aAction, nsIURI *aUri,
bool *_retval) {
return NS_ERROR_NOT_AVAILABLE;
}