2 # -*- coding: utf-8 -*-
4 # misc.py - collection of miscellaneous functions and classes
6 # Part of WiFi Radar: A utility for managing WiFi profiles on GNU/Linux.
8 # Copyright (C) 2004-2005 Ahmad Baitalmal <ahmad@baitalmal.com>
9 # Copyright (C) 2005 Nicolas Brouard <nicolas.brouard@mandrake.org>
10 # Copyright (C) 2005-2009 Brian Elliott Finley <brian@thefinleys.com>
11 # Copyright (C) 2006 David Decotigny <com.d2@free.fr>
12 # Copyright (C) 2006 Simon Gerber <gesimu@gmail.com>
13 # Copyright (C) 2006-2007 Joey Hurst <jhurst@lucubrate.org>
14 # Copyright (C) 2006, 2009 Ante Karamatic <ivoks@ubuntu.com>
15 # Copyright (C) 2009-2010,2014 Sean Robinson <robinson@tuxfamily.org>
16 # Copyright (C) 2010 Prokhor Shuchalov <p@shuchalov.ru>
18 # This program is free software; you can redistribute it and/or modify
19 # it under the terms of the GNU General Public License as published by
20 # the Free Software Foundation; version 2 of the License.
22 # This program is distributed in the hope that it will be useful,
23 # but WITHOUT ANY WARRANTY; without even the implied warranty of
24 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
25 # GNU General Public License in LICENSE.GPL for more details.
27 # You should have received a copy of the GNU General Public License
28 # along with this program; if not, write to:
29 # Free Software Foundation, Inc.
30 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
34 from __future__
import unicode_literals
43 from subprocess
import check_call
, Popen
, PIPE
, STDOUT
45 WIFI_RADAR_VERSION
= '0.0.0'
48 logger
= logging
.getLogger(__name__
)
50 __all__
= ['_', 'PYVERSION', 'WIFI_RADAR_VERSION', 'DeviceError',
51 'NoDeviceError', 'PipeError', 'KeySortedDict',
52 'get_new_profile', 'shellcmd']
54 PYVERSION
= sys
.version_info
[0]
56 # Look for string translations in the default location, then in the po
57 # directory under the current directory. This second location is useful
59 for locale_dir
in [gettext
.bindtextdomain(None), os
.path
.realpath('./po/')]:
61 trans
= gettext
.translation('wifi-radar', locale_dir
)
63 logger
.warning("{EXC} in '{DIR}'".format(EXC
=e
, DIR
=locale_dir
))
65 logger
.info("Translation file found in '{DIR}'".format(DIR
=locale_dir
))
68 logger
.warning('Translation disabled. Please report this '
69 'at http://wifi-radar.tuxfamily.org/')
70 trans
= gettext
.NullTranslations()
71 # Return a reference to the unicode version of gettext, no matter
78 class DeviceError(EnvironmentError):
79 """ DeviceError will be raised when a problem is found while changing
80 a wireless device's configuration (e.g. when 'ifconfig wlan0 up'
86 class NoDeviceError(EnvironmentError):
87 """ NoDeviceError will be raised when a wireless device cannot be
93 class PipeError(IOError):
94 """ PipeError is raised when a communication Pipe closes unexpectedly.
99 generic_formatter
= logging
.Formatter(
100 fmt
='%(asctime)s.%(msecs)d %(levelname)s %(name)s: %(message)s',
101 datefmt
='%Y-%m-%d %H:%M:%S')
104 class KeySortedDict(collections
.MutableMapping
):
105 """ :class:`KeySortedDict` provides a dictionary whose keys are always
106 returned in sorted order. Internal storage is not affected by the
109 def __init__(self
, *args
, **kwargs
):
110 self
._key
_list
= list()
111 self
._backing
_dict
= dict()
112 self
.update(*args
, **kwargs
)
114 def __setitem__(self
, key
, value
):
115 """ ksd.__setitem__(k, v) <==> ksd[k] = v """
118 self
._backing
_dict
[key
] = value
120 self
._key
_list
.append(key
)
121 self
._backing
_dict
[key
] = value
123 def __getitem__(self
, key
):
124 """ ksd.__getitem__(k) <==> ksd[k] """
125 return self
._backing
_dict
[key
]
127 def __delitem__(self
, key
):
128 """ ksd.__delitem__(k) <==> del ksd[k] """
129 del self
._backing
_dict
[key
]
130 self
._key
_list
.remove(key
)
133 """ ksd.__iter__() <==> iter(ksd) """
134 self
._key
_list
.sort()
135 for key
in self
._key
_list
:
139 """ ksd.__len__ <==> len(ksd) """
140 return len(self
._key
_list
)
143 return '{}({})'.format(self
.__class
__.__name
__, self
.items())
146 return self
._backing
_dict
.copy()
149 def get_new_profile():
150 """ Return a blank profile.
158 dictionary -- An AP profile with defaults set.
160 return { 'known': False,
170 'con_postscript': '',
172 'dis_postscript': '',
187 def shellcmd(command
, environment
=None):
188 """ Run 'command' through the shell, where 'command' is a tuple of
189 the program to execute and its arguments. 'environment' is a
190 dictionary of shell environment variables (as keys) and their
191 values. Returns 0 on success, otherwise, it raises
195 env_tmp
.update(environment
)
196 return check_call(command
, shell
=True, env
=env_tmp
)
199 # Make so we can be imported
200 if __name__
== '__main__':