Skip to content
This repository has been archived by the owner on Jun 18, 2024. It is now read-only.

Latest commit

 

History

History
747 lines (620 loc) · 32 KB

File metadata and controls

747 lines (620 loc) · 32 KB

Microsoft Graph for Calendar

In this lab, you will use the Microsoft Graph SDK to work with Office 365 calendars as part of an ASP.NET MVC5 application.

Prerequisites

  1. You must have an Office 365 tenant and Microsoft Azure subscription to complete this lab. If you do not have one, the lab for O3651-7 Setting up your Developer environment in Office 365 shows you how to obtain a trial.
  2. You must have Visual Studio 2015.

Lab Setup: Setting up your Exchange account with Sample Calendar Events for Testing

  1. Using the browser, navigate to https://outlook.office365.com and log into your Office 365 mailbox.
  2. Click the waffle icon in the top-left corner to open the App Launcher and click the Calendar tile.
  3. Add some calendar items to your calendar if you don't have any in your mailbox.
  4. Once you have verified that you have a set of calendar events for testing, you can move on to the next exercise.

Exercise 1: Create an ASP.NET MVC5 Application

In this exercise, you will create the ASP.NET MVC5 application and register it with Azure active Directory.

  1. Launch Visual Studio 2015 as administrator.
  2. In Visual Studio, click File/New/Project.
  3. In the New Project dialog
  4. Select Templates/Visual C#/Web.
  5. Select ASP.NET Web Application.
![Screenshot of the previous step](Images/01.png)
  1. Give the project the name Office365Calendar and Click OK.
  2. In the New ASP.NET Project dialog
  3. Click MVC.
  4. Click Change Authentication.
  5. Select Work And School Accounts.
  6. Select Cloud - Single Organization
  7. Input Domain of your O365 tenancy
  8. Check Read directory data under Directory Access Permissions
  9. Click OK.
  10. Uncheck Host in the cloud
  11. Click OK.
![Screenshot of the previous step](Images/03.png)

![Screenshot of the previous step](Images/02.png)
  1. Ensure the web project uses SSL by default:
  2. In the Solution Explorer tool window, select the project and look at the Properties tool window.
  3. Ensure SSL Enabled is set to TRUE.
  4. Copy the SSL URL property to the clipboard for use in the next step.
![Screenshot of the previous step](Images/SslEnabled.png)
> It is important to do this now because in the next step when you create the application in Azure AD, you want the reply URL to use HTTPS. If you did not do this now, you would have to manually make the changes the Visual Studio wizard is going to do for you in creating the app.
  1. Configure the project always goes to the homepage of the web application when debugging:
  2. In the Solution Explorer tool window, right-click the project and select Properties.
  3. Select the Web tab in the left margin.
  4. Find the section Start Action.
  5. Click the radio button Start URL and enter the SSL URL of the web project that you copied from the previous step.
![Screenshot of the previous step](Images/StartUrl.png)
  1. At this point you can test the authentication flow for your application.
  2. In Visual Studio, press F5. The browser will automatically launch taking you to the HTTPS start page for the web application.

Note: If you receive an error that indicates ASP.NET could not connect to the SQL database, please see the SQL Server Database Connection Error Resolution document to quickly resolve the issue.

  1. To sign in, click the Sign In link in the upper-right corner.
  2. Login using your Organizational Account.
  3. Upon a successful login, since this will be the first time you have logged into this app, Azure AD will present you with the common consent dialog that looks similar to the following image:
![Screenshot of the previous step](Images/ConsentDialog.png)
  1. Click Accept to approve the app's permission request on your data in Office 365.
  2. You will then be redirected back to your web application. However notice in the upper right corner, it now shows your email address & the Sign Out link.

Congratulations... at this point your app is configured with Azure AD and leverages OpenID Connect and OWIN to facilitate the authentication process!

Exercise 2: Configure Web Application to use Azure AD and OWIN

