- [new] alias LINE ButtonsTemplate to ButtonTemplate to match type name
buttons#219
Aliases:
- replyButtonsTemplate -> replyButtonTemplate
- pushButtonsTemplate -> pushButtonTemplate
- sendButtonsTemplate -> sendButtonTemplate
- [new] Add cli option
-t,--tokento overwrite token setting.
- [new] Add new Update APIs to
TelegramContext:
editMessageText(text [, options])editMessageCaption(caption [, options])editMessageReplyMarkup(replyMarkup [, options])deleteMessage(messageId)editMessageLiveLocation(location [, options])stopMessageLiveLocation(options)forwardMessageFrom(fromChatId, messageId [, options])forwardMessageTo(toChatId, messageId [, options])
- [new] Add new Get APIs to
TelegramContext:
context.getUserProfilePhotos({ limit: 1 }).then(result => {
console.log(result);
// {
// total_count: 3,
// photos: [
// [
// {
// file_id: 'AgADBAADGTo4Gz8cZAeR-ouu4XBx78EeqRkABHahi76pN-aO0UoDA050',
// file_size: 14650,
// width: 160,
// height: 160,
// },
// {
// file_id: 'AgADBAADGTo4Gz8cZAeR-ouu4XBx78EeqRkABKCfooqTgFUX0EoD5B1C',
// file_size: 39019,
// width: 320,
// height: 320,
// },
// {
// file_id: 'AgADBAADGTo4Gz8cZAeR-ouu4XBx78EeqRkABPL_pC9K3UpI0koD1B1C',
// file_size: 132470,
// width: 640,
// height: 640,
// },
// ],
// ],
// }
});context.getChat().then(result => {
console.log(result);
// {
// id: 313534466,
// first_name: 'first',
// last_name: 'last',
// username: 'username',
// type: 'private',
// }
});context.getChatAdministrators().then(result => {
console.log(result);
// [
// {
// user: {
// id: 313534466,
// first_name: 'first',
// last_name: 'last',
// username: 'username',
// languange_code: 'zh-TW',
// },
// status: 'creator',
// },
// ]
});context.getChatMembersCount().then(result => {
console.log(result);
// '6'
});context.getChatMember().then(result => {
console.log(result);
// {
// user: {
// id: 313534466,
// first_name: 'first',
// last_name: 'last',
// username: 'username',
// languange_code: 'zh-TW',
// },
// status: 'creator',
// }
});- [new] Add new Group APIs to
TelegramContext:
context.kickChatMembercontext.unbanChatMembercontext.restrictChatMembercontext.promoteChatMembercontext.exportChatInviteLinkcontext.setChatPhotocontext.deleteChatPhotocontext.setChatTitlecontext.setChatDescriptioncontext.setChatStickerSetcontext.deleteChatStickerSetcontext.pinChatMessagecontext.unpinChatMessagecontext.leaveChat
- [new] Add new Payment APIs to
TelegramContext:
- [new] Add new
LineContextmethods:
context.getUserProfile().then(profile => {
console.log(profile);
// {
// displayName: 'LINE taro',
// userId: USER_ID,
// pictureUrl: 'http://obs.line-apps.com/...',
// statusMessage: 'Hello, LINE!',
// }
});context.getMemberProfile(USER_ID).then(member => {
console.log(member);
// {
// "displayName":"LINE taro",
// "userId":"Uxxxxxxxxxxxxxx...",
// "pictureUrl":"http://obs.line-apps.com/..."
// }
});context.getMemberIds(CURSOR).then(res => {
console.log(res);
// {
// memberIds: [
// 'Uxxxxxxxxxxxxxx...',
// 'Uxxxxxxxxxxxxxx...',
// 'Uxxxxxxxxxxxxxx...'
// ],
// next: 'jxEWCEEP...'
// }
});context.getAllMemberIds().then(ids => {
console.log(ids);
// [
// 'Uxxxxxxxxxxxxxx..1',
// 'Uxxxxxxxxxxxxxx..2',
// 'Uxxxxxxxxxxxxxx..3',
// 'Uxxxxxxxxxxxxxx..4',
// 'Uxxxxxxxxxxxxxx..5',
// 'Uxxxxxxxxxxxxxx..6',
// ]
});- [new] Implement
context.getUserProfile:
const user = await context.getUserProfile();
// {
// first_name: 'Kevin',
// last_name: 'Durant',
// profile_pic: 'https://example.com/pic.png',
// locale: 'en_US',
// timezone: 8,
// gender: 'male',
// };- [new] Implement
context.sendSenderAction:
context.sendSenderAction('typing_on');
// same as
context.typingOn();- [fix] Fix metadata argument in handover methods (#208)
- [new] Implement
context.getUserDetails:
const user = await context.getUserDetails();
// {
// id: '01234567890A=',
// name: 'John McClane',
// avatar: 'http://avatar.example.com',
// country: 'UK',
// language: 'en',
// primary_device_os: 'android 7.1',
// api_version: 1,
// viber_version: '6.5.0',
// mcc: 1,
// mnc: 1,
// device_type: 'iPhone9,4',
// };- [new] Implement
context.getOnlineStatus:
const status = await context.getOnlineStatus();
// {
// id: '01234567890=',
// online_status: 0,
// online_status_message: 'online',
// }- [new] Support using
/exitto exit
- [new] Support passing
verifyTokentoMessengerBotorMessengerConnector(#204)
Support both:
- pass verifyToken in server config
const bot = new MessengerBot({
accessToken: config.accessToken,
appSecret: config.appSecret,
});
const server = createServer(bot, { verifyToken: config.verifyToken });- pass verifyToken in bot config
const bot = new MessengerBot({
accessToken: config.accessToken,
appSecret: config.appSecret,
verifyToken: config.verifyToken,
});
const server = createServer(bot);- [fix] Fix LINE join, leave events which do not have userId will fail get profile (#206)
- [fix] Upgrade
@slack/clientdeps to fix security vulnerability
- [fix] fix context simulator initialState (#198)
- [new] Support request thread control:
context.requestThreadControl();
context.event.isRequestThreadControl; // true or false
context.event.requestThreadControl;
/*
{
metadata: 'additional content that the caller wants to set',
requested_owner_app_id: '123456789',
}
*/- [fix] fix error on verifying facebook signature.
- [deps] bump messaging-apis to v0.6.13
- [new] Support trigger payload in
ConsoleBot
You > /payload PAYLOAD
Receive event:
context.event.isMessage; // false
context.event.message; // null
context.event.isText; // false
context.event.text; // null
context.event.isPayload; // true
context.event.payload; // PAYLOAD- [new] Support all of methods on
ConsoleContextwithfallbackMethods: true#184, for example:
const bot = new ConsoleBot({ fallbackMethods: true });
bot.onEvent(async context => {
await context.sendText('Hello World');
await context.sendImage('https://example.com/vr.jpg');
await context.sendButtonTemplate('What do you want to do next?', [
{
type: 'web_url',
url: 'https://petersapparel.parseapp.com',
title: 'Show Website',
},
{
type: 'postback',
title: 'Start Chatting',
payload: 'USER_DEFINED_PAYLOAD',
},
]);
});Result:
Bot > Hello World
Bot > sendImage: ["https://example.com/vr.jpg"]
Bot > sendButtonTemplate: ["What do you want to do next?",[{"type":"web_url","url":"https://petersapparel.parseapp.com","title":"Show Website"},{"type":"postback","title":"Start Chatting","payload":"USER_DEFINED_PAYLOAD"}]]
- [fix] Catch write session error in async mode #185
- [improve] Show warning when calling
setStateorresetStateafter session have been written #186
- [fix] Use sync mode in
ConsoleBotto fix some format issue.
- [new] Support
--ngrok-portwhen setting webhook #171. For example:
bottender messenger webhook set --ngrok-port 1234
bottender telegram webhook set --ngrok-port 1234
bottender viber webhook set --ngrok-port 1234
- [deps] Update part of dependencies.
- [new] Implement signature verify for viber
- [fix] Prevent applying any mutations to
initialState#164
- [new] Added
context.answerInlineQueryfor Telegram #165:
context.answerInlineQuery(
[
{
type: 'photo',
id: 'UNIQUE_ID',
photo_file_id: 'FILE_ID',
title: 'PHOTO_TITLE',
},
{
type: 'audio',
id: 'UNIQUE_ID',
audio_file_id: 'FILE_ID',
caption: 'AUDIO_TITLE',
},
],
{
cache_time: 1000,
}
);- [changed] Improve config schema validation.
- [experimental] add Slack RTM API support:
const { SlackBot } = require('bottender');
const bot = new SlackBot({
accessToken: '__FILL_YOUR_TOKEN_HERE__',
});
bot.onEvent(async context => {
await context.sendText('Hello World');
});
bot.createRtmRuntime();- [new] Handle all of telegram event types includes:
messageedited_messagechannel_postedited_channel_postinline_querychosen_inline_resultcallback_queryshipping_querypre_checkout_query
- [new] Support group chat events.
- [new] Better handle Messenger getUserProfile failure #155
If getUserProfile throw error, session.user will fallback to have only id and _updatedAt keys.
- [new] Added more event parser and getter to telegram event #150
event.isEditedMessage
event.editedMessage
event.isChannelPost
event.channelPost
event.isEditedChannelPost
event.editedChannelPost
event.isInlineQuery
event.inlineQuery
event.isChosenInlineResult
event.chosenInlineResult
event.isShippingQuery
event.shippingQuery
event.isPreCheckoutQuery
event.preCheckoutQuery
- [new] Add
context.postEphemeral:
context.postEphemeral({ text: 'hello' });- [fix] Reply to thread instead of channel when receiving events in thread #145
- [fix] Use
message.chat.idto reply #148
- [fix] Improve error handling in express middleware #142
- [new] Add optional
--yesfor Messenger force upload attachment #127 - [new] Initial State in test-utils #126
- [fix] Improve context simulator and add some tests #131
- [fix] Support Messenger webhook test requests #139
- [new] Support running Telegram bots with long polling #117
const { TelegramBot } = require('bottender');
const bot = new TelegramBot({
accessToken: '__FILL_YOUR_TOKEN_HERE__',
});
bot.onEvent(async context => {
await context.sendText('Hello World');
});
bot.createLongPollingRuntime({
limit: 100,
timeout: 60,
allowed_updates: ['message', 'callback_query'],
});See more details in examples/telegram-long-polling
- [fix] Result destructing bugs in Telegram webhook commands
- [fix] ContextSimulator: sendState to setState #108
- [deprecated]
sendXXXWithDelayis deprecated. UsesendXXXwithoptions.delayinstead.
- [fix] update messaging-api-messenger to fix empty array quick_replies bug
- [new] Add LINE
context.leave()function for group and room #107
- [fix] Fix
context.sendVideoNoteusingmessaging-api-telegramv0.6.5 - [new] Add context.methods:
sendMediaGroup:
context.sendMediaGroup([
{ type: 'photo', media: 'BQADBAADApYAAgcZZAfj2-xeidueWwI' },
]);Payment API:
context.sendInvoice({
title: 'product name',
description: 'product description',
payload: 'bot-defined invoice payload',
provider_token: 'PROVIDER_TOKEN',
start_parameter: 'pay',
currency: 'USD',
prices: [
{ label: 'product', amount: 11000 },
{ label: 'tax', amount: 11000 },
],
});Game API:
context.sendGame('Mario Bros.');
context.setGameScore(999);
context.getGameHighScores().then(result => {
console.log(result);
/*
{
ok: true,
result: [
{
position: 1,
user: {
id: 427770117,
is_bot: false,
first_name: 'first',
},
score: 999,
},
],
};
*/
});Introducing Viber Support to Bottender!
const { ViberBot } = require('bottender');
const { createServer } = require('bottender/express');
const bot = new ViberBot({
accessToken: '__FILL_YOUR_TOKEN_HERE__',
});
bot.onEvent(async context => {
if (context.event.isMessage) {
await context.sendText('Hello World');
}
});
const server = createServer(bot);
server.listen(5000, () => {
console.log('server is running on 5000 port...');
});See viber-hello-world for more details.
- [new] Add
update-notifierin CLI #99 - [deps] Update messaging API clients to
v0.6.x.
- [fix] Fix domain whitelisting usage
- [fix] Check messenger menu item length #71
- [fix] Handle LINE webhook verify request in LineConnector #100
- [new] Add Slack signature validation #94
- [improve] Let slack connector handle promises parallelly #105
- [new] Add referral getters for
MessengerEvent:
event.isReferral; // true or false
event.referral; // { source: 'SHORTLINK', type: 'OPEN_THREAD', ref: 'my_ref' }
event.ref; // 'my_ref'- [new] Create
README.mdand.gitignorewhenbottender init - [deps] Update messaging-apis to
v0.5.16
- [new] Add
event.isFromCustomerChatPlugingetter - [new] Implement CLI attachment force upload #70
- [fix] Fix CLI profile bug
- [fix] Add huge like sticker support to isLikeSticker #67
- [fix] Use timingSafeEqual to validate signature #79
- [fix] Use timingSafeEqual to validate signature #79
- [new] Add
mapPageToAccessTokento support multiple pages (Experimental) #47
new MessengerBot({
appSecret: '__FILL_YOUR_SECRET_HERE__',
mapPageToAccessToken: pageId => accessToken,
});Note: API may changes between any versions.
- [new] Export
context.replyandcontext.pushmethods. #52 - [new] New CLI commands to sync LINE rich menus: #50
$ bottender line menu get
$ bottender line menu set
$ bottender line menu delete- [new] Add support to interactive messages, and you can get action from it: #41
if (context.event.isInteractiveMessage) {
console.log(context.event.action);
}- [new] A new command to upload your messenger attachments from
/assetsdirectory (in beta):
$ bottender messenger attachment uploadThen, you can import them with getAttachment util function:
const { getAttachment } = require('bottender/utils');
console.log(getAttachment('mypic.jpg').id); // '1591074914293017'- [new] Add
--forceoption tobottender messenger profile set(delete all and set all) - [fix] Fix file export for
test-utils.js#44 - [fix] Refined affected methods in
withTyping#35
- [fix] Stop passing
as_user: true#33
- [new] Add
--skip-validatecli option to skipJoischema validation #31 - [fix] Allow unknown keys in config and fix schema rules #29
- [new] Add options for
postMessage#25
You can use it to send additional attachments, like below:
context.postMessage('I am a test message', {
attachments: [
{
text: "And here's an attachment!",
},
],
});See official docs for more available options.
-
[new] Implement richmenu api methods on context #23
context.getLinkedRichMenu()context.linkRichMenu(richMenuId)context.unlinkRichMenu()
- [new] Add new send methods #19
context.sendMessagecontext.sendTemplatecontext.sendOpenGraphTemplatecontext.sendMediaTemplate
- [new] Implement label api methods for targeting broadcast messages #18
context.associateLabel(labelId)context.dissociateLabel(labelId)context.getAssociatedLabels()
- [new] Implement thread control methods #15
context.passThreadControl(targetAppId, metadata)context.passThreadControlToPageInboxcontext.takeThreadControl
- [new] Send
messaging_typeasRESPONSEwhen reply anything in the context. #12 - [deps] Upgrade Messaging APIs clients to latest.
First public release.
