Context: I'm approaching Haskell from a standpoint of converting runtime errors to compile-time errors. My hypothesis is that this is possible if one can codify business logic within the program's types itself.
I'm writing a Telegram bot, which should be accessible by users within my company. To achieve this "restriction", whenever someone starts chatting with the bot it will look up the chat_id
in a table and check if a valid oauth_token
exists. If not, the user will first be sent a link to complete a Google OAuth (our company's email is hosted on Google Apps for Business).
share [mkPersist sqlSettings, mkMigrate "migrateAll"] [persistLowerCase|
VLUser
email String
chatId Integer
tgramUserId Integer
tgramFirstName String
tgramLastName String Maybe
tgramUsername String Maybe
oauthToken String Maybe
deriving Show
|]
Users with a valid oauth_token
will be able to give the Telegram bot some commands, which unauthenticated users should not be able to give.
Now, I'm trying to codify this logic at the type level itself. There will be some functions in my Haskell code that will have the ability to accept, as arguments, both, authenticated & unauthenticated users; while some functions should accept only authenticated users.
If I keep passing user objects of the same type, i.e. VLUser
everywhere, then I will have to be careful to check for the presence of oauthToken
in every function. Is there a way to create two user types - VLUser
and VLUserAuthenticated
where:
- Both map to the same underlying table
- A
VLUserAuthenticated
can be instantiated ONLY IF it has anoauthToken