diff options
-rw-r--r-- | Content/posts/2023-02-08-Interact-with-siri-from-the-terminal.md | 229 | ||||
-rw-r--r-- | Resources/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png | bin | 0 -> 31881 bytes | |||
-rw-r--r-- | docs/feed.rss | 249 | ||||
-rw-r--r-- | docs/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png | bin | 0 -> 31881 bytes | |||
-rw-r--r-- | docs/index.html | 21 | ||||
-rw-r--r-- | docs/posts/2023-02-08-Interact-with-siri-from-the-terminal.html | 300 | ||||
-rw-r--r-- | docs/posts/index.html | 21 |
7 files changed, 818 insertions, 2 deletions
diff --git a/Content/posts/2023-02-08-Interact-with-siri-from-the-terminal.md b/Content/posts/2023-02-08-Interact-with-siri-from-the-terminal.md new file mode 100644 index 0000000..78de77c --- /dev/null +++ b/Content/posts/2023-02-08-Interact-with-siri-from-the-terminal.md @@ -0,0 +1,229 @@ +--- +date: 2023-02-08 17:21 +description: Code snippet to interact with Siri by issuing commands from the command-line. +tags: Tutorial, Code-Snippet, Python, Siri, macOS, AppleScript +--- + +# Interacting with Siri using the command line + +My main objective was to see if I could issue multi-intent commands in one go. Obviously, Siri cannot do that (neither can Alexa, Cortana, or Google Assistant). The script here can issue either a single command, or use the help of OpenAI's DaVinci model to extract multiple commands and pass them onto siri. + +## Prerequisites + +* Run macOS +* Enable type to Siri (Settings > Accessibility -> Type to Siri) +* Enable the Terminal to control System Events (The first time you run the script, it will prompt you to enable it) + +## Show me ze code + +If you are here just for the code: + +```python +import argparse +import applescript +import openai + +from os import getenv + +openai.api_key = getenv("OPENAI_KEY") +engine = "text-davinci-003" + +def execute_with_llm(command_text: str) -> None: + llm_prompt = f"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command. + + Example: + Q: "Turn on the lights and turn off the lights" + A: ["Turn on the lights", "Turn off the lights"] + + Q: "Switch off the lights and then play some music" + A: ["Switch off the lights", "Play some music"] + + Q: "I am feeling sad today, play some music" + A: ["Play some cheerful music"] + + Q: "{command_text}" + A: + """ + + completion = openai.Completion.create(engine=engine, prompt=llm_prompt, max_tokens=len(command_text.split(" "))*2) + + for task in eval(completion.choices[0].text): + execute_command(task) + + +def execute_command(command_text: str) -> None: + """Execute a Siri command.""" + + script = applescript.AppleScript(f""" + tell application "System Events" to tell the front menu bar of process "SystemUIServer" + tell (first menu bar item whose description is "Siri") + perform action "AXPress" + end tell + end tell + + delay 2 + + tell application "System Events" + set textToType to "{command_text}" + keystroke textToType + key code 36 + end tell + """) + + script.run() + + +if __name__ == "__main__": + parser = argparse.ArgumentParser() + parser.add_argument("command", nargs="?", type=str, help="The command to pass to Siri", default="What time is it?") + parser.add_argument('--openai', action=argparse.BooleanOptionalAction, help="Use OpenAI to detect multiple intents", default=False) + args = parser.parse_args() + + if args.openai: + execute_with_llm(args.command) + else: + execute_command(args.command) +``` + +Usage: + +```bash +python3 main.py "play some taylor swift" +python3 main.py "turn off the lights and play some music" --openai +``` + +## ELI5 + +I am not actually going to explain it as if I am explaining to a five-year old kid. + +### AppleScript + +In the age of Siri Shortcuts, AppleScript can still do more. It is a scripting language created by Apple that can help you automate pretty much anything you see on your screen. + +We use the following AppleScript to trigger Siri and then type in our command: + +```applescript +tell application "System Events" to tell the front menu bar of process "SystemUIServer" + tell (first menu bar item whose description is "Siri") + perform action "AXPress" + end tell +end tell + +delay 2 + +tell application "System Events" + set textToType to "Play some rock music" + keystroke textToType + key code 36 +end tell +``` + +This first triggers Siri, waits for a couple of seconds, and then types in our command. In the script, this functionality is handled by the `execute_command` function. + +```python +import applescript + +def execute_command(command_text: str) -> None: + """Execute a Siri command.""" + + script = applescript.AppleScript(f""" + tell application "System Events" to tell the front menu bar of process "SystemUIServer" + tell (first menu bar item whose description is "Siri") + perform action "AXPress" + end tell + end tell + + delay 2 + + tell application "System Events" + set textToType to "{command_text}" + keystroke textToType + key code 36 + end tell + """) + + script.run() +``` + +### Multi-Intent Commands + +We can call OpenAI's API to autocomplete our prompt and extract multiple commands. We don't need to use OpenAI's API, and can also simply use Google's Flan-T5 model using HuggingFace's transformers library. + +#### Ze Prompt + +```text +You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command. + + Example: + Q: "Turn on the lights and turn off the lights" + A: ["Turn on the lights", "Turn off the lights"] + + Q: "Switch off the lights and then play some music" + A: ["Switch off the lights", "Play some music"] + + Q: "I am feeling sad today, play some music" + A: ["Play some cheerful music"] + + Q: "{command_text}" + A: +``` + +This prompt gives the model a few examples to increase the generation accuracy, along with instructing it to return a Python list. + + +#### Ze Code + +```python +import openai + +from os import getenv + +openai.api_key = getenv("OPENAI_KEY") +engine = "text-davinci-003" + +def execute_with_llm(command_text: str) -> None: + llm_prompt = f"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command. + + Example: + Q: "Turn on the lights and turn off the lights" + A: ["Turn on the lights", "Turn off the lights"] + + Q: "Switch off the lights and then play some music" + A: ["Switch off the lights", "Play some music"] + + Q: "I am feeling sad today, play some music" + A: ["Play some cheerful music"] + + Q: "{command_text}" + A: + """ + + completion = openai.Completion.create(engine=engine, prompt=llm_prompt, max_tokens=len(command_text.split(" "))*2) + + for task in eval(completion.choices[0].text): # NEVER EVAL IN PROD RIGHT LIKE THIS + execute_command(task) +``` + + +### Gluing together code + +To finish it all off, we can use argparse to only send the input command to OpenAI when asked to do so. + +```python +import argparse + +if __name__ == "__main__": + parser = argparse.ArgumentParser() + parser.add_argument("command", nargs="?", type=str, help="The command to pass to Siri", default="What time is it?") + parser.add_argument('--openai', action=argparse.BooleanOptionalAction, help="Use OpenAI to detect multiple intents", default=False) + args = parser.parse_args() + + if args.openai: + execute_with_llm(args.command) + else: + execute_command(args.command) +``` + +## Conclusion + +Siri is still dumb. When I ask it to `Switch off the lights`, it default to the home thousands of miles away. But, this code snippet definitely does work!
\ No newline at end of file diff --git a/Resources/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png b/Resources/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png Binary files differnew file mode 100644 index 0000000..54c04a4 --- /dev/null +++ b/Resources/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png diff --git a/docs/feed.rss b/docs/feed.rss index d81a885..22944f0 100644 --- a/docs/feed.rss +++ b/docs/feed.rss @@ -4,8 +4,8 @@ <title>Navan's Archive</title> <description>Rare Tips, Tricks and Posts</description> <link>https://web.navan.dev/</link><language>en</language> - <lastBuildDate>Wed, 01 Feb 2023 12:29:22 -0000</lastBuildDate> - <pubDate>Wed, 01 Feb 2023 12:29:22 -0000</pubDate> + <lastBuildDate>Wed, 08 Feb 2023 17:38:18 -0000</lastBuildDate> + <pubDate>Wed, 08 Feb 2023 17:38:18 -0000</pubDate> <ttl>250</ttl> <atom:link href="https://web.navan.dev/feed.rss" rel="self" type="application/rss+xml"/> @@ -4270,6 +4270,251 @@ return path(str, boost::filesystem::native); <item> <guid isPermaLink="true"> + https://web.navan.dev/posts/2023-02-08-Interact-with-siri-from-the-terminal.html + </guid> + <title> + Interacting with Siri using the command line + </title> + <description> + Code snippet to interact with Siri by issuing commands from the command-line. + </description> + <link>https://web.navan.dev/posts/2023-02-08-Interact-with-siri-from-the-terminal.html</link> + <pubDate>Wed, 08 Feb 2023 17:21:00 -0000</pubDate> + <content:encoded><![CDATA[<h1>Interacting with Siri using the command line</h1> + +<p>My main objective was to see if I could issue multi-intent commands in one go. Obviously, Siri cannot do that (neither can Alexa, Cortana, or Google Assistant). The script here can issue either a single command, or use the help of OpenAI's DaVinci model to extract multiple commands and pass them onto siri.</p> + +<h2>Prerequisites</h2> + +<ul> +<li>Run macOS</li> +<li>Enable type to Siri (Settings > Accessibility -> Type to Siri)</li> +<li>Enable the Terminal to control System Events (The first time you run the script, it will prompt you to enable it)</li> +</ul> + +<h2>Show me ze code</h2> + +<p>If you are here just for the code:</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">argparse</span> +<span class="kn">import</span> <span class="nn">applescript</span> +<span class="kn">import</span> <span class="nn">openai</span> + +<span class="kn">from</span> <span class="nn">os</span> <span class="kn">import</span> <span class="n">getenv</span> + +<span class="n">openai</span><span class="o">.</span><span class="n">api_key</span> <span class="o">=</span> <span class="n">getenv</span><span class="p">(</span><span class="s2">"OPENAI_KEY"</span><span class="p">)</span> +<span class="n">engine</span> <span class="o">=</span> <span class="s2">"text-davinci-003"</span> + +<span class="k">def</span> <span class="nf">execute_with_llm</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="n">llm_prompt</span> <span class="o">=</span> <span class="sa">f</span><span class="s2">"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command.</span> + +<span class="s2"> Example:</span> +<span class="s2"> Q: "Turn on the lights and turn off the lights"</span> +<span class="s2"> A: ["Turn on the lights", "Turn off the lights"]</span> + +<span class="s2"> Q: "Switch off the lights and then play some music"</span> +<span class="s2"> A: ["Switch off the lights", "Play some music"]</span> + +<span class="s2"> Q: "I am feeling sad today, play some music"</span> +<span class="s2"> A: ["Play some cheerful music"]</span> + +<span class="s2"> Q: "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> A: </span> +<span class="s2"> """</span> + + <span class="n">completion</span> <span class="o">=</span> <span class="n">openai</span><span class="o">.</span><span class="n">Completion</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">engine</span><span class="o">=</span><span class="n">engine</span><span class="p">,</span> <span class="n">prompt</span><span class="o">=</span><span class="n">llm_prompt</span><span class="p">,</span> <span class="n">max_tokens</span><span class="o">=</span><span class="nb">len</span><span class="p">(</span><span class="n">command_text</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s2">" "</span><span class="p">))</span><span class="o">*</span><span class="mi">2</span><span class="p">)</span> + + <span class="k">for</span> <span class="n">task</span> <span class="ow">in</span> <span class="nb">eval</span><span class="p">(</span><span class="n">completion</span><span class="o">.</span><span class="n">choices</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span><span class="o">.</span><span class="n">text</span><span class="p">):</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">task</span><span class="p">)</span> + + +<span class="k">def</span> <span class="nf">execute_command</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="sd">"""Execute a Siri command."""</span> + + <span class="n">script</span> <span class="o">=</span> <span class="n">applescript</span><span class="o">.</span><span class="n">AppleScript</span><span class="p">(</span><span class="sa">f</span><span class="s2">"""</span> +<span class="s2"> tell application "System Events" to tell the front menu bar of process "SystemUIServer"</span> +<span class="s2"> tell (first menu bar item whose description is "Siri")</span> +<span class="s2"> perform action "AXPress"</span> +<span class="s2"> end tell</span> +<span class="s2"> end tell</span> + +<span class="s2"> delay 2</span> + +<span class="s2"> tell application "System Events"</span> +<span class="s2"> set textToType to "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> keystroke textToType</span> +<span class="s2"> key code 36</span> +<span class="s2"> end tell</span> +<span class="s2"> """</span><span class="p">)</span> + + <span class="n">script</span><span class="o">.</span><span class="n">run</span><span class="p">()</span> + + +<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span> + <span class="n">parser</span> <span class="o">=</span> <span class="n">argparse</span><span class="o">.</span><span class="n">ArgumentParser</span><span class="p">()</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s2">"command"</span><span class="p">,</span> <span class="n">nargs</span><span class="o">=</span><span class="s2">"?"</span><span class="p">,</span> <span class="nb">type</span><span class="o">=</span><span class="nb">str</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"The command to pass to Siri"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="s2">"What time is it?"</span><span class="p">)</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s1">'--openai'</span><span class="p">,</span> <span class="n">action</span><span class="o">=</span><span class="n">argparse</span><span class="o">.</span><span class="n">BooleanOptionalAction</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"Use OpenAI to detect multiple intents"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="kc">False</span><span class="p">)</span> + <span class="n">args</span> <span class="o">=</span> <span class="n">parser</span><span class="o">.</span><span class="n">parse_args</span><span class="p">()</span> + + <span class="k">if</span> <span class="n">args</span><span class="o">.</span><span class="n">openai</span><span class="p">:</span> + <span class="n">execute_with_llm</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> + <span class="k">else</span><span class="p">:</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> +</code></pre> +</div> + +<p>Usage:</p> + +<div class="codehilite"> +<pre><span></span><code>python3 main.py <span class="s2">"play some taylor swift"</span> +python3 main.py <span class="s2">"turn off the lights and play some music"</span> --openai +</code></pre> +</div> + +<h2>ELI5</h2> + +<p>I am not actually going to explain it as if I am explaining to a five-year old kid.</p> + +<h3>AppleScript</h3> + +<p>In the age of Siri Shortcuts, AppleScript can still do more. It is a scripting language created by Apple that can help you automate pretty much anything you see on your screen.</p> + +<p>We use the following AppleScript to trigger Siri and then type in our command:</p> + +<div class="codehilite"> +<pre><span></span><code><span class="k">tell</span> <span class="nb">application</span> <span class="s2">"System Events"</span> <span class="k">to</span> <span class="k">tell</span> <span class="nb">the</span> <span class="nb">front</span> <span class="na">menu</span> <span class="nv">bar</span> <span class="k">of</span> <span class="nv">process</span> <span class="s2">"SystemUIServer"</span> + <span class="k">tell</span> <span class="p">(</span><span class="nb">first</span> <span class="na">menu</span> <span class="nv">bar</span> <span class="nb">item</span> <span class="nb">whose</span> <span class="nv">description</span> <span class="ow">is</span> <span class="s2">"Siri"</span><span class="p">)</span> + <span class="nb">perform action</span> <span class="s2">"AXPress"</span> + <span class="k">end</span> <span class="k">tell</span> +<span class="k">end</span> <span class="k">tell</span> + +<span class="nb">delay</span> <span class="mi">2</span> + +<span class="k">tell</span> <span class="nb">application</span> <span class="s2">"System Events"</span> + <span class="k">set</span> <span class="nv">textToType</span> <span class="k">to</span> <span class="s2">"Play some rock music"</span> + <span class="nv">keystroke</span> <span class="nv">textToType</span> + <span class="na">key code</span> <span class="mi">36</span> +<span class="k">end</span> <span class="k">tell</span> +</code></pre> +</div> + +<p>This first triggers Siri, waits for a couple of seconds, and then types in our command. In the script, this functionality is handled by the <code>execute_command</code> function.</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">applescript</span> + +<span class="k">def</span> <span class="nf">execute_command</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="sd">"""Execute a Siri command."""</span> + + <span class="n">script</span> <span class="o">=</span> <span class="n">applescript</span><span class="o">.</span><span class="n">AppleScript</span><span class="p">(</span><span class="sa">f</span><span class="s2">"""</span> +<span class="s2"> tell application "System Events" to tell the front menu bar of process "SystemUIServer"</span> +<span class="s2"> tell (first menu bar item whose description is "Siri")</span> +<span class="s2"> perform action "AXPress"</span> +<span class="s2"> end tell</span> +<span class="s2"> end tell</span> + +<span class="s2"> delay 2</span> + +<span class="s2"> tell application "System Events"</span> +<span class="s2"> set textToType to "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> keystroke textToType</span> +<span class="s2"> key code 36</span> +<span class="s2"> end tell</span> +<span class="s2"> """</span><span class="p">)</span> + + <span class="n">script</span><span class="o">.</span><span class="n">run</span><span class="p">()</span> +</code></pre> +</div> + +<h3>Multi-Intent Commands</h3> + +<p>We can call OpenAI's API to autocomplete our prompt and extract multiple commands. We don't need to use OpenAI's API, and can also simply use Google's Flan-T5 model using HuggingFace's transformers library. </p> + +<h4>Ze Prompt</h4> + +<div class="codehilite"> +<pre><span></span><code>You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command. + + Example: + Q: "Turn on the lights and turn off the lights" + A: ["Turn on the lights", "Turn off the lights"] + + Q: "Switch off the lights and then play some music" + A: ["Switch off the lights", "Play some music"] + + Q: "I am feeling sad today, play some music" + A: ["Play some cheerful music"] + + Q: "{command_text}" + A: +</code></pre> +</div> + +<p>This prompt gives the model a few examples to increase the generation accuracy, along with instructing it to return a Python list. </p> + +<h4>Ze Code</h4> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">openai</span> + +<span class="kn">from</span> <span class="nn">os</span> <span class="kn">import</span> <span class="n">getenv</span> + +<span class="n">openai</span><span class="o">.</span><span class="n">api_key</span> <span class="o">=</span> <span class="n">getenv</span><span class="p">(</span><span class="s2">"OPENAI_KEY"</span><span class="p">)</span> +<span class="n">engine</span> <span class="o">=</span> <span class="s2">"text-davinci-003"</span> + +<span class="k">def</span> <span class="nf">execute_with_llm</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="n">llm_prompt</span> <span class="o">=</span> <span class="sa">f</span><span class="s2">"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command.</span> + +<span class="s2"> Example:</span> +<span class="s2"> Q: "Turn on the lights and turn off the lights"</span> +<span class="s2"> A: ["Turn on the lights", "Turn off the lights"]</span> + +<span class="s2"> Q: "Switch off the lights and then play some music"</span> +<span class="s2"> A: ["Switch off the lights", "Play some music"]</span> + +<span class="s2"> Q: "I am feeling sad today, play some music"</span> +<span class="s2"> A: ["Play some cheerful music"]</span> + +<span class="s2"> Q: "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> A: </span> +<span class="s2"> """</span> + + <span class="n">completion</span> <span class="o">=</span> <span class="n">openai</span><span class="o">.</span><span class="n">Completion</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">engine</span><span class="o">=</span><span class="n">engine</span><span class="p">,</span> <span class="n">prompt</span><span class="o">=</span><span class="n">llm_prompt</span><span class="p">,</span> <span class="n">max_tokens</span><span class="o">=</span><span class="nb">len</span><span class="p">(</span><span class="n">command_text</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s2">" "</span><span class="p">))</span><span class="o">*</span><span class="mi">2</span><span class="p">)</span> + + <span class="k">for</span> <span class="n">task</span> <span class="ow">in</span> <span class="nb">eval</span><span class="p">(</span><span class="n">completion</span><span class="o">.</span><span class="n">choices</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span><span class="o">.</span><span class="n">text</span><span class="p">):</span> <span class="c1"># NEVER EVAL IN PROD RIGHT LIKE THIS</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">task</span><span class="p">)</span> +</code></pre> +</div> + +<h3>Gluing together code</h3> + +<p>To finish it all off, we can use argparse to only send the input command to OpenAI when asked to do so.</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">argparse</span> + +<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span> + <span class="n">parser</span> <span class="o">=</span> <span class="n">argparse</span><span class="o">.</span><span class="n">ArgumentParser</span><span class="p">()</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s2">"command"</span><span class="p">,</span> <span class="n">nargs</span><span class="o">=</span><span class="s2">"?"</span><span class="p">,</span> <span class="nb">type</span><span class="o">=</span><span class="nb">str</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"The command to pass to Siri"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="s2">"What time is it?"</span><span class="p">)</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s1">'--openai'</span><span class="p">,</span> <span class="n">action</span><span class="o">=</span><span class="n">argparse</span><span class="o">.</span><span class="n">BooleanOptionalAction</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"Use OpenAI to detect multiple intents"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="kc">False</span><span class="p">)</span> + <span class="n">args</span> <span class="o">=</span> <span class="n">parser</span><span class="o">.</span><span class="n">parse_args</span><span class="p">()</span> + + <span class="k">if</span> <span class="n">args</span><span class="o">.</span><span class="n">openai</span><span class="p">:</span> + <span class="n">execute_with_llm</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> + <span class="k">else</span><span class="p">:</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> +</code></pre> +</div> + +<h2>Conclusion</h2> + +<p>Siri is still dumb. When I ask it to <code>Switch off the lights</code>, it default to the home thousands of miles away. But, this code snippet definitely does work!</p> +]]></content:encoded> + </item> + + <item> + <guid isPermaLink="true"> https://web.navan.dev/posts/2020-01-16-Image-Classifier-Using-Turicreate.html </guid> <title> diff --git a/docs/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png b/docs/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png Binary files differnew file mode 100644 index 0000000..54c04a4 --- /dev/null +++ b/docs/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png diff --git a/docs/index.html b/docs/index.html index 335b013..c414507 100644 --- a/docs/index.html +++ b/docs/index.html @@ -59,6 +59,27 @@ <ul> + <li><a href="/posts/2023-02-08-Interact-with-siri-from-the-terminal.html">Interacting with Siri using the command line</a></li> + <ul> + <li>Code snippet to interact with Siri by issuing commands from the command-line.</li> + <li>Published On: 2023-02-08 17:21</li> + <li>Tags: + + Tutorial, + + Code-Snippet, + + Python, + + Siri, + + macOS, + + AppleScript + + </ul> + + <li><a href="/posts/2022-12-25-blog-to-toot.html">Posting blogs as Mastodon Toots</a></li> <ul> <li>Cross posting blog posts to Mastodon</li> diff --git a/docs/posts/2023-02-08-Interact-with-siri-from-the-terminal.html b/docs/posts/2023-02-08-Interact-with-siri-from-the-terminal.html new file mode 100644 index 0000000..2db536b --- /dev/null +++ b/docs/posts/2023-02-08-Interact-with-siri-from-the-terminal.html @@ -0,0 +1,300 @@ +<!DOCTYPE html> +<html lang="en"> +<head> + + <link rel="stylesheet" href="/assets/main.css" /> + <link rel="stylesheet" href="/assets/sakura.css" /> + <meta charset="utf-8"> + <meta name="viewport" content="width=device-width, initial-scale=1.0"> + <title>Interacting with Siri using the command line</title> + <meta name="og:site_name" content="Navan Chauhan" /> + <link rel="canonical" href="https://web.navan.dev/" /> + <meta name="twitter:url" content="https://web.navan.dev/" /> + <meta name="og:url" content="https://web.navan.dev/" /> + <meta name="twitter:title" content="Interacting with Siri using the command line" /> + <meta name="og:title" content="Interacting with Siri using the command line" /> + <meta name="description" content="Code snippet to interact with Siri by issuing commands from the command-line." /> + <meta name="twitter:description" content="Code snippet to interact with Siri by issuing commands from the command-line." /> + <meta name="og:description" content="Code snippet to interact with Siri by issuing commands from the command-line." /> + <meta name="twitter:card" content="summary_large_image" /> + <meta name="viewport" content="width=device-width, initial-scale=1.0" /> + <link rel="shortcut icon" href="/images/favicon.png" type="image/png" /> + <link rel="alternate" href="/feed.rss" type="application/rss+xml" title="Subscribe to Navan Chauhan" /> + <meta name="twitter:image" content="https://web.navan.dev/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png" /> + <meta name="og:image" content="https://web.navan.dev/images/opengraph/posts/2023-02-08-Interact-with-siri-from-the-terminal.png" /> + <link rel="manifest" href="manifest.json" /> + <meta name="google-site-verification" content="LVeSZxz-QskhbEjHxOi7-BM5dDxTg53x2TwrjFxfL0k" /> + <script data-goatcounter="https://navanchauhan.goatcounter.com/count" + async src="//gc.zgo.at/count.js"></script> + <script defer data-domain="web.navan.dev" src="https://plausible.io/js/plausible.js"></script> + <script defer data-domain="web.navan.dev" src="https://plausible.navan.dev/js/plausible.js"></script> + <!-- Begin Inspectlet Asynchronous Code. Only for some testing, will be removed soon --> + <script type="text/javascript"> + (function() { + window.__insp = window.__insp || []; + __insp.push(['wid', 1038401947]); + var ldinsp = function(){ + if(typeof window.__inspld != "undefined") return; window.__inspld = 1; var insp = document.createElement('script'); insp.type = 'text/javascript'; insp.async = true; insp.id = "inspsync"; insp.src = ('https:' == document.location.protocol ? 'https' : 'http') + '://cdn.inspectlet.com/inspectlet.js?wid=1038401947&r=' + Math.floor(new Date().getTime()/3600000); var x = document.getElementsByTagName('script')[0]; x.parentNode.insertBefore(insp, x); }; + setTimeout(ldinsp, 0); + })(); + </script> + <!-- End Inspectlet Asynchronous Code --> + +</head> +<body> + <nav style="display: block;"> +| +<a href="/">home</a> | +<a href="/about/">about/links</a> | +<a href="/posts/">posts</a> | +<a href="/publications/">publications</a> | +<a href="/repo/">iOS repo</a> | +<a href="/feed.rss">RSS Feed</a> | +</nav> + +<main> + + <h1>Interacting with Siri using the command line</h1> + +<p>My main objective was to see if I could issue multi-intent commands in one go. Obviously, Siri cannot do that (neither can Alexa, Cortana, or Google Assistant). The script here can issue either a single command, or use the help of OpenAI's DaVinci model to extract multiple commands and pass them onto siri.</p> + +<h2>Prerequisites</h2> + +<ul> +<li>Run macOS</li> +<li>Enable type to Siri (Settings > Accessibility -> Type to Siri)</li> +<li>Enable the Terminal to control System Events (The first time you run the script, it will prompt you to enable it)</li> +</ul> + +<h2>Show me ze code</h2> + +<p>If you are here just for the code:</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">argparse</span> +<span class="kn">import</span> <span class="nn">applescript</span> +<span class="kn">import</span> <span class="nn">openai</span> + +<span class="kn">from</span> <span class="nn">os</span> <span class="kn">import</span> <span class="n">getenv</span> + +<span class="n">openai</span><span class="o">.</span><span class="n">api_key</span> <span class="o">=</span> <span class="n">getenv</span><span class="p">(</span><span class="s2">"OPENAI_KEY"</span><span class="p">)</span> +<span class="n">engine</span> <span class="o">=</span> <span class="s2">"text-davinci-003"</span> + +<span class="k">def</span> <span class="nf">execute_with_llm</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="n">llm_prompt</span> <span class="o">=</span> <span class="sa">f</span><span class="s2">"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command.</span> + +<span class="s2"> Example:</span> +<span class="s2"> Q: "Turn on the lights and turn off the lights"</span> +<span class="s2"> A: ["Turn on the lights", "Turn off the lights"]</span> + +<span class="s2"> Q: "Switch off the lights and then play some music"</span> +<span class="s2"> A: ["Switch off the lights", "Play some music"]</span> + +<span class="s2"> Q: "I am feeling sad today, play some music"</span> +<span class="s2"> A: ["Play some cheerful music"]</span> + +<span class="s2"> Q: "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> A: </span> +<span class="s2"> """</span> + + <span class="n">completion</span> <span class="o">=</span> <span class="n">openai</span><span class="o">.</span><span class="n">Completion</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">engine</span><span class="o">=</span><span class="n">engine</span><span class="p">,</span> <span class="n">prompt</span><span class="o">=</span><span class="n">llm_prompt</span><span class="p">,</span> <span class="n">max_tokens</span><span class="o">=</span><span class="nb">len</span><span class="p">(</span><span class="n">command_text</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s2">" "</span><span class="p">))</span><span class="o">*</span><span class="mi">2</span><span class="p">)</span> + + <span class="k">for</span> <span class="n">task</span> <span class="ow">in</span> <span class="nb">eval</span><span class="p">(</span><span class="n">completion</span><span class="o">.</span><span class="n">choices</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span><span class="o">.</span><span class="n">text</span><span class="p">):</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">task</span><span class="p">)</span> + + +<span class="k">def</span> <span class="nf">execute_command</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="sd">"""Execute a Siri command."""</span> + + <span class="n">script</span> <span class="o">=</span> <span class="n">applescript</span><span class="o">.</span><span class="n">AppleScript</span><span class="p">(</span><span class="sa">f</span><span class="s2">"""</span> +<span class="s2"> tell application "System Events" to tell the front menu bar of process "SystemUIServer"</span> +<span class="s2"> tell (first menu bar item whose description is "Siri")</span> +<span class="s2"> perform action "AXPress"</span> +<span class="s2"> end tell</span> +<span class="s2"> end tell</span> + +<span class="s2"> delay 2</span> + +<span class="s2"> tell application "System Events"</span> +<span class="s2"> set textToType to "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> keystroke textToType</span> +<span class="s2"> key code 36</span> +<span class="s2"> end tell</span> +<span class="s2"> """</span><span class="p">)</span> + + <span class="n">script</span><span class="o">.</span><span class="n">run</span><span class="p">()</span> + + +<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span> + <span class="n">parser</span> <span class="o">=</span> <span class="n">argparse</span><span class="o">.</span><span class="n">ArgumentParser</span><span class="p">()</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s2">"command"</span><span class="p">,</span> <span class="n">nargs</span><span class="o">=</span><span class="s2">"?"</span><span class="p">,</span> <span class="nb">type</span><span class="o">=</span><span class="nb">str</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"The command to pass to Siri"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="s2">"What time is it?"</span><span class="p">)</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s1">'--openai'</span><span class="p">,</span> <span class="n">action</span><span class="o">=</span><span class="n">argparse</span><span class="o">.</span><span class="n">BooleanOptionalAction</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"Use OpenAI to detect multiple intents"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="kc">False</span><span class="p">)</span> + <span class="n">args</span> <span class="o">=</span> <span class="n">parser</span><span class="o">.</span><span class="n">parse_args</span><span class="p">()</span> + + <span class="k">if</span> <span class="n">args</span><span class="o">.</span><span class="n">openai</span><span class="p">:</span> + <span class="n">execute_with_llm</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> + <span class="k">else</span><span class="p">:</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> +</code></pre> +</div> + +<p>Usage:</p> + +<div class="codehilite"> +<pre><span></span><code>python3 main.py <span class="s2">"play some taylor swift"</span> +python3 main.py <span class="s2">"turn off the lights and play some music"</span> --openai +</code></pre> +</div> + +<h2>ELI5</h2> + +<p>I am not actually going to explain it as if I am explaining to a five-year old kid.</p> + +<h3>AppleScript</h3> + +<p>In the age of Siri Shortcuts, AppleScript can still do more. It is a scripting language created by Apple that can help you automate pretty much anything you see on your screen.</p> + +<p>We use the following AppleScript to trigger Siri and then type in our command:</p> + +<div class="codehilite"> +<pre><span></span><code><span class="k">tell</span> <span class="nb">application</span> <span class="s2">"System Events"</span> <span class="k">to</span> <span class="k">tell</span> <span class="nb">the</span> <span class="nb">front</span> <span class="na">menu</span> <span class="nv">bar</span> <span class="k">of</span> <span class="nv">process</span> <span class="s2">"SystemUIServer"</span> + <span class="k">tell</span> <span class="p">(</span><span class="nb">first</span> <span class="na">menu</span> <span class="nv">bar</span> <span class="nb">item</span> <span class="nb">whose</span> <span class="nv">description</span> <span class="ow">is</span> <span class="s2">"Siri"</span><span class="p">)</span> + <span class="nb">perform action</span> <span class="s2">"AXPress"</span> + <span class="k">end</span> <span class="k">tell</span> +<span class="k">end</span> <span class="k">tell</span> + +<span class="nb">delay</span> <span class="mi">2</span> + +<span class="k">tell</span> <span class="nb">application</span> <span class="s2">"System Events"</span> + <span class="k">set</span> <span class="nv">textToType</span> <span class="k">to</span> <span class="s2">"Play some rock music"</span> + <span class="nv">keystroke</span> <span class="nv">textToType</span> + <span class="na">key code</span> <span class="mi">36</span> +<span class="k">end</span> <span class="k">tell</span> +</code></pre> +</div> + +<p>This first triggers Siri, waits for a couple of seconds, and then types in our command. In the script, this functionality is handled by the <code>execute_command</code> function.</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">applescript</span> + +<span class="k">def</span> <span class="nf">execute_command</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="sd">"""Execute a Siri command."""</span> + + <span class="n">script</span> <span class="o">=</span> <span class="n">applescript</span><span class="o">.</span><span class="n">AppleScript</span><span class="p">(</span><span class="sa">f</span><span class="s2">"""</span> +<span class="s2"> tell application "System Events" to tell the front menu bar of process "SystemUIServer"</span> +<span class="s2"> tell (first menu bar item whose description is "Siri")</span> +<span class="s2"> perform action "AXPress"</span> +<span class="s2"> end tell</span> +<span class="s2"> end tell</span> + +<span class="s2"> delay 2</span> + +<span class="s2"> tell application "System Events"</span> +<span class="s2"> set textToType to "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> keystroke textToType</span> +<span class="s2"> key code 36</span> +<span class="s2"> end tell</span> +<span class="s2"> """</span><span class="p">)</span> + + <span class="n">script</span><span class="o">.</span><span class="n">run</span><span class="p">()</span> +</code></pre> +</div> + +<h3>Multi-Intent Commands</h3> + +<p>We can call OpenAI's API to autocomplete our prompt and extract multiple commands. We don't need to use OpenAI's API, and can also simply use Google's Flan-T5 model using HuggingFace's transformers library. </p> + +<h4>Ze Prompt</h4> + +<div class="codehilite"> +<pre><span></span><code>You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command. + + Example: + Q: "Turn on the lights and turn off the lights" + A: ["Turn on the lights", "Turn off the lights"] + + Q: "Switch off the lights and then play some music" + A: ["Switch off the lights", "Play some music"] + + Q: "I am feeling sad today, play some music" + A: ["Play some cheerful music"] + + Q: "{command_text}" + A: +</code></pre> +</div> + +<p>This prompt gives the model a few examples to increase the generation accuracy, along with instructing it to return a Python list. </p> + +<h4>Ze Code</h4> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">openai</span> + +<span class="kn">from</span> <span class="nn">os</span> <span class="kn">import</span> <span class="n">getenv</span> + +<span class="n">openai</span><span class="o">.</span><span class="n">api_key</span> <span class="o">=</span> <span class="n">getenv</span><span class="p">(</span><span class="s2">"OPENAI_KEY"</span><span class="p">)</span> +<span class="n">engine</span> <span class="o">=</span> <span class="s2">"text-davinci-003"</span> + +<span class="k">def</span> <span class="nf">execute_with_llm</span><span class="p">(</span><span class="n">command_text</span><span class="p">:</span> <span class="nb">str</span><span class="p">)</span> <span class="o">-></span> <span class="kc">None</span><span class="p">:</span> + <span class="n">llm_prompt</span> <span class="o">=</span> <span class="sa">f</span><span class="s2">"""You are provided with multiple commands as a single command. Break down all the commands and return them in a list of strings. If you are provided with a single command, return a list with a single string, trying your best to understand the command.</span> + +<span class="s2"> Example:</span> +<span class="s2"> Q: "Turn on the lights and turn off the lights"</span> +<span class="s2"> A: ["Turn on the lights", "Turn off the lights"]</span> + +<span class="s2"> Q: "Switch off the lights and then play some music"</span> +<span class="s2"> A: ["Switch off the lights", "Play some music"]</span> + +<span class="s2"> Q: "I am feeling sad today, play some music"</span> +<span class="s2"> A: ["Play some cheerful music"]</span> + +<span class="s2"> Q: "</span><span class="si">{</span><span class="n">command_text</span><span class="si">}</span><span class="s2">"</span> +<span class="s2"> A: </span> +<span class="s2"> """</span> + + <span class="n">completion</span> <span class="o">=</span> <span class="n">openai</span><span class="o">.</span><span class="n">Completion</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">engine</span><span class="o">=</span><span class="n">engine</span><span class="p">,</span> <span class="n">prompt</span><span class="o">=</span><span class="n">llm_prompt</span><span class="p">,</span> <span class="n">max_tokens</span><span class="o">=</span><span class="nb">len</span><span class="p">(</span><span class="n">command_text</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s2">" "</span><span class="p">))</span><span class="o">*</span><span class="mi">2</span><span class="p">)</span> + + <span class="k">for</span> <span class="n">task</span> <span class="ow">in</span> <span class="nb">eval</span><span class="p">(</span><span class="n">completion</span><span class="o">.</span><span class="n">choices</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span><span class="o">.</span><span class="n">text</span><span class="p">):</span> <span class="c1"># NEVER EVAL IN PROD RIGHT LIKE THIS</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">task</span><span class="p">)</span> +</code></pre> +</div> + +<h3>Gluing together code</h3> + +<p>To finish it all off, we can use argparse to only send the input command to OpenAI when asked to do so.</p> + +<div class="codehilite"> +<pre><span></span><code><span class="kn">import</span> <span class="nn">argparse</span> + +<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span> + <span class="n">parser</span> <span class="o">=</span> <span class="n">argparse</span><span class="o">.</span><span class="n">ArgumentParser</span><span class="p">()</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s2">"command"</span><span class="p">,</span> <span class="n">nargs</span><span class="o">=</span><span class="s2">"?"</span><span class="p">,</span> <span class="nb">type</span><span class="o">=</span><span class="nb">str</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"The command to pass to Siri"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="s2">"What time is it?"</span><span class="p">)</span> + <span class="n">parser</span><span class="o">.</span><span class="n">add_argument</span><span class="p">(</span><span class="s1">'--openai'</span><span class="p">,</span> <span class="n">action</span><span class="o">=</span><span class="n">argparse</span><span class="o">.</span><span class="n">BooleanOptionalAction</span><span class="p">,</span> <span class="n">help</span><span class="o">=</span><span class="s2">"Use OpenAI to detect multiple intents"</span><span class="p">,</span> <span class="n">default</span><span class="o">=</span><span class="kc">False</span><span class="p">)</span> + <span class="n">args</span> <span class="o">=</span> <span class="n">parser</span><span class="o">.</span><span class="n">parse_args</span><span class="p">()</span> + + <span class="k">if</span> <span class="n">args</span><span class="o">.</span><span class="n">openai</span><span class="p">:</span> + <span class="n">execute_with_llm</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> + <span class="k">else</span><span class="p">:</span> + <span class="n">execute_command</span><span class="p">(</span><span class="n">args</span><span class="o">.</span><span class="n">command</span><span class="p">)</span> +</code></pre> +</div> + +<h2>Conclusion</h2> + +<p>Siri is still dumb. When I ask it to <code>Switch off the lights</code>, it default to the home thousands of miles away. But, this code snippet definitely does work!</p> + + <blockquote>If you have scrolled this far, consider subscribing to my mailing list <a href="https://listmonk.navan.dev/subscription/form">here.</a> You can subscribe to either a specific type of post you are interested in, or subscribe to everything with the "Everything" list.</blockquote> + <script data-isso="//comments.navan.dev/" + src="//comments.navan.dev/js/embed.min.js"></script> + <section id="isso-thread"> + <noscript>Javascript needs to be activated to view comments.</noscript> + </section> +</main> + + +<script src="assets/manup.min.js"></script> +<script src="/pwabuilder-sw-register.js"></script> +</body> +</html>
\ No newline at end of file diff --git a/docs/posts/index.html b/docs/posts/index.html index ace6ce7..5fed354 100644 --- a/docs/posts/index.html +++ b/docs/posts/index.html @@ -62,6 +62,27 @@ <ul> + <li><a href="/posts/2023-02-08-Interact-with-siri-from-the-terminal.html">Interacting with Siri using the command line</a></li> + <ul> + <li>Code snippet to interact with Siri by issuing commands from the command-line.</li> + <li>Published On: 2023-02-08 17:21</li> + <li>Tags: + + Tutorial, + + Code-Snippet, + + Python, + + Siri, + + macOS, + + AppleScript, + + </ul> + + <li><a href="/posts/2022-12-25-blog-to-toot.html">Posting blogs as Mastodon Toots</a></li> <ul> <li>Cross posting blog posts to Mastodon</li> |