In this exercise you will take the ASP.NET MVC web application you created in the previous exercise and configure it to use Azure AD & OpenID Connect for user & app authentication. You will do this by utilizing the OWIN framework. Once authenticated, you can use the access token returned by Azure AD to access the Microsoft Graph.

  1. Grant App Necessary Permissions.

  2. Browse to the Azure Management Portal and sign in with your Organizational Account.

  3. In the left-hand navigation, click Active Directory.

  4. Select the directory you share with your Office 365 subscription.

  5. Select the application you created for this lab.

  6. Open Configure tab

  7. Scroll down to the permissions to other applications section.

  8. Click the Add Application button.

  9. In the Permissions to other applications dialog, click the PLUS icon next to the Microsoft Graph option.

  10. Click the CHECK icon in the lower right corner.

  11. For the new Microsoft Graph application permission entry, select the Delegated Permissions dropdown on the same line and then select the following permissions: * Have full access to user calendars

  12. Click the Save button at the bottom of the page.

    Screenshot of the previous step

  13. Add a helper class that will be used to harvest settings out of the web.config and create the necessary strings that will be used for authentication:

  14. Right-click the project and select Add/New Folder. Give the folder the name Utils.

  15. Locate the [\\O3653-13 Deep Dive into Office 365 with the Microsoft Graph API for Calendar\Lab Files](./Lab Files) folder provided with this lab and find the [SettingsHelper.cs](./Lab Files/SettingsHelper.cs) file . Drag the [SettingsHelper.cs](./Lab Files/SettingsHelper.cs) file to the Utils folder in the project.

  16. Update _Layout file to add Events link:

    1. Open the _Layout.cshtml file found in the Views/Shared folder.
    2. Locate the part of the file that includes a few links at the top of the page... it should look similar to the following code:
     ````asp
     <div class="navbar-collapse collapse">
       <ul class="nav navbar-nav">
         <li>@Html.ActionLink("Home", "Index", "Home")</li>
         <li>@Html.ActionLink("About", "About", "Home")</li>
         <li>@Html.ActionLink("Contact", "Contact", "Home")</li>
       </ul>
       @Html.Partial("_LoginPartial")
     </div>
     ````
    
    1. Update that navigation to have a new link (the Events link added below) as well as a reference to the login control you just created:
     ````asp
     <div class="navbar-collapse collapse">
       <ul class="nav navbar-nav">
         <li>@Html.ActionLink("Home", "Index", "Home")</li>
         <li>@Html.ActionLink("About", "About", "Home")</li>
         <li>@Html.ActionLink("Contact", "Contact", "Home")</li>
         <li>@Html.ActionLink("Events", "Index", "Calendar")</li>
       </ul>
       @Html.Partial("_LoginPartial")
     </div>
     ````
    
     > The **Events** link will not work yet... you will add that in the next exercise.
    

Exercise 3: Code the Calendar API

