使用python守护程序在OS X 10.7上获取活动的应用程序
作者:互联网
我正在尝试在python中构建守护程序,我想获取当前活动应用程序的名称.
对于守护进程,我使用的是Sander Marechal的code snipped
当我不将应用程序作为守护程序运行时,以下行在OS X 10.7上可以正常运行,尽管文档说“ activeApplication()”在10.6上已被弃用
activeAppName = str(NSWorkspace.sharedWorkspace().activeApplication()['NSApplicationName'])
但是,一旦我将应用程序作为守护程序运行,应用程序就会崩溃.
但是,只有当我这样做时,守护程序才不会崩溃
workspace = str(NSWorkspace.sharedWorkspace())
返回:
<NSWorkspace: 0x7ffe7cc013c0>
所以我的问题是:
>为什么它仅作为守护程序崩溃?
>如何通过python获取活动应用程序
OS X 10.7(也可与守护程序一起使用;-))?
我不明白该错误信息,但也许你们其中一个人知道:
Process: Python [7920]
Path: /System/Library/Frameworks/Python.framework/Versions/2.7/Resources/Python.app/Contents/MacOS/Python
Identifier: Python
Version: ??? (???)
Code Type: X86-64 (Native)
Parent Process: ??? [1]
Date/Time: 2012-02-29 23:35:25.202 +0100
OS Version: Mac OS X 10.7.3 (11D50b)
Report Version: 9
Interval Since Last Report: 818421 sec
Crashes Since Last Report: 21
Per-App Crashes Since Last Report: 15
Anonymous UUID: 05B412BD-4629-472B-964D-BE4A88B06DD1
Crashed Thread: 0 Dispatch queue: com.apple.main-thread
Exception Type: EXC_BAD_ACCESS (SIGSEGV)
Exception Codes: KERN_INVALID_ADDRESS at 0x0000000000000108
VM Regions Near 0x108:
-->
__TEXT 0000000102e90000-0000000102e91000 [ 4K] r-x/rwx SM=COW /System/Library/Frameworks/Python.framework/Versions/2.7/Resources/Python.app/Contents/MacOS/Python
Application Specific Information:
*** single-threaded process forked ***
objc[7918]: garbage collection is OFF
Thread 0 Crashed:: Dispatch queue: com.apple.main-thread
0 libdispatch.dylib 0x00007fff8ceb7ce9 _dispatch_wakeup + 108
1 libdispatch.dylib 0x00007fff8ceba876 _dispatch_resume_slow + 20
2 com.apple.CoreServices.CarbonCore 0x00007fff8d34f919 _ZL22connectToCoreServicesDv + 269
3 com.apple.CoreServices.CarbonCore 0x00007fff8d34f7d5 _ZL9getStatusv + 24
4 com.apple.CoreServices.CarbonCore 0x00007fff8d34f74f scCreateSystemServiceVersion + 50
5 com.apple.LaunchServices 0x00007fff90b5ace1 _ZL45SetupCoreApplicationServicesCommunicationPortv + 147
6 com.apple.LaunchServices 0x00007fff90b5b37a getProcessDispatchTable() + 19
7 com.apple.LaunchServices 0x00007fff90b56de0 LSClientSideSharedMemory::GetClientSideSharedMemory(LSSessionID, bool) + 158
8 com.apple.LaunchServices 0x00007fff90b6b152 _LSCopyFrontApplication + 42
9 com.apple.AppKit 0x00007fff899adc5d -[NSWorkspace activeApplication] + 26
10 libffi.dylib 0x00007fff91df2e7c ffi_call_unix64 + 76
11 libffi.dylib 0x00007fff91df3ae9 ffi_call + 728
12 _objc.so 0x00000001031c7d60 PyObjCFFI_Caller + 2272
13 _objc.so 0x00000001031dd169 0x1031ae000 + 192873
14 org.python.python 0x0000000102ea0d32 PyObject_Call + 97
15 org.python.python 0x0000000102f20f63 PyEval_EvalFrameEx + 14353
16 org.python.python 0x0000000102f23df7 0x102e99000 + 568823
17 org.python.python 0x0000000102f20e0a PyEval_EvalFrameEx + 14008
18 org.python.python 0x0000000102f23df7 0x102e99000 + 568823
19 org.python.python 0x0000000102f20e0a PyEval_EvalFrameEx + 14008
20 org.python.python 0x0000000102f23cd8 PyEval_EvalCodeEx + 1996
21 org.python.python 0x0000000102f23d4d PyEval_EvalCode + 54
22 org.python.python 0x0000000102f3b08f 0x102e99000 + 663695
23 org.python.python 0x0000000102f3b14f PyRun_FileExFlags + 157
24 org.python.python 0x0000000102f3c2a2 PyRun_SimpleFileExFlags + 392
25 org.python.python 0x0000000102f4c2af Py_Main + 2715
26 org.python.python 0x0000000102e90e88 0x102e90000 + 3720
解决方法:
我对此进行了一些测试,我认为您的问题可能是您守护该工具,然后尝试进行要求Windows服务可能不可用的呼叫的方式.这里的链接暗示了这种情况:http://grokbase.com/t/python/pythonmac-sig/08axst378p/appscript-and-launching-apps-from-background-only-python-processes
我首先使用守护程序脚本测试了这一理论,并通过AppleScript进行了osascript调用以找到活动的应用程序:
from subprocess import Popen, PIPE
cmd = """osascript \
-e 'tell application "System Events"' \
-e 'set app_name to name of the first process whose frontmost is true' \
-e 'end tell' """
v = Popen(cmd, shell=True, stdout=PIPE).stdout.read()
Popen是一种在子进程中启动系统命令并能够检查其返回码或读取其输出(或发送输入)http://docs.python.org/library/subprocess.html的方法. Osascript是用于调用Apple脚本的命令行工具.
对我来说,这是有效的,因为它启动了一个我认为可以访问窗口服务器的新子进程?
然后,我创建了一个启动的plist,而不使用您的守护程序脚本.这有效:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>com.company.test</string>
<key>Nice</key>
<integer>1</integer>
<key>OnDemand</key>
<false/>
<key>Program</key>
<string>/path/to/script.py</string>
</dict>
</plist>
启动的是OSX守护进程管理器,它似乎确实以对窗口服务器具有完全访问权限的方式启动程序.
对于script.py,我只是让它循环,将最前面的应用程序名称写入文件并休眠.
更新资料
既然您提到过不赞成使用pyobjc方法,并且您似乎喜欢applescript方法,那么我想我会采用Python方式使用appscript – the python bindings to apple script
from appscript import app, its
activeApp = app('System Events').processes[its.frontmost == True].first()
print activeApp
#result
app(u'/System/Library/CoreServices/System Events.app').application_processes[u'Terminal']
activeApp是代表最前端应用程序的对象,由系统事件应用程序报告.
标签:pyobjc,objective-c,daemon,osx-lion,python 来源: https://codeday.me/bug/20191101/1985029.html