1  __doc__ = """GNUmed general tools.""" 
  2   
  3   
  4  __version__ = "$Revision: 1.13 $" 
  5  __author__ = "K. Hilbert <Karsten.Hilbert@gmx.net>" 
  6  __license__ = "GPL v2 or later (details at http://www.gnu.org)" 
  7   
  8   
  9   
 10  import os 
 11  import sys 
 12  import logging 
 13  import subprocess 
 14  import shlex 
 15   
 16   
 17  _log = logging.getLogger('gm.shell') 
 18  _log.info(__version__) 
 19   
 20   
 22   
 23          _log.debug('cmd: [%s]', cmd) 
 24          dirname = os.path.dirname(cmd) 
 25          _log.debug('dir: [%s]', dirname) 
 26          if dirname != u'': 
 27                  _log.info('command with full or relative path, not searching in PATH for binary') 
 28                  return (None, None) 
 29   
 30          env_paths = os.environ['PATH'] 
 31          _log.debug('${PATH}: %s', env_paths) 
 32          for path in env_paths.split(os.pathsep): 
 33                  candidate = os.path.join(path, cmd).encode(sys.getfilesystemencoding()) 
 34                  if os.access(candidate, os.X_OK): 
 35                          _log.debug('found [%s]', candidate) 
 36                          return (True, candidate.decode(sys.getfilesystemencoding())) 
 37                  else: 
 38                          _log.debug('not found: %s', candidate) 
 39   
 40          _log.debug('command not found in PATH') 
 41   
 42          return (False, None) 
  43   
 45   
 46          if not cmd.startswith('wine'): 
 47                  _log.debug('not a WINE call: %s', cmd) 
 48                  return (False, None) 
 49   
 50          exe_path = cmd.encode(sys.getfilesystemencoding()) 
 51   
 52          exe_path = exe_path[4:].strip().strip('"').strip() 
 53           
 54          if os.access(exe_path, os.R_OK): 
 55                  _log.debug('WINE call with UNIX path: %s', exe_path) 
 56                  return (True, cmd) 
 57   
 58           
 59          found, full_winepath_path = is_cmd_in_path(cmd = r'winepath') 
 60          if not found: 
 61                  _log.error('[winepath] not found, cannot check WINE call for Windows path conformance: %s', exe_path) 
 62                  return (False, None) 
 63   
 64           
 65          cmd_line = r'%s -u "%s"' % ( 
 66                  full_winepath_path.encode(sys.getfilesystemencoding()), 
 67                  exe_path 
 68          ) 
 69          _log.debug('converting Windows path to UNIX path: %s' % cmd_line) 
 70          cmd_line = shlex.split(cmd_line) 
 71          try: 
 72                  winepath = subprocess.Popen ( 
 73                          cmd_line, 
 74                          stdout = subprocess.PIPE, 
 75                          stderr = subprocess.PIPE, 
 76                          universal_newlines = True 
 77                  ) 
 78          except OSError: 
 79                  _log.exception('cannot run <winepath>') 
 80                  return (False, None) 
 81   
 82          stdout, stderr = winepath.communicate() 
 83          full_path = stdout.strip('\r\n') 
 84          _log.debug('UNIX path: %s', full_path) 
 85   
 86          if winepath.returncode != 0: 
 87                  _log.error('<winepath -u> returned [%s], failed to convert path', winepath.returncode) 
 88                  return (False, None) 
 89   
 90          if os.access(full_path, os.R_OK): 
 91                  _log.debug('WINE call with Windows path') 
 92                  return (True, cmd) 
 93   
 94          _log.warning('Windows path [%s] not verifiable under UNIX: %s', exe_path, full_path) 
 95          return (False, None) 
  96   
 98          _log.debug('searching for [%s]', binary) 
 99   
100          binary = binary.lstrip() 
101   
102           
103          if os.access(binary, os.X_OK): 
104                  _log.debug('found: executable explicit path') 
105                  return (True, binary) 
106   
107           
108          found, full_path = is_cmd_in_path(cmd = binary) 
109          if found: 
110                  if os.access(full_path, os.X_OK): 
111                          _log.debug('found: executable in ${PATH}') 
112                          return (True, full_path) 
113   
114           
115          is_wine_call, full_path = is_executable_by_wine(cmd = binary) 
116          if is_wine_call: 
117                  _log.debug('found: is valid WINE call') 
118                  return (True, full_path) 
119   
120           
121          if os.name == 'nt': 
122                   
123                  if not (binary.endswith('.exe') or binary.endswith('.bat')): 
124                          exe_binary = binary + r'.exe' 
125                          _log.debug('re-testing as %s', exe_binary) 
126                          found_dot_exe_binary, full_path = detect_external_binary(binary = exe_binary) 
127                          if found_dot_exe_binary: 
128                                  return (True, full_path) 
129                           
130                          bat_binary = binary + r'.bat' 
131                          _log.debug('re-testing as %s', bat_binary) 
132                          found_bat_binary, full_path = detect_external_binary(binary = bat_binary) 
133                          if found_bat_binary: 
134                                  return (True, full_path) 
135          else: 
136                  _log.debug('not running under Windows, not testing .exe/.bat') 
137   
138          return (False, None) 
 139   