In this exercise, you will create a repository object for wrapping CRUD operations associated with the Calendar API.

  1. In the Solution Explorer, locate the Models folder in the Office365Calendar project.

  2. Right-click the Models folder and select Add/Class.

  3. In the Add New Item dialog, name the new class MyEvent and click Add to create the new source file for the class.

    1. At the top of the source file MyEvent.cs, add the following using statement just after the using statements that are already there.
    using System.ComponentModel;
    using System.ComponentModel.DataAnnotations;
    1. Implement the new class MyEvent using the following class definition.
    public class MyEvent {
      public string Id { get; set; }
         
      [DisplayName("Subject")]
      public string Subject { get; set; }
      
      [DisplayName("Start Time")]
      [DisplayFormat(DataFormatString = "{0:MM/dd/yyyy}", ApplyFormatInEditMode = true)]
      public DateTimeOffset? Start { get; set; }
      
      [DisplayName("End Time")]
      [DisplayFormat(DataFormatString = "{0:MM/dd/yyyy}", ApplyFormatInEditMode = true)]
      public DateTimeOffset? End { get; set; }
      
      [DisplayName("Location")]
      public string Location { get; set; }
      
      [DisplayName("Body")]
      public string Body { get; set; }      
    }
  4. Assembly references are not added to the shared projects in ASP.NET MVC, rather they are added to the actual client projects. Therefore you need to add the following NuGet packages manually.

    1. Open the Package Manager Console: View/Other Windows/Package Manager Console.

    2. Enter each line below in the console, one at a time, pressing ENTER after each one. NuGet will install the package and all dependent packages:

      PM> Install-Package Microsoft.Graph
  5. Right-click the Models folder and select Add/Class. In the Add New Item dialog, name the new class MyEventsRepository and click Add to create the new source file for the class.

    1. Use the following using statements instead of the MyEventsRepository class old using statements.
    using System;
    using System.Collections.Generic;
    using System.Security.Claims;
    using System.Threading.Tasks;
    using Office365Calendar.Utils;
    using System.Net.Http.Headers;
    using Microsoft.IdentityModel.Clients.ActiveDirectory;
    using System.Linq;
    using Microsoft.Graph;
    1. Add a function named GetGraphAccessTokenAsync to the MyEventsRepository class with the following implementation to get access token for Microsoft Graph Authentication.
    private async Task<string> GetGraphAccessTokenAsync()
    {
        var signInUserId = ClaimsPrincipal.Current.FindFirst(ClaimTypes.NameIdentifier).Value;
        var userObjectId = ClaimsPrincipal.Current.FindFirst(SettingsHelper.ClaimTypeObjectIdentifier).Value;
    
        var clientCredential = new ClientCredential(SettingsHelper.ClientId, SettingsHelper.ClientSecret);
        var userIdentifier = new UserIdentifier(userObjectId, UserIdentifierType.UniqueId);
    
        AuthenticationContext authContext = new AuthenticationContext(SettingsHelper.AzureAdAuthority, new ADALTokenCache(signInUserId));
        var result = await authContext.AcquireTokenSilentAsync(SettingsHelper.AzureAdGraphResourceURL, clientCredential, userIdentifier);
        return result.AccessToken;
    }
    1. Add a function named GetGraphServiceAsync to the MyEventsRepository class with the following implementation to get Graph service client.
    private async Task<GraphServiceClient> GetGraphServiceAsync()
    {
        var accessToken = await GetGraphAccessTokenAsync();
        var graphserviceClient = new GraphServiceClient(SettingsHelper.GraphResourceUrl,
                                      new DelegateAuthenticationProvider(
                                                    (requestMessage) =>
                                                    {
                                                        requestMessage.Headers.Authorization = new AuthenticationHeaderValue("bearer", accessToken);
                                                        return Task.FromResult(0);
                                                    }));
    
        return graphserviceClient;
    }
    1. Add a function named GetEvents to the MyEventsRepository class to retrieve and return a list of MyEvent objects.
    public async Task<List<MyEvent>> GetEvents(int pageIndex, int pageSize)
    {
        try
        {
            var graphServiceClient = await GetGraphServiceAsync();
            var requestEvents = await graphServiceClient.Me.Events.Request().Top(pageSize).Skip(pageIndex * pageSize).GetAsync();
            var eventsResults = requestEvents.CurrentPage.Select(x => new MyEvent
            {
                Id = x.Id,
                Subject = x.Subject,
                Body = x.Body.Content,
                Location = x.Location.DisplayName,
                Start = DateTime.SpecifyKind(DateTime.Parse(x.Start.DateTime), x.Start.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local).ToLocalTime(),
                End = DateTime.SpecifyKind(DateTime.Parse(x.End.DateTime), x.End.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local).ToLocalTime(),
            }).ToList();
            return eventsResults;
        }
        catch
        {
            return null;
        }
    }
    1. Add a GetEvent function to the MyEventsRepository class to get a specific event:
    public async Task<MyEvent> GetEvent(string id)
    {
        try
        {
            var graphServiceClient = await GetGraphServiceAsync();
            var requestEvent = await graphServiceClient.Me.Events[id].Request().GetAsync();
            var eventResult = new MyEvent
            {
                Id = requestEvent.Id,
                Subject = requestEvent.Subject,
                Body = requestEvent.Body.Content,
                Location = requestEvent.Location.DisplayName,
                Start = DateTime.SpecifyKind(DateTime.Parse(requestEvent.Start.DateTime), requestEvent.Start.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local).ToLocalTime(),
                End = DateTime.SpecifyKind(DateTime.Parse(requestEvent.End.DateTime), requestEvent.End.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local).ToLocalTime(),
            };
            return eventResult;
        }
        catch
        {
            return null;
        }
    
    }
    1. Add a DeleteEvent function to the MyEventsRepository class to delete an event.
    public async Task DeleteEvent(string id)
    {
        try
        {
            var graphServiceClient = await GetGraphServiceAsync();
            await graphServiceClient.Me.Events[id].Request().DeleteAsync();
        }
        catch
        {
        }
        return;
    
    }
    1. Add an AddEvent function to the MyEventsRepository class to create a new event.
    public async Task AddEvent(MyEvent myEvent)
    {
        try
        {
            var graphServiceClient = await GetGraphServiceAsync();
            var requestEvent = new Microsoft.Graph.Event
            {
                Subject = myEvent.Subject,
                Start = new DateTimeTimeZone() { DateTime = myEvent.Start.ToString(), TimeZone = DateTimeKind.Local.ToString() },
                End = new DateTimeTimeZone { DateTime = myEvent.End.ToString(), TimeZone = DateTimeKind.Local.ToString() },
                Location = new Microsoft.Graph.Location { DisplayName = myEvent.Location },
                Body = new ItemBody { Content = myEvent.Body }
            };
            await graphServiceClient.Me.Events.Request().AddAsync(requestEvent);
        }
        catch
        {
        }
        return;
    }
    1. Finally, add a Search function to the MyEventsRepository class to add search functionality:
    public async Task<List<MyEvent>> Search(string searchTerm)
    {
        try
        {
            var graphServiceClient = await GetGraphServiceAsync();
            var requestEvents = await graphServiceClient.Me.Events.Request().Filter(string.Format("startswith(subject,+'{0}')", searchTerm)).GetAsync();
            var eventsResults = requestEvents.CurrentPage.Select(x => new MyEvent
            {
                Id = x.Id,
                Subject = x.Subject,
                Body = x.Body.Content,
                Location = x.Location.DisplayName,
                Start = DateTime.SpecifyKind(DateTime.Parse(x.Start.DateTime), x.Start.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local),
                End = DateTime.SpecifyKind(DateTime.Parse(x.End.DateTime), x.End.TimeZone == "UTC" ? DateTimeKind.Utc : DateTimeKind.Local),
            }).OrderBy(x=>x.Start).ToList();
            return eventsResults;
        }
        catch
        {
            return null;
        }
    }

