A sample bot that passes simple media attachments (images) to a user activity.
The minimum prerequisites to run this sample are:
- The latest update of Visual Studio 2015. You can download the community version here for free.
- The Bot Framework Emulator. To install the Bot Framework Emulator, download it from here. Please refer to this documentation article to know more about the Bot Framework Emulator.
Many messaging channels provide the ability to attach richer objects. To pass a simple media attachment (image/audio/video/file) to an activity you add a simple attachment data structure with a link to the content, setting the ContentType, ContentUrl and Name properties.
The Attachments property is an array of Attachment objects which allow you to send and receive images and other content. Check out the key code located in the SendAttachmentDialog class where the replyMessage.Attachments
property of the message activity is populated with an image attachment.
public async Task ProcessSelectedOptionAsync(IDialogContext context, IAwaitable<string> argument)
{
var message = await argument;
var replyMessage = context.MakeMessage();
Attachment attachment = null;
switch (message)
{
case "1":
attachment = GetInlineAttachment();
break;
case "2":
attachment = await GetUploadedAttachmentAsync(replyMessage.ServiceUrl, replyMessage.Conversation.Id);
break;
case "3":
attachment = GetInternetAttachment();
break;
}
// The Attachments property allows you to send and receive images and other content
replyMessage.Attachments = new List<Attachment> { attachment };
await context.PostAsync(replyMessage);
await this.DisplayOptionsAsync(context);
}
As a developer, you have three ways to send the attachment. The attachment can be:
- An inline file, by encoding the file as base64 and use it in the contentUrl
- A file uploaded to the channel's store via the Connection API, then using the attachmentId to create the contentUrl
- An externally hosted file, by just specifying the Url of the file (it should be publicly accessible)
It consists on sending the file contents, encoded in base64, along with the message payload. This option works for small files, like icon size images.
You'll need to encode file's content, then set the attachment's contentUrl
as follows:
…
Checkout GetInlineAttachment to see how to convert a file read from filesystem and return the attachment instance to add to the attachments collection in the key method shown above (ProcessSelectedOptionAsync
).
private static Attachment GetInlineAttachment()
{
var imagePath = HttpContext.Current.Server.MapPath("~/images/small-image.png");
var imageData = Convert.ToBase64String(File.ReadAllBytes(imagePath));
return new Attachment
{
Name = "small-image.png",
ContentType = "image/png",
ContentUrl = $"data:image/png;base64,{imageData}"
};
}
This option should be used when the file to send is less than 256Kb in size when encoded to base64. A good scenario are images generated based on user input.
Checkout GetUploadedAttachmentAsync to see how to get the required information to create the attachment instance to add to the attachments collection in the key method shown above (ProcessSelectedOptionAsync
).
It does require a few more steps than the other methods, but leverages the channels store to store the file:
- Get an instance to the Connector API which will handle communication with channel storage (relevant code)
- Create a new attachment set providing the Connector API instance as argument (relevant code)
- Read (or generate) the content file and store it in a Byte array to use it as argument within the AttachmentData members (relevant code)
- Perform the attachment upload to the channel (relevant code)
- Get the attachment Uri (using the uploaded Id) where the channel stored the uploaded image (relevant code)
private static async Task<Attachment> GetUploadedAttachmentAsync(string serviceUrl, string conversationId)
{
var imagePath = HttpContext.Current.Server.MapPath("~/images/big-image.png");
using (var connector = new ConnectorClient(new Uri(serviceUrl)))
{
var attachments = new Attachments(connector);
var response = await attachments.Client.Conversations.UploadAttachmentAsync(
conversationId,
new AttachmentData
{
Name = "big-image.png",
OriginalBase64 = File.ReadAllBytes(imagePath),
Type = "image/png"
});
var attachmentUri = attachments.GetAttachmentUri(response.Id);
return new Attachment
{
Name = "big-image.png",
ContentType = "image/png",
ContentUrl = attachmentUri
};
}
}
This option is the simplest but requires the image to be already on the Internet and be publicly accesible. You could also provide an Url pointing to your own site.
Checkout GetInternetAttachment to see how to get the required information to create the attachment instance to add to the attachments collection in the key method shown above (ProcessSelectedOptionAsync
).
private static Attachment GetInternetAttachment()
{
return new Attachment
{
Name = "BotFrameworkOverview.png",
ContentType = "image/png",
ContentUrl = "https://docs.botframework.com/en-us/images/faq-overview/botframework_overview_july.png"
};
}
You will see the following in the Bot Framework Emulator when selecting the inline attachment. See how the image is encoded in the contentUrl
of the attachment.
You will see the following in your Facebook Messenger when selecting to upload the attachment.
On the other hand, you will see the following in Skype when selecting an Internet attachment.
To get more information about how to get started in Bot Builder for .NET and Attachments please review the following resources: