aboutsummaryrefslogtreecommitdiff
path: root/src/Tomato/Post.hs
blob: 5f5d8af13d3fcf9e7984eeaa31e36a8080adac3e (plain)
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
module Tomato.Post
    ( postTomato
    , uploadTomato
    , postMessage
    ) where

import RIO

import Network.HTTP.Req
    ( jsonResponse
    , ignoreResponse
    , req
    , Req
    , defaultHttpConfig
    , runReq
    , ReqBodyFile (..)
    , ReqBodyJson (..)
    , (/:)
    , POST (..)
    , Req
    , https
    , header
    , responseBody
    )

import Data.HashMap.Strict (insert)

import Tomato.Data.Except (DecodeException (..))
import Tomato.Data.Message (GMIUrl (..), OutMessage (..))
import Tomato.App (App (..))

import qualified Data.Aeson as Ae
import qualified RIO.Text as T


-- | Post the current tomato
postTomato :: RIO App ()
postTomato = do
    url <- uploadTomato
    postMessage $ OutMessage url

-- | Upload the current tomato to the GroupMe image server
uploadTomato :: RIO App GMIUrl
uploadTomato = do
    tomatoFile <- asks appFile
    accessToken <- asks appToken
    let url = https "image.groupme.com" /: "pictures"
    js <- rr $ req POST url (ReqBodyFile tomatoFile) jsonResponse $
        header "X-Access-Token" accessToken <>
        header "Content-Type" "image/png"
    gmURL <- case Ae.fromJSON (responseBody js) of
        Ae.Success r -> pure r
        Ae.Error s -> throwM $ DecodeException $ "Get groupme url:" <> T.pack s
    return gmURL

-- | Post a message to the group with image as attachment
postMessage :: OutMessage -> RIO App ()
postMessage outMes = do
    botId <- asks appBotId
    accessToken <- asks appToken
    let url = https "api.groupme.com" /: "v3" /: "bots" /: "post"
        outMes' = case Ae.toJSON outMes of
            Ae.Object o -> Ae.Object $ insert "bot_id" (Ae.toJSON botId) o
            _           -> error "exceptional"
    rr $ req POST url (ReqBodyJson outMes') ignoreResponse $
        header "X-Access-Token" accessToken
    return ()

rr :: Req a -> RIO App a
rr = runReq defaultHttpConfig