141          found = False 
142          binary = None 
143   
144          for cmd in binaries: 
145                  _log.debug('looking for [%s]', cmd) 
146                  if cmd is None: 
147                          continue 
148                  found, binary = detect_external_binary(binary = cmd) 
149                  if found: 
150                          break 
151   
152          return (found, binary) 
 153   
155          """Runs a command in a subshell via standard-C system(). 
156   
157          <command> 
158                  The shell command to run including command line options. 
159          <blocking> 
160                  This will make the code *block* until the shell command exits. 
161                  It will likely only work on UNIX shells where "cmd &" makes sense. 
162          """ 
163          if acceptable_return_codes is None: 
164                  acceptable_return_codes = [0] 
165   
166          _log.debug('shell command >>>%s<<<', command) 
167          _log.debug('blocking: %s', blocking) 
168          _log.debug('acceptable return codes: %s', str(acceptable_return_codes)) 
169   
170           
171          command = command.strip() 
172   
173           
174           
175           
176           
177           
178           
179           
180           
181           
182           
183           
184           
185           
186           
187           
188           
189           
190           
191          if blocking is True: 
192                  if command[-2:] == ' &': 
193                          command = command[:-2] 
194          elif blocking is False: 
195                  if command[-2:] != ' &': 
196                          command += ' &' 
197   
198          _log.info('running shell command >>>%s<<<', command) 
199           
200          ret_val = os.system(command.encode(sys.getfilesystemencoding())) 
201          _log.debug('os.system() returned: [%s]', ret_val) 
202   
203          exited_normally = False 
204   
205          if not hasattr(os, 'WIFEXITED'): 
206                  _log.error('platform does not support exit status differentiation') 
207                  if ret_val in acceptable_return_codes: 
208                          _log.info('os.system() return value contained in acceptable return codes') 
209                          _log.info('continuing and hoping for the best') 
210                          return True 
211                  return exited_normally 
212   
213          _log.debug('exited via exit(): %s', os.WIFEXITED(ret_val)) 
214          if os.WIFEXITED(ret_val): 
215                  _log.debug('exit code: [%s]', os.WEXITSTATUS(ret_val)) 
216                  exited_normally = (os.WEXITSTATUS(ret_val) in acceptable_return_codes) 
217                  _log.debug('normal exit: %s', exited_normally) 
218          _log.debug('dumped core: %s', os.WCOREDUMP(ret_val)) 
219          _log.debug('stopped by signal: %s', os.WIFSIGNALED(ret_val)) 
220          if os.WIFSIGNALED(ret_val): 
221                  try: 
222                          _log.debug('STOP signal was: [%s]', os.WSTOPSIG(ret_val)) 
223                  except AttributeError: 
224                          _log.debug('platform does not support os.WSTOPSIG()') 
225                  try: 
226                          _log.debug('TERM signal was: [%s]', os.WTERMSIG(ret_val)) 
227                  except AttributeError: 
228                          _log.debug('platform does not support os.WTERMSIG()') 
229   
230          return exited_normally 
 231   
233   
234          found, binary = find_first_binary(binaries = binaries) 
235   
236          if not found: 
237                  _log.warning('cannot find any of: %s', binaries) 
238                  if run_last_one_anyway: 
239                          binary = binaries[-1] 
240                          _log.debug('falling back to trying to run [%s] anyway', binary) 
241                  else: 
242                          return False 
243   
244          return run_command_in_shell(command = '%s %s' % (binary, args), blocking = blocking, acceptable_return_codes = acceptable_return_codes) 
 245   
246   
247   
248  if __name__ == '__main__': 
249   
250          if len(sys.argv) < 2: 
251                  sys.exit() 
252   
253          if sys.argv[1] != u'test': 
254                  sys.exit() 
255   
256          logging.basicConfig(level = logging.DEBUG) 
257           
259                  found, path = detect_external_binary(binary = sys.argv[2]) 
260                  if found: 
261                          print "found as:", path 
262                  else: 
263                          print sys.argv[2], "not found" 
 264           
266                  print "-------------------------------------" 
267                  print "running:", sys.argv[2] 
268                  if run_command_in_shell(command=sys.argv[2], blocking=True): 
269                          print "-------------------------------------" 
270                          print "success" 
271                  else: 
272                          print "-------------------------------------" 
273                          print "failure, consult log" 
 274           
277           
280           
281           
282          test_detect_external_binary() 
283           
284           
285   
286   
287