At this point you have created the repository that will be used to talk to the Microsoft Graph SDK.

Exercise 4: Code the MVC Application

In this exercise, you will code the CalendarController of the MVC application to display events as well as adding behavior for adding and deleting events.

  1. Right-click the Controllers folder and select Add/Controller.

  2. In the Add Scaffold dialog, select MVC 5 Controller - Empty.

  3. Click Add.

  4. When prompted for a name, enter CalendarController.

  5. Click Add.

  6. Within the CalendarController file, use the following using statements instead of the old using statements :

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Web;
    using System.Web.Mvc;
    using Office365Calendar.Models; 
    using System.Threading.Tasks;
  7. Within the CalendarController class, add the following field to get a reference to the repository you previously created:

    MyEventsRepository _repo = new MyEventsRepository();
  8. Within the CalendarController class, add a route handler and view to list all the events:

  9. Replace the Index method with the following code to read events.

````c#
[Authorize]
public async Task<ActionResult> Index(int? pageNumber)
{
    // setup paging
    const int pageSize = 5;
    if (pageNumber == null)
        pageNumber = 1;

    List<MyEvent> events = null;
    events = await _repo.GetEvents((int)pageNumber - 1, pageSize);
    ViewBag.pageNumber = pageNumber;
    if(events != null)
        ViewBag.morePagesAvailable = events.Count < pageSize ? false : true;

    return View(events);
}
````

> Notice how the route handler takes in an optional parameter for the page number. This will be used to implement paging for the controller. Right now the page size is small, set to 5, for demonstration purposes.
  1. Finally, update the view to display the results. 1. Within the CalendarController class, right click the View(events) at the end of the Index() action method and select Add View. 1. Within the Add View dialog, set the following values:

    1. View Name: Index.
    2. Template: Empty (without model).

    Leave all other fields blank & unchecked.

    1. Click Add. 1. Within the Views/Calendar/Index.cshtml file, delete all the code in the file and replace it with the following code:
    @model IEnumerable<Office365Calendar.Models.MyEvent>
    @{ ViewBag.Title = "My Events"; }
    <h2>My Events</h2>
    <p>
        @Html.ActionLink("Create New", "Create") |
        @Html.ActionLink("Search Events", "Search")
    </p>
    <table id="eventsTable" class="table table-striped table-bordered">
        <tr>
            <th>@Html.DisplayNameFor(model => model.Subject)</th>
            <th>@Html.DisplayNameFor(model => model.Start)</th>
            <th>@Html.DisplayNameFor(model => model.End)</th>
            <th>@Html.DisplayNameFor(model => model.Location)</th>
            <th></th>
        </tr>
        @foreach (var item in Model)
        {
            <tr>
                <td>@Html.DisplayFor(modelItem => item.Subject)</td>
                <td>@Html.DisplayFor(modelItem => item.Start)</td>
                <td>@Html.DisplayFor(modelItem => item.End)</td>
                <td>@Html.DisplayFor(modelItem => item.Location)</td>
                <td>
                    @Html.ActionLink("Details", "Details", new { id = item.Id }) |
                    @Html.ActionLink("Delete", "Delete", new { id = item.Id })
                </td>
            </tr>
        }
    </table>
    <div class="row">
        <h4>Paging Control</h4>
        <div class="btn btn-group-sm">
            @{
                var pageLinkAttributes = new Dictionary<string, object> { { "class", "btn btn-default" } };
    
                int pageNumber = ViewBag.pageNumber;
    
                // do prev link if not on first page
                if (pageNumber > 1)
                {
                    var routeValues = new RouteValueDictionary { { "pageNumber", pageNumber - 1 } };
                    @Html.ActionLink("Previous Page", "Index", "Calendar", routeValues, pageLinkAttributes);
                }		
    
                // do next link if current page = max page size
                if (ViewBag.morePagesAvailable)
                {
                    var routeValues = new RouteValueDictionary { { "pageNumber", pageNumber + 1 } };
                    @Html.ActionLink("Next Page", "Index", "Calendar", routeValues, pageLinkAttributes);
                }
            }
        </div>
    </div>
  2. Test the new view:

  3. In Visual Studio, hit F5 to begin debugging.

