зеркало из https://github.com/mozilla/gecko-dev.git
Bug 959591 - Add a test to verify functionality. r=fabrice
This commit is contained in:
Родитель
75b71c8774
Коммит
f1ea45eb4c
|
@ -590,6 +590,116 @@ DeviceStorageFile::Init()
|
|||
MOZ_ASSERT(typeChecker);
|
||||
}
|
||||
|
||||
// The OverrideRootDir is needed to facilitate testing of the
|
||||
// device.storage.overrideRootDir preference. The preference is normally
|
||||
// only read once during initialization, but since the test environment has
|
||||
// no convenient way to restart, we use a pref watcher instead.
|
||||
class OverrideRootDir MOZ_FINAL : public nsIObserver
|
||||
{
|
||||
public:
|
||||
NS_DECL_ISUPPORTS
|
||||
NS_DECL_NSIOBSERVER
|
||||
|
||||
static OverrideRootDir* GetSingleton();
|
||||
~OverrideRootDir();
|
||||
void Init();
|
||||
private:
|
||||
static mozilla::StaticRefPtr<OverrideRootDir> sSingleton;
|
||||
};
|
||||
|
||||
NS_IMPL_ISUPPORTS1(OverrideRootDir, nsIObserver)
|
||||
|
||||
mozilla::StaticRefPtr<OverrideRootDir>
|
||||
OverrideRootDir::sSingleton;
|
||||
|
||||
OverrideRootDir*
|
||||
OverrideRootDir::GetSingleton()
|
||||
{
|
||||
if (sSingleton) {
|
||||
return sSingleton;
|
||||
}
|
||||
// Preference changes are automatically forwarded from parent to child
|
||||
// in ContentParent::Observe, so we'll see the change in both the parent
|
||||
// and the child process.
|
||||
|
||||
sSingleton = new OverrideRootDir();
|
||||
Preferences::AddStrongObserver(sSingleton, "device.storage.overrideRootDir");
|
||||
ClearOnShutdown(&sSingleton);
|
||||
|
||||
return sSingleton;
|
||||
}
|
||||
|
||||
OverrideRootDir::~OverrideRootDir()
|
||||
{
|
||||
Preferences::RemoveObserver(this, "device.storage.overrideRootDir");
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
OverrideRootDir::Observe(nsISupports *aSubject,
|
||||
const char *aTopic,
|
||||
const char16_t *aData)
|
||||
{
|
||||
MOZ_ASSERT(NS_IsMainThread());
|
||||
|
||||
if (sSingleton) {
|
||||
sSingleton->Init();
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
void
|
||||
OverrideRootDir::Init()
|
||||
{
|
||||
MOZ_ASSERT(NS_IsMainThread());
|
||||
|
||||
if (!sDirs) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (mozilla::Preferences::GetBool("device.storage.testing", false)) {
|
||||
nsCOMPtr<nsIProperties> dirService
|
||||
= do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID);
|
||||
MOZ_ASSERT(dirService);
|
||||
dirService->Get(NS_OS_TEMP_DIR, NS_GET_IID(nsIFile),
|
||||
getter_AddRefs(sDirs->overrideRootDir));
|
||||
if (sDirs->overrideRootDir) {
|
||||
sDirs->overrideRootDir->AppendRelativeNativePath(
|
||||
NS_LITERAL_CSTRING("device-storage-testing"));
|
||||
}
|
||||
} else {
|
||||
// For users running on desktop, it's convenient to be able to override
|
||||
// all of the directories to point to a single tree, much like what happens
|
||||
// on a real device.
|
||||
const nsAdoptingString& overrideRootDir =
|
||||
mozilla::Preferences::GetString("device.storage.overrideRootDir");
|
||||
if (overrideRootDir && overrideRootDir.Length() > 0) {
|
||||
NS_NewLocalFile(overrideRootDir, false,
|
||||
getter_AddRefs(sDirs->overrideRootDir));
|
||||
} else {
|
||||
sDirs->overrideRootDir = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
if (sDirs->overrideRootDir) {
|
||||
if (XRE_GetProcessType() == GeckoProcessType_Default) {
|
||||
// Only the parent process can create directories. In testing, because
|
||||
// the preference is updated after startup, its entirely possible that
|
||||
// the preference updated notification will be received by a child
|
||||
// prior to the parent.
|
||||
nsresult rv
|
||||
= sDirs->overrideRootDir->Create(nsIFile::DIRECTORY_TYPE, 0777);
|
||||
nsString path;
|
||||
sDirs->overrideRootDir->GetPath(path);
|
||||
if (NS_FAILED(rv) && rv != NS_ERROR_FILE_ALREADY_EXISTS) {
|
||||
nsPrintfCString msg("DeviceStorage: Unable to create directory '%s'",
|
||||
NS_LossyConvertUTF16toASCII(path).get());
|
||||
NS_WARNING(msg.get());
|
||||
}
|
||||
}
|
||||
sDirs->overrideRootDir->Normalize();
|
||||
}
|
||||
}
|
||||
|
||||
// Directories which don't depend on a volume should be calculated once
|
||||
// here. Directories which depend on the root directory of a volume
|
||||
// should be calculated in DeviceStorageFile::GetRootDirectoryForType.
|
||||
|
@ -677,37 +787,7 @@ InitDirs()
|
|||
#endif
|
||||
}
|
||||
|
||||
if (mozilla::Preferences::GetBool("device.storage.testing", false)) {
|
||||
dirService->Get(NS_OS_TEMP_DIR, NS_GET_IID(nsIFile),
|
||||
getter_AddRefs(sDirs->overrideRootDir));
|
||||
if (sDirs->overrideRootDir) {
|
||||
sDirs->overrideRootDir->AppendRelativeNativePath(
|
||||
NS_LITERAL_CSTRING("device-storage-testing"));
|
||||
}
|
||||
} else {
|
||||
// For users running on desktop, it's convenient to be able to override
|
||||
// all of the directories to point to a single tree, much like what happens
|
||||
// on a real device.
|
||||
const nsAdoptingString& overrideRootDir =
|
||||
mozilla::Preferences::GetString("device.storage.overrideRootDir");
|
||||
if (overrideRootDir) {
|
||||
NS_NewLocalFile(overrideRootDir, false,
|
||||
getter_AddRefs(sDirs->overrideRootDir));
|
||||
}
|
||||
}
|
||||
|
||||
if (sDirs->overrideRootDir) {
|
||||
nsresult rv
|
||||
= sDirs->overrideRootDir->Create(nsIFile::DIRECTORY_TYPE, 0777);
|
||||
if (NS_FAILED(rv) && rv != NS_ERROR_FILE_ALREADY_EXISTS) {
|
||||
nsString path;
|
||||
sDirs->overrideRootDir->GetPath(path);
|
||||
nsPrintfCString msg("DeviceStorage: Unable to create directory '%s'",
|
||||
NS_LossyConvertUTF16toASCII(path).get());
|
||||
NS_WARNING(msg.get());
|
||||
}
|
||||
sDirs->overrideRootDir->Normalize();
|
||||
}
|
||||
OverrideRootDir::GetSingleton()->Init();
|
||||
}
|
||||
|
||||
void
|
||||
|
|
|
@ -8,6 +8,7 @@ support-files = devicestorage_common.js
|
|||
[test_addCorrectType.html]
|
||||
[test_available.html]
|
||||
[test_basic.html]
|
||||
[test_overrideDir.html]
|
||||
# [test_diskSpace.html]
|
||||
# Possible race between the time we write a file, and the
|
||||
# time it takes to be reflected by statfs(). Bug # 791287
|
||||
|
|
|
@ -0,0 +1,163 @@
|
|||
<!--
|
||||
Any copyright is dedicated to the Public Domain.
|
||||
http://creativecommons.org/publicdomain/zero/1.0/
|
||||
-->
|
||||
<!DOCTYPE HTML>
|
||||
<html> <!--
|
||||
https://bugzilla.mozilla.org/show_bug.cgi?id=717103
|
||||
-->
|
||||
<head>
|
||||
<title>Test for the device storage API </title>
|
||||
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
||||
<script type="text/javascript" src="devicestorage_common.js"></script>
|
||||
|
||||
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
|
||||
</head>
|
||||
<body>
|
||||
<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=717103">Mozilla Bug 717103</a>
|
||||
<p id="display"></p>
|
||||
<div id="content" style="display: none">
|
||||
|
||||
</div>
|
||||
<pre id="test">
|
||||
<script class="testbody" type="text/javascript">
|
||||
|
||||
devicestorage_setup();
|
||||
|
||||
var gFileName = "devicestorage/" + randomFilename(12) + "/hi.png";
|
||||
var gData = "My name is Doug Turner. My IRC nick is DougT. I like Maple cookies."
|
||||
var gDataBlob = new Blob([gData], {type: 'image/png'});
|
||||
var gFileReader = new FileReader();
|
||||
|
||||
function getAfterDeleteSuccess(e) {
|
||||
ok(false, "file was deleted not successfully");
|
||||
devicestorage_cleanup();
|
||||
}
|
||||
|
||||
function getAfterDeleteError(e) {
|
||||
ok(true, "file was deleted successfully");
|
||||
devicestorage_cleanup();
|
||||
}
|
||||
|
||||
function deleteSuccess(e) {
|
||||
|
||||
ok(e.target.result == gFileName, "File name should match");
|
||||
dump(e.target.result + "\n")
|
||||
|
||||
// File was deleted using the sdcard stoage area. It should be gone
|
||||
// from the pictures as well.
|
||||
var storage = navigator.getDeviceStorage("pictures");
|
||||
request = storage.get(e.target.result);
|
||||
request.onsuccess = getAfterDeleteSuccess;
|
||||
request.onerror = getAfterDeleteError;
|
||||
}
|
||||
|
||||
function deleteError(e) {
|
||||
ok(false, "deleteError was called : " + e.target.error.name);
|
||||
devicestorage_cleanup();
|
||||
}
|
||||
|
||||
function getSuccess(e) {
|
||||
// We wrote the file out using pictures type. Since we've over-ridden the
|
||||
// root directory, we should be able to read it back using the sdcard
|
||||
// storage area.
|
||||
var storage = navigator.getDeviceStorage("sdcard");
|
||||
ok(navigator.getDeviceStorage, "Should have getDeviceStorage");
|
||||
|
||||
ok(e.target.result.name == gFileName, "File name should match");
|
||||
ok(e.target.result.size > 0, "File size be greater than zero");
|
||||
ok(e.target.result.type, "File should have a mime type");
|
||||
ok(e.target.result.lastModifiedDate, "File should have a last modified date");
|
||||
|
||||
var name = e.target.result.name;
|
||||
|
||||
gFileReader.readAsArrayBuffer(gDataBlob);
|
||||
gFileReader.onload = function(e) {
|
||||
readerCallback(e);
|
||||
|
||||
request = storage.delete(name)
|
||||
request.onsuccess = deleteSuccess;
|
||||
request.onerror = deleteError;
|
||||
}
|
||||
}
|
||||
|
||||
function readerCallback(e) {
|
||||
|
||||
ab = e.target.result;
|
||||
|
||||
is(ab.byteLength, gData.length, "wrong arraybuffer byteLength");
|
||||
var u8v = new Uint8Array(ab);
|
||||
is(String.fromCharCode.apply(String, u8v), gData, "wrong values");
|
||||
}
|
||||
|
||||
function getError(e) {
|
||||
ok(false, "getError was called : " + e.target.error.name);
|
||||
devicestorage_cleanup();
|
||||
}
|
||||
|
||||
function addSuccess(e) {
|
||||
|
||||
var filename = e.target.result;
|
||||
if (filename[0] == "/") {
|
||||
// We got /storageName/prefix/filename
|
||||
// Remove the storageName (this shows up on FirefoxOS)
|
||||
filename = filename.substring(1); // Remove leading slash
|
||||
var slashIndex = filename.indexOf("/");
|
||||
if (slashIndex >= 0) {
|
||||
filename = filename.substring(slashIndex + 1); // Remove storageName
|
||||
}
|
||||
}
|
||||
ok(filename == gFileName, "File name should match");
|
||||
|
||||
// Update gFileName to be the fully qualified name so that
|
||||
// further checks will pass.
|
||||
gFileName = e.target.result;
|
||||
|
||||
var storage = navigator.getDeviceStorage("pictures");
|
||||
request = storage.get(gFileName);
|
||||
request.onsuccess = getSuccess;
|
||||
request.onerror = getError;
|
||||
|
||||
ok(true, "addSuccess was called");
|
||||
}
|
||||
|
||||
function addError(e) {
|
||||
ok(false, "addError was called : " + e.target.error.name);
|
||||
devicestorage_cleanup();
|
||||
}
|
||||
|
||||
function startTest() {
|
||||
ok(navigator.getDeviceStorage, "Should have getDeviceStorage");
|
||||
|
||||
var storage = navigator.getDeviceStorage("pictures");
|
||||
ok(storage, "Should have gotten a storage");
|
||||
|
||||
request = storage.addNamed(gDataBlob, gFileName);
|
||||
ok(request, "Should have a non-null request");
|
||||
|
||||
request.onsuccess = addSuccess;
|
||||
request.onerror = addError;
|
||||
}
|
||||
|
||||
try {
|
||||
const Cc = SpecialPowers.Cc;
|
||||
const Ci = SpecialPowers.Ci;
|
||||
var directoryService = Cc["@mozilla.org/file/directory_service;1"].getService(Ci.nsIProperties);
|
||||
var f = directoryService.get("TmpD", Ci.nsIFile);
|
||||
f.appendRelativePath("device-storage-sdcard");
|
||||
try {
|
||||
// The remove will fail if the directory doesn't exist, which is fine.
|
||||
f.remove(true);
|
||||
} catch (e) {}
|
||||
SpecialPowers.pushPrefEnv({'set': [["device.storage.overrideRootDir", f.path],
|
||||
["device.storage.testing", false]]},
|
||||
function() {
|
||||
startTest();
|
||||
});
|
||||
} catch(e) {}
|
||||
|
||||
</script>
|
||||
</pre>
|
||||
</body>
|
||||
</html>
|
||||
|
|
@ -16,6 +16,7 @@
|
|||
"telephony":{},
|
||||
"voicemail":{},
|
||||
"device-storage:pictures":{ "access": "readwrite" },
|
||||
"device-storage:sdcard":{ "access": "readwrite" },
|
||||
"settings":{ "access": "readwrite" },
|
||||
"storage":{},
|
||||
"camera":{},
|
||||
|
|
Загрузка…
Ссылка в новой задаче