背景:這段時間某位同事須要頻繁登陸個人帳號,而且要驗證手機驗證碼,每次都須要我手動將驗證碼轉發給他,以爲很是的麻煩,便想看一下有沒有什麼可以解放雙手的辦法,而後發現了
AppleScript
這麼一個很是好用的東西 :)html
AppleScript
,那麼一臺MAC電腦天然是不可少的(macOS Catalina 10.15.5)AppleScript是用來編寫運行於mac的腳本的,而且可以操做應用程序。蘋果官方也要求開發mac上的軟件須要留出可以共AppleScript可以操做的方法。利用AppleScript咱們能很是方便實現一些日常工做中重複工做的腳本化,提高工做效率,避免重複勞動。sql
AppleScript 的語法很是接近天然語言,幾乎沒有標點符號,語法不像其餘語言那樣嚴格chrome
-- 將百度網址賦值給變量url_str set url_str to "https://www.baidu.com"
-- 將百度網址賦值給變量url_str set url_str to "https://www.baidu.com" -- 通知 display notification "url set success"
-- 將百度網址賦值給變量url_str set url_str to "https://www.baidu.com" -- 將bing網址賦值給變量url_str2 set url_str2 to "https://www.bing.com" -- 通知 display notification "url set success" -- 操做應用 tell application "Google Chrome" -- 建立新窗口 set newWin to make new window -- 獲取新窗口tab set theTab to get active tab of newWin -- 操做tab tell theTab -- 設置tabURL set URL of theTab to url_str -- 訪問 go forward -- 延遲3秒 delay 3 --從新設置tabURL set URL of theTab to url_str2 go forward end tell end tell
蘋果官方會要求mac軟件的開發提供可以操做應用軟件的方法,能夠經過字典查詢可否由AppleScript操控軟件數據庫
Script Editer -> 打開詞典 -> 選擇想要操做的軟件vim
chrome dictionary.
安全
瞭解 AppleScript 的基礎語法是掌握 AppleScript 自動化的第一步,在對 AppleScript 有了必定的瞭解後,在往後發現本身有重複操做時,就能夠先想一想能夠如何經過 AppleScript 解決它app
回到正題,由於對AppleScript
有過必定了解,而且也作過簡單的測試,因此第一時間想到的就是能不能用自動話腳原本處理重複的短信轉發工做(在這以前還從沒用AppleScript解決過現實問題)iphone
首先須要的就是打開字典,由於Message自己就是蘋果開發的,因此確定會有豐富的API供咱們使用的ide
經過翻找,咱們可以基本定位咱們發送短信須要用到的send方法,send方法中還須要咱們明確一個buddy,繼續查看buddy學習
查看咱們的buddy,他是包含在一個service裏面的,繼續service ...
service 上層爲application包含,所以咱們能夠書寫咱們的腳本了
tell application "Messages" send "test" to buddy "+86199xxxx" of service "SMS" end tell
很是簡單的一段代碼,能夠測試發送了。在測試的過程當中發現須要有和該號碼發過消息的記錄(即須要打開有一個窗口)
在可以成功發送短信後,咱們距離解放雙手已經很近了,接下來咱們只須要在接收到驗證碼信息的時候,調用咱們的發送腳本將咱們收到的消息轉發出去就大功告成了:)
繼續查看咱們的字典,有一個message received
事件,看字面上的意思就能知足咱們的須要,就是他了
很簡單,咱們只須要獲取到咱們須要的消息,以及發送給咱們的人用來判斷(畢竟不能把全部的消息都轉發出去)
using terms from application "Messages" on message received theMessage from theBuddy -- 通知接收到消息 display notification "message" -- 將Message App可見 set visible of window "Messages" to true -- 將Message App置頂 set frontmost of window "Message" to true end message received end using terms from
事情到這裏就結束了嘛?哪有那麼簡單,正所謂「天之道,沒那麼容易讓你成功~」
通過不斷的測試,發現這個腳本根本就沒達到個人預期,沒有通知也沒有置頂,是我哪裏寫的有問題嘛?好嘛,知錯要改,要虛心學習。百度~,嗯,查不到,害,谷歌~(我的習慣,先百度,當發現查不到想要的信息或者百度上全是同一篇文章抄來抄去的時候才谷歌,畢竟谷歌是不被容許,我但是乖孩子)
不斷的變換關鍵詞及描述,用個人散裝英語終於找到了一些解決辦法
發現個人腳本寫的是沒有問題的,只是還少一些操做步驟。例如
copy the script into a new AppleScript Editor document and save it in ~/Library/Application Scripts/com.apple.iChat/ Then activate it by selecting it in Messages -> Preferences -> General -> AppleScript Handler:
以及
AppleScript file copied to ~/Library/Scripts/Messages by Messages, it started to work.
不一樣的路徑,可是好像都須要拷貝,因而分別考到目錄下,可是一直沒有找到Message -> Preference —> General(發現個人MAC好像和廣大網友不太同樣啊,買到了假貨了吧)總之仍是不成功,繼續google...
終於,原來Message.app Message Received Event Handler
在macOS High Sierra 10.13.4
版本被移除,而且開通了Business Chat
emmmmmmm.....,害~
事件的方式走不通了,只能換一種方式了。還能夠用什麼方式呢,因而我想到了一種可能,個人這些消息,哪怕時間再久遠都能查到,他是怎麼存放的呢。
OK,有思路就有方向,最後不斷的搜索發現是存在目錄/Users/herbert/Library/Messages
下 的chat.db
文件中
用sqlite打開看一下
sqlite3 /Users/herbert/Library/Messages/chat.db
查看錶結構及表內部分數據
.schema message select * from message order by ROWID desc limit 5;
最終根據梳理的表結構關係,及傳入我須要查詢的號碼整理SQL
-- 查詢一分鐘內來自指定號碼xxxxxx_number的短信 SELECT text, handle_id,date,datetime(date/1000000000 + strftime('%s','2001-01-01'), 'unixepoch', 'localtime') as date_utc FROM message T1 INNER JOIN chat_message_join T2 ON T1.ROWID=T2.message_id INNER JOIN chat T3 ON T2.chat_id=T3.ROWID AND T3.chat_identifier = 'xxxxxx_number' where datetime(date/1000000000 + strftime('%s','2001-01-01'), 'unixepoch', 'localtime') > datetime('now','localtime','-1 minute') ORDER BY T1.date;
在查詢數據的時候還發現了一個不常見的日期存儲(我以前沒有見過)一個18位的時間數字
處理方式:
datetime(date/1000000000 + 978307200,'unixepoch','localtime')
.
datetime(date/1000000000+ strftime('%s','2001-01-01'),'unixepoch','localtime')
咱們已經拿到咱們須要的數據了,接下來就只須要每一個一分鐘取一次數據,若是可以去到數據,那麼調用咱們發送短信的腳本就OK了
編寫腳本auto_forward.sh
#/bin/sh phone="+86xxxx" call_apple() { while read line do osascript /Users/herbert/Documents/AppleScript/ForwardMessage/SendMessage.scpt $line $phone done } export -f call_apple sqlite3 /Users/herbert/Library/Messages/chat.db '.read /Users/herbert/Documents/AppleScript/ForwardMessage/select.txt' | call_apple
將咱們的sql存在單獨的文件裏,方便修改select.txt
SELECT text FROM message T1 INNER JOIN chat_message_join T2 ON T1.ROWID=T2.message_id INNER JOIN chat T3 ON T2.chat_id=T3.ROWID AND T3.chat_identifier = 'from_number' where datetime(date/1000000000 + strftime('%s','2001-01-01'), 'unixepoch', 'localtime') > datetime('now','localtime','-1 minute') ORDER BY T1.date;
SendMessage.scpt
on run argv set msgContent to item 1 of argv set phone to item 2 of argv sendMsg(msgContent, phone) end run on sendMsg(msgContent, phone) tell application "Messages" send msgContent to buddy phone of service "SMS" end tell end sendMsg
萬事具有,只須要加入crontab中就好了
crontab -e */1 * * * * sh /Users/herbert/Documents/AppleScript/ForwardMessage/auto_forward.sh
此處在測試的時候仍是沒有達到預期,可是單獨執行腳本的時候可以成功,推測是cron的問題,因而查看日誌
vim /var/mail/herbert
果真發現問題
sh: /Users/herbert/Documents/test/demo.sh: Operation not permitted
此問題會在Mojave 10.14, Catalina 10.15 以及後續版本出現 解決方法:系統偏好設置 -> 安全性與隱私 -> 隱私 -> 徹底磁盤訪問權限 -> 添加cron