forked from nvaccess/nvda
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathappModuleHandler.py
490 lines (447 loc) · 19.5 KB
/
appModuleHandler.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
# -*- coding: UTF-8 -*-
#appModuleHandler.py
#A part of NonVisual Desktop Access (NVDA)
#Copyright (C) 2006-2018 NV Access Limited, Peter Vágner, Aleksey Sadovoy, Patrick Zajda, Joseph Lee, Babbage B.V.
#This file is covered by the GNU General Public License.
#See the file COPYING for more details.
"""Manages appModules.
@var runningTable: a dictionary of the currently running appModules, using their application's main window handle as a key.
@type runningTable: dict
"""
import itertools
import array
import ctypes
import ctypes.wintypes
import os
import sys
import winVersion
import pkgutil
import threading
import tempfile
import comtypes.client
import baseObject
import globalVars
from logHandler import log
import NVDAHelper
import ui
import winUser
import winKernel
import config
import NVDAObjects #Catches errors before loading default appModule
import api
import appModules
import watchdog
#Dictionary of processID:appModule paires used to hold the currently running modules
runningTable={}
#: The process ID of NVDA itself.
NVDAProcessID=None
_importers=None
_getAppModuleLock=threading.RLock()
class processEntry32W(ctypes.Structure):
_fields_ = [
("dwSize",ctypes.wintypes.DWORD),
("cntUsage", ctypes.wintypes.DWORD),
("th32ProcessID", ctypes.wintypes.DWORD),
("th32DefaultHeapID", ctypes.wintypes.DWORD),
("th32ModuleID",ctypes.wintypes.DWORD),
("cntThreads",ctypes.wintypes.DWORD),
("th32ParentProcessID",ctypes.wintypes.DWORD),
("pcPriClassBase",ctypes.c_long),
("dwFlags",ctypes.wintypes.DWORD),
("szExeFile", ctypes.c_wchar * 260)
]
def getAppNameFromProcessID(processID,includeExt=False):
"""Finds out the application name of the given process.
@param processID: the ID of the process handle of the application you wish to get the name of.
@type processID: int
@param includeExt: C{True} to include the extension of the application's executable filename, C{False} to exclude it.
@type window: bool
@returns: application name
@rtype: unicode or str
"""
if processID==NVDAProcessID:
return "nvda.exe" if includeExt else "nvda"
FSnapshotHandle = winKernel.kernel32.CreateToolhelp32Snapshot (2,0)
FProcessEntry32 = processEntry32W()
FProcessEntry32.dwSize = ctypes.sizeof(processEntry32W)
ContinueLoop = winKernel.kernel32.Process32FirstW(FSnapshotHandle, ctypes.byref(FProcessEntry32))
appName = unicode()
while ContinueLoop:
if FProcessEntry32.th32ProcessID == processID:
appName = FProcessEntry32.szExeFile
break
ContinueLoop = winKernel.kernel32.Process32NextW(FSnapshotHandle, ctypes.byref(FProcessEntry32))
winKernel.kernel32.CloseHandle(FSnapshotHandle)
if not includeExt:
appName=os.path.splitext(appName)[0].lower()
if not appName:
return appName
# This might be an executable which hosts multiple apps.
# Try querying the app module for the name of the app being hosted.
try:
# Python 2.x can't properly handle unicode module names, so convert them.
mod = __import__("appModules.%s" % appName.encode("mbcs"),
globals(), locals(), ("appModules",))
return mod.getAppNameFromHost(processID)
except (ImportError, AttributeError, LookupError):
pass
return appName
def getAppModuleForNVDAObject(obj):
if not isinstance(obj,NVDAObjects.NVDAObject):
return
return getAppModuleFromProcessID(obj.processID)
def getAppModuleFromProcessID(processID):
"""Finds the appModule that is for the given process ID. The module is also cached for later retreavals.
@param processID: The ID of the process for which you wish to find the appModule.
@type processID: int
@returns: the appModule, or None if there isn't one
@rtype: appModule
"""
with _getAppModuleLock:
mod=runningTable.get(processID)
if not mod:
# #5323: Certain executables contain dots as part of their file names.
appName=getAppNameFromProcessID(processID).replace(".","_")
mod=fetchAppModule(processID,appName)
if not mod:
raise RuntimeError("error fetching default appModule")
runningTable[processID]=mod
return mod
def update(processID,helperLocalBindingHandle=None,inprocRegistrationHandle=None):
"""Tries to load a new appModule for the given process ID if need be.
@param processID: the ID of the process.
@type processID: int
@param helperLocalBindingHandle: an optional RPC binding handle pointing to the RPC server for this process
@param inprocRegistrationHandle: an optional rpc context handle representing successful registration with the rpc server for this process
"""
# This creates a new app module if necessary.
mod=getAppModuleFromProcessID(processID)
if helperLocalBindingHandle:
mod.helperLocalBindingHandle=helperLocalBindingHandle
if inprocRegistrationHandle:
mod._inprocRegistrationHandle=inprocRegistrationHandle
def cleanup():
"""Removes any appModules from the cache whose process has died.
"""
for deadMod in [mod for mod in runningTable.itervalues() if not mod.isAlive]:
log.debug("application %s closed"%deadMod.appName)
del runningTable[deadMod.processID]
if deadMod in set(o.appModule for o in api.getFocusAncestors()+[api.getFocusObject()] if o and o.appModule):
if hasattr(deadMod,'event_appLoseFocus'):
deadMod.event_appLoseFocus()
import eventHandler
eventHandler.handleAppTerminate(deadMod)
try:
deadMod.terminate()
except:
log.exception("Error terminating app module %r" % deadMod)
def doesAppModuleExist(name):
return any(importer.find_module("appModules.%s" % name) for importer in _importers)
def fetchAppModule(processID,appName):
"""Returns an appModule found in the appModules directory, for the given application name.
@param processID: process ID for it to be associated with
@type processID: integer
@param appName: the application name for which an appModule should be found.
@type appName: unicode or str
@returns: the appModule, or None if not found
@rtype: AppModule
"""
# First, check whether the module exists.
# We need to do this separately because even though an ImportError is raised when a module can't be found, it might also be raised for other reasons.
# Python 2.x can't properly handle unicode module names, so convert them.
modName = appName.encode("mbcs")
if doesAppModuleExist(modName):
try:
return __import__("appModules.%s" % modName, globals(), locals(), ("appModules",)).AppModule(processID, appName)
except:
log.error("error in appModule %r"%modName, exc_info=True)
# We can't present a message which isn't unicode, so use appName, not modName.
# Translators: This is presented when errors are found in an appModule (example output: error in appModule explorer).
ui.message(_("Error in appModule %s")%appName)
# Use the base AppModule.
return AppModule(processID, appName)
def reloadAppModules():
"""Reloads running appModules.
especially, it clears the cache of running appModules and deletes them from sys.modules.
Each appModule will then be reloaded immediately.
"""
global appModules
state = []
for mod in runningTable.itervalues():
state.append({key: getattr(mod, key) for key in ("processID",
# #2892: We must save nvdaHelperRemote handles, as we can't reinitialize without a foreground/focus event.
# Also, if there is an active context handle such as a loaded buffer,
# nvdaHelperRemote can't reinit until that handle dies.
"helperLocalBindingHandle", "_inprocRegistrationHandle",
# #5380: We must save config profile triggers so they can be cleaned up correctly.
# Otherwise, they'll remain active forever.
"_configProfileTrigger",
) if hasattr(mod, key)})
# #2892: Don't disconnect from nvdaHelperRemote during termination.
mod._helperPreventDisconnect = True
terminate()
del appModules
mods=[k for k,v in sys.modules.iteritems() if k.startswith("appModules") and v is not None]
for mod in mods:
del sys.modules[mod]
import appModules
initialize()
for entry in state:
pid = entry.pop("processID")
mod = getAppModuleFromProcessID(pid)
mod.__dict__.update(entry)
# The appModule property for existing NVDAObjects will now be None, since their AppModule died.
# Force focus, navigator, etc. objects to re-fetch,
# since NVDA depends on the appModule property for these.
for obj in itertools.chain((api.getFocusObject(), api.getNavigatorObject()), api.getFocusAncestors()):
try:
del obj._appModuleRef
except AttributeError:
continue
# Fetch and cache right away; the process could die any time.
obj.appModule
def initialize():
"""Initializes the appModule subsystem.
"""
global NVDAProcessID,_importers
NVDAProcessID=os.getpid()
config.addConfigDirsToPythonPackagePath(appModules)
_importers=list(pkgutil.iter_importers("appModules.__init__"))
def terminate():
for processID, app in runningTable.iteritems():
try:
app.terminate()
except:
log.exception("Error terminating app module %r" % app)
runningTable.clear()
def handleAppSwitch(oldMods, newMods):
newModsSet = set(newMods)
processed = set()
nextStage = []
# Determine all apps that are losing focus and fire appropriate events.
for mod in reversed(oldMods):
if mod in processed:
# This app has already been handled.
continue
processed.add(mod)
if mod in newModsSet:
# This app isn't losing focus.
continue
processed.add(mod)
# This app is losing focus.
nextStage.append(mod)
if not mod.sleepMode and hasattr(mod,'event_appModule_loseFocus'):
try:
mod.event_appModule_loseFocus()
except watchdog.CallCancelled:
pass
nvdaGuiLostFocus = nextStage and nextStage[-1].appName == "nvda"
if not nvdaGuiLostFocus and (not oldMods or oldMods[-1].appName != "nvda") and newMods[-1].appName == "nvda":
# NVDA's GUI just got focus.
import gui
if gui.shouldConfigProfileTriggersBeSuspended():
config.conf.suspendProfileTriggers()
with config.conf.atomicProfileSwitch():
# Exit triggers for apps that lost focus.
for mod in nextStage:
mod._configProfileTrigger.exit()
mod._configProfileTrigger = None
nextStage = []
# Determine all apps that are gaining focus and enter triggers.
for mod in newMods:
if mod in processed:
# This app isn't gaining focus or it has already been handled.
continue
processed.add(mod)
# This app is gaining focus.
nextStage.append(mod)
trigger = mod._configProfileTrigger = AppProfileTrigger(mod.appName)
trigger.enter()
if nvdaGuiLostFocus:
import gui
if not gui.shouldConfigProfileTriggersBeSuspended():
config.conf.resumeProfileTriggers()
# Fire appropriate events for apps gaining focus.
for mod in nextStage:
if not mod.sleepMode and hasattr(mod,'event_appModule_gainFocus'):
mod.event_appModule_gainFocus()
#base class for appModules
class AppModule(baseObject.ScriptableObject):
"""Base app module.
App modules provide specific support for a single application.
Each app module should be a Python module in the appModules package named according to the executable it supports;
e.g. explorer.py for the explorer.exe application.
It should containa C{AppModule} class which inherits from this base class.
App modules can implement and bind gestures to scripts.
These bindings will only take effect while an object in the associated application has focus.
See L{ScriptableObject} for details.
App modules can also receive NVDAObject events for objects within the associated application.
This is done by implementing methods called C{event_eventName},
where C{eventName} is the name of the event; e.g. C{event_gainFocus}.
These event methods take two arguments: the NVDAObject on which the event was fired
and a callable taking no arguments which calls the next event handler.
Some executables host many different applications; e.g. javaw.exe.
In this case, it is desirable that a specific app module be loaded for each
actual application, rather than the one for the hosting executable.
To support this, the module for the hosting executable
(not the C{AppModule} class within it) can implement the function
C{getAppNameFromHost(processId)}, where C{processId} is the id of the host process.
It should return a unicode string specifying the name that should be used.
Alternatively, it can raise C{LookupError} if a name couldn't be determined.
"""
#: Whether NVDA should sleep while in this application (e.g. the application is self-voicing).
#: If C{True}, all events and script requests inside this application are silently dropped.
#: @type: bool
sleepMode=False
def __init__(self,processID,appName=None):
super(AppModule,self).__init__()
#: The ID of the process this appModule is for.
#: @type: int
self.processID=processID
if appName is None:
appName=getAppNameFromProcessID(processID)
#: The application name.
#: @type: str
self.appName=appName
self.processHandle=winKernel.openProcess(winKernel.SYNCHRONIZE|winKernel.PROCESS_QUERY_INFORMATION,False,processID)
self.helperLocalBindingHandle=None
self._inprocRegistrationHandle=None
def _setProductInfo(self):
"""Set productName and productVersion attributes.
"""
# Sometimes (I.E. when NVDA starts) handle is 0, so stop if it is the case
if not self.processHandle:
raise RuntimeError("processHandle is 0")
# Create the buffer to get the executable name
exeFileName = ctypes.create_unicode_buffer(ctypes.wintypes.MAX_PATH)
length = ctypes.wintypes.DWORD(ctypes.wintypes.MAX_PATH)
if not ctypes.windll.Kernel32.QueryFullProcessImageNameW(self.processHandle, 0, exeFileName, ctypes.byref(length)):
raise ctypes.WinError()
fileName = exeFileName.value
# Get size needed for buffer (0 if no info)
size = ctypes.windll.version.GetFileVersionInfoSizeW(fileName, None)
if not size:
raise RuntimeError("No version information")
# Create buffer
res = ctypes.create_string_buffer(size)
# Load file informations into buffer res
ctypes.windll.version.GetFileVersionInfoW(fileName, None, size, res)
r = ctypes.c_uint()
l = ctypes.c_uint()
# Look for codepages
ctypes.windll.version.VerQueryValueW(res, u'\\VarFileInfo\\Translation',
ctypes.byref(r), ctypes.byref(l))
if not l.value:
raise RuntimeError("No codepage")
# Take the first codepage (what else ?)
codepage = array.array('H', ctypes.string_at(r.value, 4))
codepage = "%04x%04x" % tuple(codepage)
# Extract product name and put it to self.productName
ctypes.windll.version.VerQueryValueW(res,
u'\\StringFileInfo\\%s\\ProductName' % codepage,
ctypes.byref(r), ctypes.byref(l))
self.productName = ctypes.wstring_at(r.value, l.value-1)
# Extract product version and put it to self.productVersion
ctypes.windll.version.VerQueryValueW(res,
u'\\StringFileInfo\\%s\\ProductVersion' % codepage,
ctypes.byref(r), ctypes.byref(l))
self.productVersion = ctypes.wstring_at(r.value, l.value-1)
def _get_productName(self):
self._setProductInfo()
return self.productName
def _get_productVersion(self):
self._setProductInfo()
return self.productVersion
def __repr__(self):
return "<%r (appName %r, process ID %s) at address %x>"%(self.appModuleName,self.appName,self.processID,id(self))
def _get_appModuleName(self):
return self.__class__.__module__.split('.')[-1]
def _get_isAlive(self):
return bool(winKernel.waitForSingleObject(self.processHandle,0))
def terminate(self):
"""Terminate this app module.
This is called to perform any clean up when this app module is being destroyed.
Subclasses should call the superclass method first.
"""
winKernel.closeHandle(self.processHandle)
if getattr(self, "_helperPreventDisconnect", False):
return
if self._inprocRegistrationHandle:
ctypes.windll.rpcrt4.RpcSsDestroyClientContext(ctypes.byref(self._inprocRegistrationHandle))
if self.helperLocalBindingHandle:
ctypes.windll.rpcrt4.RpcBindingFree(ctypes.byref(self.helperLocalBindingHandle))
def chooseNVDAObjectOverlayClasses(self, obj, clsList):
"""Choose NVDAObject overlay classes for a given NVDAObject.
This is called when an NVDAObject is being instantiated after L{NVDAObjects.NVDAObject.findOverlayClasses} has been called on the API-level class.
This allows an AppModule to add or remove overlay classes.
See L{NVDAObjects.NVDAObject.findOverlayClasses} for details about overlay classes.
@param obj: The object being created.
@type obj: L{NVDAObjects.NVDAObject}
@param clsList: The list of classes, which will be modified by this method if appropriate.
@type clsList: list of L{NVDAObjects.NVDAObject}
"""
# optimisation: Make it easy to detect that this hasn't been overridden.
chooseNVDAObjectOverlayClasses._isBase = True
def _get_is64BitProcess(self):
"""Whether the underlying process is a 64 bit process.
@rtype: bool
"""
if os.environ.get("PROCESSOR_ARCHITEW6432") not in ("AMD64","ARM64"):
# This is 32 bit Windows.
self.is64BitProcess = False
return False
res = ctypes.wintypes.BOOL()
if ctypes.windll.kernel32.IsWow64Process(self.processHandle, ctypes.byref(res)) == 0:
self.is64BitProcess = False
return False
self.is64BitProcess = not res
return self.is64BitProcess
def isGoodUIAWindow(self,hwnd):
"""
returns C{True} if the UIA implementation of the given window must be used, regardless whether native or not.
This function is the counterpart of and takes precedence over L{isBadUIAWindow}.
If both functions return C{False}, the decision of whether to use UIA for the window is left to core.
Warning: this may be called outside of NVDA's main thread, therefore do not try accessing NVDAObjects and such, rather just check window class names.
"""
return False
def isBadUIAWindow(self,hwnd):
"""
returns C{True} if the UIA implementation of the given window must be ignored due to it being broken in some way.
This function is the counterpart of L{isGoodUIAWindow}.
When both functions return C{True}, L{isGoodUIAWindow} takes precedence.
If both functions return C{False}, the decision of whether to use UIA for the window is left to core.
Warning: this may be called outside of NVDA's main thread, therefore do not try accessing NVDAObjects and such, rather just check window class names.
"""
return False
def dumpOnCrash(self):
"""Request that this process writes a minidump when it crashes for debugging.
This should only be called if instructed by a developer.
"""
path = os.path.join(tempfile.gettempdir(),
"nvda_crash_%s_%d.dmp" % (self.appName, self.processID)).decode("mbcs")
NVDAHelper.localLib.nvdaInProcUtils_dumpOnCrash(
self.helperLocalBindingHandle, path)
print "Dump path: %s" % path
class AppProfileTrigger(config.ProfileTrigger):
"""A configuration profile trigger for when a particular application has focus.
"""
def __init__(self, appName):
self.spec = "app:%s" % appName
def getWmiProcessInfo(processId):
"""Retrieve the WMI Win32_Process class instance for a given process.
For details about the available properties, see
http://msdn.microsoft.com/en-us/library/aa394372%28v=vs.85%29.aspx
@param processId: The id of the process in question.
@type processId: int
@return: The WMI Win32_Process class instance.
@raise LookupError: If there was an error retrieving the instance.
"""
try:
wmi = comtypes.client.CoGetObject(r"winmgmts:root\cimv2", dynamic=True)
results = wmi.ExecQuery("select * from Win32_Process "
"where ProcessId = %d" % processId)
for result in results:
return result
except:
raise LookupError("Couldn't get process information using WMI")
raise LookupError("No such process")