2008-02-09 23:14:18 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2010-02-17 21:11:36 +00:00
|
|
|
# openbox-xdg-autostart runs things based on the XDG autostart specification
|
2008-02-09 23:14:18 +00:00
|
|
|
# Copyright (C) 2008 Dana Jansens
|
|
|
|
#
|
|
|
|
# XDG autostart specification can be found here:
|
|
|
|
# http://standards.freedesktop.org/autostart-spec/
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# LICENSE:
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
|
2010-02-17 21:11:36 +00:00
|
|
|
ME="openbox-xdg-autostart"
|
|
|
|
VERSION="1.1"
|
2008-02-09 23:14:18 +00:00
|
|
|
|
|
|
|
import os, glob, sys
|
2008-02-09 23:26:15 +00:00
|
|
|
try:
|
|
|
|
from xdg import BaseDirectory
|
|
|
|
from xdg.DesktopEntry import DesktopEntry
|
|
|
|
from xdg.Exceptions import ParsingError
|
|
|
|
except ImportError:
|
|
|
|
print
|
2012-08-16 23:57:08 +00:00
|
|
|
print >>sys.stderr, "ERROR:", ME, "requires PyXDG to be installed"
|
2008-02-09 23:26:15 +00:00
|
|
|
print
|
|
|
|
sys.exit(1)
|
2008-02-09 23:14:18 +00:00
|
|
|
|
|
|
|
def main(argv=sys.argv):
|
|
|
|
if "--help" in argv[1:]:
|
|
|
|
show_help()
|
|
|
|
return 0
|
|
|
|
if "--version" in argv[1:]:
|
|
|
|
show_version()
|
|
|
|
return 0
|
|
|
|
|
|
|
|
# get the autostart directories
|
|
|
|
autodirs = BaseDirectory.load_config_paths("autostart")
|
|
|
|
|
|
|
|
# find all the autostart files
|
|
|
|
files = []
|
|
|
|
for dir in autodirs:
|
|
|
|
for path in glob.glob(os.path.join(dir, '*.desktop')):
|
|
|
|
try:
|
|
|
|
autofile = AutostartFile(path)
|
|
|
|
except ParsingError:
|
|
|
|
print "Invalid .desktop file: " + path
|
|
|
|
else:
|
|
|
|
if not autofile in files:
|
|
|
|
files.append(autofile)
|
|
|
|
|
|
|
|
list = False
|
|
|
|
if "--list" in argv[1:]:
|
|
|
|
list = True
|
|
|
|
argv.remove("--list")
|
|
|
|
|
|
|
|
# run them !
|
|
|
|
environments = argv[1:]
|
|
|
|
for autofile in files:
|
2008-03-02 15:45:28 +00:00
|
|
|
if list: autofile.display(environments)
|
2008-02-09 23:14:18 +00:00
|
|
|
else: autofile.run(environments)
|
|
|
|
|
2008-02-11 14:40:24 +00:00
|
|
|
class AutostartFile:
|
2008-02-09 23:14:18 +00:00
|
|
|
def __init__(self, path):
|
|
|
|
self.path = path
|
|
|
|
self.filename = os.path.basename(path)
|
|
|
|
self.dirname = os.path.dirname(path)
|
|
|
|
self.de = DesktopEntry(path)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
return self.filename == other.filename
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.path + " : " + self.de.getName()
|
|
|
|
|
2009-12-10 14:40:20 +00:00
|
|
|
def _isexecfile(self, path):
|
2008-02-09 23:14:18 +00:00
|
|
|
return os.access(path, os.X_OK)
|
|
|
|
|
2008-03-02 15:45:28 +00:00
|
|
|
def _findFile(self, path, search, match_func):
|
2008-02-09 23:14:18 +00:00
|
|
|
# check empty path
|
|
|
|
if not path: return None
|
|
|
|
# check absolute path
|
|
|
|
if path[0] == '/':
|
|
|
|
if match_func(path): return path
|
|
|
|
else: return None
|
2008-03-02 15:53:09 +00:00
|
|
|
else:
|
2008-02-09 23:14:18 +00:00
|
|
|
# check relative path
|
|
|
|
for dirname in search.split(os.pathsep):
|
|
|
|
if dirname != "":
|
|
|
|
candidate = os.path.join(dirname, path)
|
|
|
|
if (match_func(candidate)): return candidate
|
|
|
|
|
2008-03-02 15:45:28 +00:00
|
|
|
def _alert(self, str, info=False):
|
2008-02-09 23:14:18 +00:00
|
|
|
if info:
|
|
|
|
print "\t ", str
|
|
|
|
else:
|
|
|
|
print "\t*", str
|
|
|
|
|
2008-03-02 15:45:28 +00:00
|
|
|
def _showInEnvironment(self, envs, verbose=False):
|
2008-02-09 23:14:18 +00:00
|
|
|
default = not self.de.getOnlyShowIn()
|
|
|
|
noshow = False
|
|
|
|
force = False
|
|
|
|
for i in self.de.getOnlyShowIn():
|
|
|
|
if i in envs: force = True
|
|
|
|
for i in self.de.getNotShowIn():
|
|
|
|
if i in envs: noshow = True
|
|
|
|
|
|
|
|
if verbose:
|
|
|
|
if not default and not force:
|
|
|
|
s = ""
|
|
|
|
for i in self.de.getOnlyShowIn():
|
|
|
|
if s: s += ", "
|
|
|
|
s += i
|
2008-03-02 15:45:28 +00:00
|
|
|
self._alert("Excluded by: OnlyShowIn (" + s + ")")
|
2008-02-09 23:14:18 +00:00
|
|
|
if default and noshow and not force:
|
|
|
|
s = ""
|
2008-03-02 16:41:08 +00:00
|
|
|
for i in self.de.getNotShowIn():
|
2008-02-09 23:14:18 +00:00
|
|
|
if s: s += ", "
|
|
|
|
s += i
|
2008-03-02 15:45:28 +00:00
|
|
|
self._alert("Excluded by: NotShowIn (" + s + ")")
|
2008-02-09 23:14:18 +00:00
|
|
|
return (default and not noshow) or force
|
|
|
|
|
2008-03-02 15:45:28 +00:00
|
|
|
def _shouldRun(self, envs, verbose=False):
|
2008-02-09 23:14:18 +00:00
|
|
|
if not self.de.getExec():
|
2008-03-02 15:45:28 +00:00
|
|
|
if verbose: self._alert("Excluded by: Missing Exec field")
|
2008-02-09 23:14:18 +00:00
|
|
|
return False
|
|
|
|
if self.de.getHidden():
|
2008-03-02 15:45:28 +00:00
|
|
|
if verbose: self._alert("Excluded by: Hidden")
|
2008-02-09 23:14:18 +00:00
|
|
|
return False
|
|
|
|
if self.de.getTryExec():
|
2008-03-02 15:45:28 +00:00
|
|
|
if not self._findFile(self.de.getTryExec(), os.getenv("PATH"),
|
|
|
|
self._isexecfile):
|
|
|
|
if verbose: self._alert("Excluded by: TryExec (" +
|
|
|
|
self.de.getTryExec() + ")")
|
2008-02-09 23:14:18 +00:00
|
|
|
return False
|
2008-03-02 15:45:28 +00:00
|
|
|
if not self._showInEnvironment(envs, verbose):
|
2008-02-09 23:14:18 +00:00
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2008-03-02 15:45:28 +00:00
|
|
|
def display(self, envs):
|
|
|
|
if self._shouldRun(envs):
|
2008-02-09 23:14:18 +00:00
|
|
|
print "[*] " + self.de.getName()
|
|
|
|
else:
|
|
|
|
print "[ ] " + self.de.getName()
|
2008-03-02 15:45:28 +00:00
|
|
|
self._alert("File: " + self.path, info=True)
|
2008-02-09 23:14:18 +00:00
|
|
|
if self.de.getExec():
|
2008-03-02 15:45:28 +00:00
|
|
|
self._alert("Executes: " + self.de.getExec(), info=True)
|
|
|
|
self._shouldRun(envs, True)
|
2008-02-09 23:14:18 +00:00
|
|
|
print
|
|
|
|
|
|
|
|
def run(self, envs):
|
|
|
|
here = os.getcwd()
|
|
|
|
if self.de.getPath():
|
|
|
|
os.chdir(self.de.getPath())
|
2008-03-02 15:45:28 +00:00
|
|
|
if self._shouldRun(envs):
|
2008-02-09 23:26:15 +00:00
|
|
|
args = ["/bin/sh", "-c", "exec " + self.de.getExec()]
|
|
|
|
os.spawnv(os.P_NOWAIT, args[0], args);
|
2008-02-09 23:14:18 +00:00
|
|
|
os.chdir(here)
|
|
|
|
|
|
|
|
def show_help():
|
|
|
|
print "Usage:", ME, "[OPTION]... [ENVIRONMENT]..."
|
|
|
|
print
|
|
|
|
print "This tool will run xdg autostart .desktop files"
|
|
|
|
print
|
|
|
|
print "OPTIONS"
|
|
|
|
print " --list Show a list of the files which would be run"
|
|
|
|
print " Files which would be run are marked with an asterix"
|
|
|
|
print " symbol [*]. For files which would not be run,"
|
|
|
|
print " information is given for why they are excluded"
|
|
|
|
print " --help Show this help and exit"
|
|
|
|
print " --version Show version and copyright information"
|
|
|
|
print
|
|
|
|
print "ENVIRONMENT specifies a list of environments for which to run autostart"
|
2008-02-10 07:06:46 +00:00
|
|
|
print "applications. If none are specified, only applications which do not "
|
2008-02-09 23:14:18 +00:00
|
|
|
print "limit themselves to certain environments will be run."
|
|
|
|
print
|
|
|
|
print "ENVIRONMENT can be one or more of:"
|
|
|
|
print " GNOME Gnome Desktop"
|
|
|
|
print " KDE KDE Desktop"
|
|
|
|
print " ROX ROX Desktop"
|
|
|
|
print " XFCE XFCE Desktop"
|
|
|
|
print " Old Legacy systems"
|
|
|
|
print
|
|
|
|
|
|
|
|
def show_version():
|
|
|
|
print ME, VERSION
|
|
|
|
print "Copyright (c) 2008 Dana Jansens"
|
|
|
|
print
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
sys.exit(main())
|