I am attempting to write a reasonml binding for the amqplib npm package:
http://www.squaremobius.net/amqp.node/
In particular this function:
http://www.squaremobius.net/amqp.node/channel_api.html#channel_get
class type amqpMessageT = [@bs] {
pub content: nodeBuffer
};
type amqpMessage = Js.t(amqpMessageT);
type gottenMessage = Js.Nullable.t(amqpMessage);
type qualifiedMessage = Message(gottenMessage) | Boolean(bool);
class type amqpChannelT = [@bs] {
pub assertQueue: string => queueParams => Js.Promise.t(unit);
pub consume: string => (amqpMessage => unit) => unit;
pub ack: amqpMessage => unit;
pub get: string => Js.Promise.t(qualifiedMessage);
pub purgeQueue: string => Js.Promise.t(unit);
pub deleteQueue: string => Js.Promise.t(unit);
pub sendToQueue: string => nodeBuffer => messageParams => unit;
};
And then I have the following code:
....
channel##get("MyQueue")
|> Js.Promise.then_(message => {
switch message {
| Boolean(false) => Js.Promise.resolve(Js.log("No Message"));
| Message(msg) => Js.Promise.resolve(Js.log("Has Message, Will Travel"));
| Boolean(true) => Js.Promise.resolve(Js.log("Impossible Message"!));
}
}
However this goes down the "Message(msg)" path always, even when the js call returns false.
Now adding the following binding:
let unsafeGet: amqpChannel => string => Js.Promise.t(gottenMessage) = [%bs.raw{|function(channel, queueName) {
return channel.get(queueName).then((value) => {
if(value === false) {
return Promise.resolve(null)
} else {
return Promise.resolve(value)
}
})
}|}];
I've been able to sidestep the problem, but I'm not a huge fan of using bs.raw
if I'm honest. What's the issue with my initial untagged union type? How can I fix this problem?