Skip to content

Latest commit

 

History

History
182 lines (119 loc) · 9.69 KB

File metadata and controls

182 lines (119 loc) · 9.69 KB

Getting hands dirty with Clojure, GitHub Actions and IFTTT Webhooks for Personal Automation

As a software engineer, I cannot help but think of making my life more efficient. These are a few recent ideas I have had:

  • As my salary arrives, transfer it to different accounts based on a pre-set allocation
  • Turn the heater on or off depending on the room temperature
  • Update workout plans with new weights according to a progression plan
  • Send me tweets with more than 100 likes from last week from people I follow

When it comes to actually implementing these ideas, the choice is to either use existing automation tools like IFTTT or write my own scripts. Tools like IFTTT are great because they are easy to use and well-integrated with thousands of services but they lack the flexibility you can achieve with your own code - maybe you want to process data in a certain way or talk to an unsupported API?

On the other hand, writing code is not just about writing code but also about deploying, running and maintaining it. Plus, now you have to write code for every service yourself (even the ones supported by IFTTT).

Is there a way where we can get the best of both worlds?

  1. Make use of existing integrations
  2. Write custom code when necessary
  3. Painless code maintenance/management
  4. Free deployment

This article covers how we can achieve exactly that.

Email me transcripts of new TED Talks

For my usecase, I will be implementing this idea I had about turning new TED videos into text so I can read them instead of taking out the time to watch them. This is the plan:

  1. Watch out for new talks on their YouTube Channel
  2. Find the talk on the official website and grab the transcript
  3. Email it to myself

Note: We could also grab the transcript from the Youtube video itself but that requires setting up a Google Developer account. Talking to the TED website is much easier.

IFTTT has integrations which make #1 and #3 a piece of cake but nothing to help us with #2. This is a problem no-code approaches run into often - so how do we fill this gap?

Just Code It

I know right, just code whatever is missing? Exactly but then we run into two problems:

1. How do we plug it into IFTTT?

Well, all we need is some way of triggering our code when IFTTT is done and some way of letting IFTTT know when our code is done. That is exactly what their webhooks help with. Similar functionality exists in Zapier and Apple Shortcuts so we can mix our code with other tools too.

2. Where do we run it?

Writing code is just a piece of the puzzle. Maintaining, deploying and running it is where it gets trickier. Here are a few options:

In addition to costing money, the above options require even more code depending on whether you decide to build any of the following:

  • a server to receive triggers
  • authentication/authorization
  • continuous deployment

All this to say that there is a solution: GitHub Actions

If you haven't heard of it or similar solutions before, these are workflows that run in your git repository and can be triggered in all sorts of ways. Common usecases involve testing, building and deploying code but the possibilities are endless.

All we need to do is write a workflow which starts when https://api.github.com/***/workflows/{workflow-id} is hit. Inside the workflow, we can run our code and hit IFTTT when we are done.

For personal use, the usage limits are more than enough so we essentially have an easy way of running our code for free. 🥳

Back to TED

Let's get our hands dirty and actually code it out. We need a workflow url before we can ask IFTTT to hit it so let's get the code done first. For this part, you can use any language you want - I prefer Clojure as it's concise and a delight to use.

Feel free to go through the repository yourself but these are the major parts:

1. GitHub Actions Workflow to receive the trigger from IFTTT

I created a .github/workflows/ted-talk-transcript.yaml workflow that can be triggered manually from the UI or by hitting https://api.github.com/repos/tumblingpointers/003-clojure-cron-job-github-actions/actions/workflows/ted-talk-transcript.yaml/dispatches. It needs one input: talk_title.

name: Ted Talk Transcript 🏃

on:
  workflow_dispatch:
    inputs:
      talk_title:
        description: Title of the TED Talk
        required: true

jobs:
  ted-talk-transcript:
    steps:
      - name: Run
        env:
          IFTTT_API_KEY: ${{ secrets.IFTTT_API_KEY }}
        run: clojure -X core/run :job ted-talk-transcript :input '"${{ github.event.inputs.talk_title }}"'

2. Find the talk on TED.com

Once I had the title of the talk from Youtube, I searched for it on https://www.ted.com/talks and looked through the html to find all links of the pattern /talks/video-id using regex. Each talk on the search results page has two links so I used distinct to get a unique set of video ids.

(def base-url "https://www.ted.com")

(defn- search-talks [query]
  (let [query-url   (str base-url "/talks?sort=relevance&q=" query)
        results     (-> (hc/get query-url)
                        :body)
        links       (re-seq #"/talks/(.*)'" results)
        video-ids   (distinct (map second links))]
    video-ids))

3. Grab the transcript

Using Inspect network activity on their talk videos I found out that they have a neat GraphQL endpoint: https://www.ted.com/graphql. All I had to do was provide the video id and I got back the entire transcript.

...
  (let [gql (str "{translation(language:\"en\", videoId:\""
                 video-id
                 "\") {id paragraphs { cues { text }}}}")
        response (hc/post (str base-url "/graphql") {:form-params {:operationName nil
                                                                   :query gql}
                                                     :content-type :json})
        json (-> response
                 :body
                 (cheshire/parse-string true))
...

4. Send the transcript to IFTTT

Once I had the transcript, I sent it as JSON payload to IFTTT. IFTTT in turn emails me the transcript but more on this later.

(let [url "https://maker.ifttt.com/trigger/email/json/with/key/IFTTT_API_KEY"]
  (hc/post url {:form-params payload
                :content-type :json}))

We are not done yet

Let's recap what we had to do:

  1. Watch out for new talks on TED's YouTube Channel
  2. Find the talk on the official website and grab the transcript
  3. Email it to myself

So far we are done with #2 but we still need to get #1 and #3 done.

#1 Watch out for new talks

Let's create a IFTTT job for exactly this:

YouTube Applet Screenshot 1

YouTube Applet Screenshot 2

YouTube Applet Screenshot 3

Few things to note:

  1. You need a GitHub Personal Access Token with repo access. Replace ghp_token in the above screenshot with your token. GitHub Personal Access Token
  2. Remember to add User-Agent header (it won't work otherwise)

#3 Email

Let's create a IFTTT applet which emails whatever it receives on the webhook.

Email Applet Screenshot 1

Email Applet Screenshot 2

Email Applet Screenshot 3

Recap

  1. We created a IFTTT applet which watches out for any new TED videos on Youtube. When it finds out about a new video, it sends the talk title to a GitHub endpoint.
  2. On GitHub, a workflow is triggered which receives the talk's title and runs Clojure code to grab its transcript from the TED official website.
  3. Once it has the transcript, it sends it to another IFTTT applet which in turn emails it to me.

As the above example demonstrates, we can use no-code automation tools alongside custom code while avoiding additional cost and maintenance. All of this enables us to get the best of both worlds.

I hope you found article useful, if you have any feedback or ideas for improvement let me know or simply create a PR on the repository 💪