Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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 "mozPersonalDictionary.h"
#include <utility>
#include "nsAppDirectoryServiceDefs.h"
#include "nsCRT.h"
#include "nsIFile.h"
#include "nsIInputStream.h"
#include "nsIObserverService.h"
#include "nsIOutputStream.h"
#include "nsIRunnable.h"
#include "nsISafeOutputStream.h"
#include "nsIUnicharInputStream.h"
#include "nsIWeakReference.h"
#include "nsNetCID.h"
#include "nsNetUtil.h"
#include "nsProxyRelease.h"
#include "nsReadableUtils.h"
#include "nsStringEnumerator.h"
#include "nsTArray.h"
#include "nsThreadUtils.h"
#include "nsUnicharInputStream.h"
#include "prio.h"
#define MOZ_PERSONAL_DICT_NAME u"persdict.dat"
/**
* This is the most braindead implementation of a personal dictionary possible.
* There is not much complexity needed, though. It could be made much faster,
* and probably should, but I don't see much need for more in terms of
* interface.
*
* Allowing personal words to be associated with only certain dictionaries
* maybe.
*
* TODO:
* Implement the suggestion record.
*/
NS_IMPL_ADDREF(mozPersonalDictionary)
NS_IMPL_RELEASE(mozPersonalDictionary)
NS_INTERFACE_MAP_BEGIN(mozPersonalDictionary)
NS_INTERFACE_MAP_ENTRY(mozIPersonalDictionary)
NS_INTERFACE_MAP_ENTRY(nsIObserver)
NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, mozIPersonalDictionary)
NS_INTERFACE_MAP_END
class mozPersonalDictionaryLoader final : public mozilla::Runnable {
public:
explicit mozPersonalDictionaryLoader(mozPersonalDictionary* dict)
: mozilla::Runnable("mozPersonalDictionaryLoader"), mDict(dict) {}
NS_IMETHOD Run() override {
mDict->SyncLoad();
// Release the dictionary on the main thread
NS_ReleaseOnMainThread("mozPersonalDictionaryLoader::mDict",
mDict.forget().downcast<mozIPersonalDictionary>());
return NS_OK;
}
private:
RefPtr<mozPersonalDictionary> mDict;
};
class mozPersonalDictionarySave final : public mozilla::Runnable {
public:
explicit mozPersonalDictionarySave(mozPersonalDictionary* aDict,
nsCOMPtr<nsIFile> aFile,
nsTArray<nsString>&& aDictWords)
: mozilla::Runnable("mozPersonalDictionarySave"),
mDictWords(std::move(aDictWords)),
mFile(aFile),
mDict(aDict) {}
NS_IMETHOD Run() override {
nsresult res;
MOZ_ASSERT(!NS_IsMainThread());
{
mozilla::MonitorAutoLock mon(mDict->mMonitorSave);
nsCOMPtr<nsIOutputStream> outStream;
NS_NewSafeLocalFileOutputStream(getter_AddRefs(outStream), mFile,
PR_CREATE_FILE | PR_WRONLY | PR_TRUNCATE,
0664);
// Get a buffered output stream 4096 bytes big, to optimize writes.
nsCOMPtr<nsIOutputStream> bufferedOutputStream;
res = NS_NewBufferedOutputStream(getter_AddRefs(bufferedOutputStream),
outStream.forget(), 4096);
if (NS_FAILED(res)) {
return res;
}
uint32_t bytesWritten;
nsAutoCString utf8Key;
for (uint32_t i = 0; i < mDictWords.Length(); ++i) {
CopyUTF16toUTF8(mDictWords[i], utf8Key);
bufferedOutputStream->Write(utf8Key.get(), utf8Key.Length(),
&bytesWritten);
bufferedOutputStream->Write("\n", 1, &bytesWritten);
}
nsCOMPtr<nsISafeOutputStream> safeStream =
do_QueryInterface(bufferedOutputStream);
NS_ASSERTION(safeStream, "expected a safe output stream!");
if (safeStream) {
res = safeStream->Finish();
if (NS_FAILED(res)) {
NS_WARNING(
"failed to save personal dictionary file! possible data loss");
}
}
// Save is done, reset the state variable and notify those who are
// waiting.
mDict->mSavePending = false;
mon.Notify();
// Leaving the block where 'mon' was declared will call the destructor
// and unlock.
}
// Release the dictionary on the main thread.
NS_ReleaseOnMainThread("mozPersonalDictionarySave::mDict",
mDict.forget().downcast<mozIPersonalDictionary>());
return NS_OK;
}
private:
nsTArray<nsString> mDictWords;
nsCOMPtr<nsIFile> mFile;
RefPtr<mozPersonalDictionary> mDict;
};
mozPersonalDictionary::mozPersonalDictionary()
: mIsLoaded(false),
mSavePending(false),
mMonitor("mozPersonalDictionary::mMonitor"),
mMonitorSave("mozPersonalDictionary::mMonitorSave") {}
mozPersonalDictionary::~mozPersonalDictionary() {}
nsresult mozPersonalDictionary::Init() {
nsCOMPtr<nsIObserverService> svc =
do_GetService("@mozilla.org/observer-service;1");
NS_ENSURE_STATE(svc);
// we want to reload the dictionary if the profile switches
nsresult rv = svc->AddObserver(this, "profile-do-change", true);
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
rv = svc->AddObserver(this, "profile-before-change", true);
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
Load();
return NS_OK;
}
void mozPersonalDictionary::WaitForLoad() {
// If the dictionary is already loaded, we return straight away.
if (mIsLoaded) {
return;
}
// If the dictionary hasn't been loaded, we try to lock the same monitor
// that the thread uses that does the load. This way the main thread will
// be suspended until the monitor becomes available.
mozilla::MonitorAutoLock mon(mMonitor);
// The monitor has become available. This can have two reasons:
// 1: The thread that does the load has finished.
// 2: The thread that does the load hasn't even started.
// In this case we need to wait.
if (!mIsLoaded) {
mon.Wait();
}
}
nsresult mozPersonalDictionary::LoadInternal() {
nsresult rv;
mozilla::MonitorAutoLock mon(mMonitor);
if (mIsLoaded) {
return NS_OK;
}
rv =
NS_GetSpecialDirectory(NS_APP_USER_PROFILE_50_DIR, getter_AddRefs(mFile));
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
if (!mFile) {
return NS_ERROR_FAILURE;
}
rv = mFile->Append(nsLiteralString(MOZ_PERSONAL_DICT_NAME));
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
nsCOMPtr<nsIEventTarget> target =
do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &rv);
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
nsCOMPtr<nsIRunnable> runnable = new mozPersonalDictionaryLoader(this);
rv = target->Dispatch(runnable, NS_DISPATCH_NORMAL);
if (NS_WARN_IF(NS_FAILED(rv))) {
return rv;
}
return NS_OK;
}
NS_IMETHODIMP mozPersonalDictionary::Load() {
nsresult rv = LoadInternal();
if (NS_FAILED(rv)) {
mIsLoaded = true;
}
return rv;
}
void mozPersonalDictionary::SyncLoad() {
MOZ_ASSERT(!NS_IsMainThread());
mozilla::MonitorAutoLock mon(mMonitor);
if (mIsLoaded) {
return;
}
SyncLoadInternal();
mIsLoaded = true;
mon.Notify();
}
void mozPersonalDictionary::SyncLoadInternal() {
MOZ_ASSERT(!NS_IsMainThread());
// FIXME Deinst -- get dictionary name from prefs;
nsresult rv;
bool dictExists;
rv = mFile->Exists(&dictExists);
if (NS_FAILED(rv)) {
return;
}
if (!dictExists) {
// Nothing is really wrong...
return;
}
nsCOMPtr<nsIInputStream> inStream;
NS_NewLocalFileInputStream(getter_AddRefs(inStream), mFile);
nsCOMPtr<nsIUnicharInputStream> convStream;
rv = NS_NewUnicharInputStream(inStream, getter_AddRefs(convStream));
if (NS_FAILED(rv)) {
return;
}
// we're rereading to get rid of the old data -- we shouldn't have any,
// but...
mDictionaryTable.Clear();
char16_t c;
uint32_t nRead;
bool done = false;
do { // read each line of text into the string array.
if ((NS_OK != convStream->Read(&c, 1, &nRead)) || (nRead != 1)) break;
while (!done && ((c == '\n') || (c == '\r'))) {
if ((NS_OK != convStream->Read(&c, 1, &nRead)) || (nRead != 1))
done = true;
}
if (!done) {
nsAutoString word;
while ((c != '\n') && (c != '\r') && !done) {
word.Append(c);
if ((NS_OK != convStream->Read(&c, 1, &nRead)) || (nRead != 1))
done = true;
}
mDictionaryTable.Insert(word);
}
} while (!done);
}
void mozPersonalDictionary::WaitForSave() {
// If no save is pending, we return straight away.
if (!mSavePending) {
return;
}
// If a save is pending, we try to lock the same monitor that the thread uses
// that does the save. This way the main thread will be suspended until the
// monitor becomes available.
mozilla::MonitorAutoLock mon(mMonitorSave);
// The monitor has become available. This can have two reasons:
// 1: The thread that does the save has finished.
// 2: The thread that does the save hasn't even started.
// In this case we need to wait.
if (mSavePending) {
mon.Wait();
}
}
NS_IMETHODIMP mozPersonalDictionary::Save() {
nsCOMPtr<nsIFile> theFile;
nsresult res;
WaitForSave();
mSavePending = true;
// FIXME Deinst -- get dictionary name from prefs;
res = NS_GetSpecialDirectory(NS_APP_USER_PROFILE_50_DIR,
getter_AddRefs(theFile));
if (NS_FAILED(res)) return res;
if (!theFile) return NS_ERROR_FAILURE;
res = theFile->Append(nsLiteralString(MOZ_PERSONAL_DICT_NAME));
if (NS_FAILED(res)) return res;
nsCOMPtr<nsIEventTarget> target =
do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &res);
if (NS_WARN_IF(NS_FAILED(res))) {
return res;
}
nsCOMPtr<nsIRunnable> runnable = new mozPersonalDictionarySave(
this, theFile, mozilla::ToTArray<nsTArray<nsString>>(mDictionaryTable));
res = target->Dispatch(runnable, NS_DISPATCH_NORMAL);
if (NS_WARN_IF(NS_FAILED(res))) {
return res;
}
return res;
}
NS_IMETHODIMP mozPersonalDictionary::GetWordList(nsIStringEnumerator** aWords) {
NS_ENSURE_ARG_POINTER(aWords);
*aWords = nullptr;
WaitForLoad();
nsTArray<nsString>* array = new nsTArray<nsString>(
mozilla::ToTArray<nsTArray<nsString>>(mDictionaryTable));
array->Sort();
return NS_NewAdoptingStringEnumerator(aWords, array);
}
NS_IMETHODIMP
mozPersonalDictionary::Check(const nsAString& aWord, bool* aResult) {
NS_ENSURE_ARG_POINTER(aResult);
WaitForLoad();
*aResult = (mDictionaryTable.Contains(aWord) || mIgnoreTable.Contains(aWord));
return NS_OK;
}
NS_IMETHODIMP
mozPersonalDictionary::AddWord(const nsAString& aWord) {
nsresult res;
WaitForLoad();
mDictionaryTable.Insert(aWord);
res = Save();
return res;
}
NS_IMETHODIMP
mozPersonalDictionary::RemoveWord(const nsAString& aWord) {
nsresult res;
WaitForLoad();
mDictionaryTable.Remove(aWord);
res = Save();
return res;
}
NS_IMETHODIMP
mozPersonalDictionary::IgnoreWord(const nsAString& aWord) {
// avoid adding duplicate words to the ignore list
mIgnoreTable.EnsureInserted(aWord);
return NS_OK;
}
NS_IMETHODIMP mozPersonalDictionary::EndSession() {
WaitForLoad();
WaitForSave();
mIgnoreTable.Clear();
return NS_OK;
}
NS_IMETHODIMP mozPersonalDictionary::Observe(nsISupports* aSubject,
const char* aTopic,
const char16_t* aData) {
if (!nsCRT::strcmp(aTopic, "profile-do-change")) {
// The observer is registered in Init() which calls Load and in turn
// LoadInternal(); i.e. Observe() can't be called before Load().
WaitForLoad();
mIsLoaded = false;
Load(); // load automatically clears out the existing dictionary table
} else if (!nsCRT::strcmp(aTopic, "profile-before-change")) {
WaitForSave();
}
return NS_OK;
}