Note: If you receive an error that indicates ASP.NET could not connect to the SQL database, please see the SQL Server Database Connection Error Resolution document to quickly resolve the issue.

  1. When prompted, log in with your Organizational Account.
  2. Once the application is loaded click the Events link in the top menu bar.
  3. Verify that your application displays calendar events from your Office 365 account.
![Screenshot of the previous step](Images/04.png)
  1. Close the browser window, terminate the debugging session and return to Visual Studio.

  2. Add a route handler to delete an event:

  3. In the CalendarController.cs file, add an action method named Delete using the following code to delete an event.

````c#
[Authorize]
public async Task<ActionResult> Delete(string id)
{
    if (id != null)
    {
        await _repo.DeleteEvent(id);
    }

    return Redirect("/Calendar");

}
````
  1. Add a route handler and views to handle creating events:
  2. In the CalendarController.cs file, add an action method named Create using the following code to create a new event. Notice how you are adding two items, when the create form is requested (the HttpGet option) and one for when the form is submitted (the HttpPost option).
````c#
[HttpGet]
[Authorize]
public async Task<ActionResult> Create()
{
    var myEvent = new MyEvent
    {
        Start = DateTimeOffset.Now,
        End = DateTimeOffset.Now.AddDays(1)
    };

    return View(myEvent);
}

[HttpPost]
[Authorize]
public async Task<ActionResult> Create(MyEvent myEvent)
{

    await _repo.AddEvent(myEvent);
    return Redirect("/Calendar");
}
````
  1. Within the CalendarController class, right click the View(myEvent) at the end of the Create() action method and select Add View.
  2. In the Add View dialog, set the following options on the dialog and click Add. + View name: Create + Template: Create + Model class: MyEvent (Office365Calendar.Models) + Create as partial view: unchecked + Reference script libraries: unchecked + Use a layout page: checked + Click Add
  3. Open the Create.cshtml file. Delete all the code in the file and replace it with the following code:
