remoteSearch: Factor out collectFromDatadirsAsync() utility function
Processing files from a subdirectory of various share/gnome-shell directories asynchronously is a common enough pattern to share the common code. https://bugzilla.gnome.org/show_bug.cgi?id=689304
This commit is contained in:
parent
8a17f512f4
commit
c528401b62
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
const Gio = imports.gi.Gio;
|
const Gio = imports.gi.Gio;
|
||||||
const GLib = imports.gi.GLib;
|
const GLib = imports.gi.GLib;
|
||||||
|
const Lang = imports.lang;
|
||||||
|
const Params = imports.misc.params;
|
||||||
|
|
||||||
function listDirAsync(file, callback) {
|
function listDirAsync(file, callback) {
|
||||||
let allFiles = [];
|
let allFiles = [];
|
||||||
@ -23,6 +25,58 @@ function listDirAsync(file, callback) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function _collectFromDirectoryAsync(dir, loadState) {
|
||||||
|
dir.query_info_async('standard:type', Gio.FileQueryInfoFlags.NONE,
|
||||||
|
GLib.PRIORITY_DEFAULT, null, function(object, res) {
|
||||||
|
try {
|
||||||
|
object.query_info_finish(res);
|
||||||
|
} catch (e) {
|
||||||
|
if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.NOT_FOUND))
|
||||||
|
log(e.message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
loadState.numLoading++;
|
||||||
|
listDirAsync(dir, Lang.bind(this, function(infos) {
|
||||||
|
for (let i = 0; i < infos.length; i++)
|
||||||
|
loadState.processFile(dir.get_child(infos[i].get_name()),
|
||||||
|
infos[i], loadState.data);
|
||||||
|
loadState.numLoading--;
|
||||||
|
if (loadState.loadedCallback &&
|
||||||
|
loadState.numLoading == 0)
|
||||||
|
loadState.loadedCallback(loadState.data);
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function collectFromDatadirsAsync(subdir, params) {
|
||||||
|
params = Params.parse(params, { includeUserDir: false,
|
||||||
|
processFile: null,
|
||||||
|
loadedCallback: null,
|
||||||
|
data: null });
|
||||||
|
let loadState = { data: params.data,
|
||||||
|
numLoading: 0,
|
||||||
|
loadedCallback: params.loadedCallback,
|
||||||
|
processFile: params.processFile };
|
||||||
|
|
||||||
|
if (params.processFile == null) {
|
||||||
|
if (params.loadedCallback)
|
||||||
|
params.loadedCallback(params.data);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let dataDirs = GLib.get_system_data_dirs();
|
||||||
|
if (params.includeUserDir)
|
||||||
|
dataDirs.unshift(GLib.get_user_data_dir());
|
||||||
|
|
||||||
|
for (let i = 0; i < dataDirs.length; i++) {
|
||||||
|
let path = GLib.build_filenamev([dataDirs[i], 'gnome-shell', subdir]);
|
||||||
|
let dir = Gio.File.new_for_path(path);
|
||||||
|
|
||||||
|
_collectFromDirectoryAsync(dir, loadState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function deleteGFile(file) {
|
function deleteGFile(file) {
|
||||||
// Work around 'delete' being a keyword in JS.
|
// Work around 'delete' being a keyword in JS.
|
||||||
return file['delete'](null);
|
return file['delete'](null);
|
||||||
|
@ -32,50 +32,28 @@ const SearchProviderIface = <interface name="org.gnome.Shell.SearchProvider">
|
|||||||
var SearchProviderProxy = Gio.DBusProxy.makeProxyWrapper(SearchProviderIface);
|
var SearchProviderProxy = Gio.DBusProxy.makeProxyWrapper(SearchProviderIface);
|
||||||
|
|
||||||
function loadRemoteSearchProviders(addProviderCallback) {
|
function loadRemoteSearchProviders(addProviderCallback) {
|
||||||
let loadState = { loadedProviders: [],
|
let data = { loadedProviders: [],
|
||||||
objectPaths: {},
|
objectPaths: {},
|
||||||
numLoading: 0,
|
|
||||||
addProviderCallback: addProviderCallback };
|
addProviderCallback: addProviderCallback };
|
||||||
|
FileUtils.collectFromDatadirsAsync('search-providers',
|
||||||
let dataDirs = GLib.get_system_data_dirs();
|
{ loadedCallback: remoteProvidersLoaded,
|
||||||
for (let i = 0; i < dataDirs.length; i++) {
|
processFile: loadRemoteSearchProvider,
|
||||||
let path = GLib.build_filenamev([dataDirs[i], 'gnome-shell', 'search-providers']);
|
data: data
|
||||||
let dir = Gio.file_new_for_path(path);
|
|
||||||
|
|
||||||
dir.query_info_async('standard:type', Gio.FileQueryInfoFlags.NONE,
|
|
||||||
GLib.PRIORITY_DEFAULT, null,
|
|
||||||
function(object, res) {
|
|
||||||
let exists = false;
|
|
||||||
try {
|
|
||||||
object.query_info_finish(res);
|
|
||||||
exists = true;
|
|
||||||
} catch (e) {
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!exists)
|
|
||||||
return;
|
|
||||||
|
|
||||||
loadState.numLoading++;
|
|
||||||
loadRemoteSearchProvidersFromDir(dir, loadState);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
function loadRemoteSearchProvidersFromDir(dir, loadState) {
|
function loadRemoteSearchProvider(file, info, data) {
|
||||||
let dirPath = dir.get_path();
|
|
||||||
FileUtils.listDirAsync(dir, Lang.bind(this, function(files) {
|
|
||||||
for (let i = 0; i < files.length; i++) {
|
|
||||||
let keyfile = new GLib.KeyFile();
|
let keyfile = new GLib.KeyFile();
|
||||||
let path = GLib.build_filenamev([dirPath, files[i].get_name()]);
|
let path = file.get_path();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
keyfile.load_from_file(path, 0);
|
keyfile.load_from_file(path, 0);
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!keyfile.has_group(KEY_FILE_GROUP))
|
if (!keyfile.has_group(KEY_FILE_GROUP))
|
||||||
continue;
|
return;
|
||||||
|
|
||||||
let remoteProvider;
|
let remoteProvider;
|
||||||
try {
|
try {
|
||||||
@ -83,8 +61,8 @@ function loadRemoteSearchProvidersFromDir(dir, loadState) {
|
|||||||
let busName = keyfile.get_string(group, 'BusName');
|
let busName = keyfile.get_string(group, 'BusName');
|
||||||
let objectPath = keyfile.get_string(group, 'ObjectPath');
|
let objectPath = keyfile.get_string(group, 'ObjectPath');
|
||||||
|
|
||||||
if (loadState.objectPaths[objectPath])
|
if (data.objectPaths[objectPath])
|
||||||
continue;
|
return;
|
||||||
|
|
||||||
let appInfo = null;
|
let appInfo = null;
|
||||||
try {
|
try {
|
||||||
@ -92,30 +70,20 @@ function loadRemoteSearchProvidersFromDir(dir, loadState) {
|
|||||||
appInfo = Gio.DesktopAppInfo.new(desktopId);
|
appInfo = Gio.DesktopAppInfo.new(desktopId);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
log('Ignoring search provider ' + path + ': missing DesktopId');
|
log('Ignoring search provider ' + path + ': missing DesktopId');
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
remoteProvider = new RemoteSearchProvider(appInfo,
|
remoteProvider = new RemoteSearchProvider(appInfo,
|
||||||
busName,
|
busName,
|
||||||
objectPath);
|
objectPath);
|
||||||
loadState.objectPaths[objectPath] = remoteProvider;
|
data.objectPaths[objectPath] = remoteProvider;
|
||||||
loadState.loadedProviders.push(remoteProvider);
|
data.loadedProviders.push(remoteProvider);
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
log('Failed to add search provider %s: %s'.format(path, e.toString()));
|
log('Failed to add search provider %s: %s'.format(path, e.toString()));
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
remoteProvidersDirLoaded(loadState);
|
function remoteProvidersLoaded(loadState) {
|
||||||
}));
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
function remoteProvidersDirLoaded(loadState) {
|
|
||||||
loadState.numLoading--;
|
|
||||||
if (loadState.numLoading > 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
let searchSettings = new Gio.Settings({ schema: Search.SEARCH_PROVIDERS_SCHEMA });
|
let searchSettings = new Gio.Settings({ schema: Search.SEARCH_PROVIDERS_SCHEMA });
|
||||||
let sortOrder = searchSettings.get_strv('sort-order');
|
let sortOrder = searchSettings.get_strv('sort-order');
|
||||||
let numSorted = sortOrder.length;
|
let numSorted = sortOrder.length;
|
||||||
|
Loading…
Reference in New Issue
Block a user