````html
@model Office365Calendar.Models.MyEvent
@{
    ViewBag.Title = "Create";
}	
<h2>Create</h2>	
@using (Html.BeginForm()) 
{    
    <div class="form-horizontal">
        <h4>MyEvent</h4>
        <hr />
        @Html.ValidationSummary(true, "", new { @class = "text-danger" })
        <div class="form-group">
            @Html.LabelFor(model => model.Subject, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.Subject, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.Subject, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            @Html.LabelFor(model => model.Start, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.Start, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.Start, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            @Html.LabelFor(model => model.End, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.End, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.End, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            @Html.LabelFor(model => model.Location, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.Location, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.Location, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            @Html.LabelFor(model => model.Body, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.Body, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.Body, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            <div class="col-md-offset-2 col-md-10">
                <input type="submit" value="Create" class="btn btn-default" />
            </div>
        </div>
    </div>
}

<div>
    @Html.ActionLink("Back to List", "Index")
</div>
````
  1. Test the new view:
  2. In Visual Studio, hit F5 to begin debugging.

Note: If you receive an error that indicates ASP.NET could not connect to the SQL database, please see the SQL Server Database Connection Error Resolution document to quickly resolve the issue.

  1. When Prompted, log in with your Organizational Account.
  2. Once the application is loaded click the Events link in the top menu bar.
  3. Click the Delete link. The event would be deleted successfully.
  4. Click the Create New link. You should see the form below. Fill the form out to add a new item:
![Screenshot of the previous step](Images/05.png)
  1. Close the browser window, terminate the debugging session and return to Visual Studio.

  2. Add a route handler and view to handle showing the details of a selected event:

  3. In the CalendarController.cs file, add an action method named Details using the following code to view an event.

````c#
[Authorize]
public async Task<ActionResult> Details(string id)
{

    MyEvent myEvent = null;
    myEvent = await _repo.GetEvent(id);
    return View(myEvent);
}
````
  1. Within the CalendarController class, right click the View(myEvent) at the end of the Details() action method and select Add View.

  2. In the Add View dialog, set the following options on the dialog and click Add. + View name: Details + Template: Details + Model class: MyEvent (Office365Calendar.Models) + Create as partial view: unchecked + Reference script libraries: unchecked + Use a layout page: checked + Click Add

  3. Open the Details.cshtml file. Find the following code and remove it:

       @Html.ActionLink("Edit", "Edit", new { id = Model.Id }) |
  4. Test the new view:

  5. In Visual Studio, hit F5 to begin debugging.

Note: If you receive an error that indicates ASP.NET could not connect to the SQL database, please see the SQL Server Database Connection Error Resolution document to quickly resolve the issue.

  1. When Prompted, log in with your Organizational Account.
  2. Once the application is loaded click the Events link in the top menu bar.
  3. Click the Details link for one of the items.
![Screenshot of the previous step](Images/06.png)
  1. Close the browser window, terminate the debugging session and return to Visual Studio.

  2. Add a route handler and view to handle search for events:

  3. In the CalendarController.cs file, add two action methods named Search using the following code to search events.

````c#
[HttpGet]
[Authorize]
public async Task<ActionResult> Search()
{
    return View();
}

[HttpPost]
[Authorize]
public async Task<ActionResult> Search(string searchTerm)
{
    var events = await _repo.Search(searchTerm);
    return View(events);
}
````
  1. Within the CalendarController class, right click the View() at the end of the Search() action method and select Add View.
  2. In the Add View dialog, set the following options on the dialog and click Add. + View name: Search + Template: Empty + Create as partial view: unchecked + Reference script libraries: unchecked + Use a layout page: checked + Click Add
  3. Open the generated Search.cshtml view and replace the markup with the following code:
````html
@model IEnumerable<Office365Calendar.Models.MyEvent>
@{
    ViewBag.Title = "Search";
}
<h2>Search</h2>
@using (Html.BeginForm("Search", "Calendar", FormMethod.Post))
{
    <p>
        <div class="form-horizontal">
            <div class="form-group">
                <div class="col-md-10">
                    <input type="text" id="searchTerm" name="searchTerm" class="form-control" /><br />
                    <button type="submit">search for events</button>
                </div>
            </div>
        </div>
    </p>
}
<table class="table">
    <tr>
        <th>@Html.DisplayNameFor(model => model.Subject)</th>
        <th>@Html.DisplayNameFor(model => model.Start)</th>
        <th>@Html.DisplayNameFor(model => model.End)</th>
        <th></th>
    </tr>
    @if (Model != null)
    {
        foreach (var item in Model)
        {
            <tr>
                <td>@Html.DisplayFor(modelItem => item.Subject)</td>
                <td>@Html.DisplayFor(modelItem => item.Start)</td>
                <td>@Html.DisplayFor(modelItem => item.End)</td>
                <td>
                    @Html.ActionLink("Details", "Details", new { id = item.Id }) |
                    @Html.ActionLink("Delete", "Delete", new { id = item.Id })
                </td>
            </tr>
        }
    }
</table>
<p>
    @Html.ActionLink("Back to List", "Index")
</p>
````
  1. Test the new view:
  2. In Visual Studio, hit F5 to begin debugging.

Note: If you receive an error that indicates ASP.NET could not connect to the SQL database, please see the SQL Server Database Connection Error Resolution document to quickly resolve the issue.

  1. When Prompted, log in with your Organizational Account.
  2. Once the application is loaded click the Events link in the top menu bar.
  3. Click the Search Events link and enter a search term on the next page. You should see a filtered set of results:
![Screenshot of the previous step](Images/07.png)
  1. Close the browser window, terminate the debugging session and return to Visual Studio.

Congratulations! You have completed working with the Microsoft Graph APi for Calendar.