How to do paging with Exchange Web Services CalendarView

前端 未结 3 592
故里飘歌
故里飘歌 2021-02-07 12:20

If I do this:

_calendar = (CalendarFolder)Folder.Bind(_service, WellKnownFolderName.Calendar);

var findResults = _calendar.FindAppointments(
    new CalendarVie         


        
3条回答
  •  名媛妹妹
    2021-02-07 12:59

    CalendarView is not actually derived from PagedView, so all of the paging logic that you expect isn't possible. MaxItemsReturned is more of an upper limit than a page size. The error that's returned is more relevant to the PagedView derived view types.

    I played around with some PowerShell to emulate paging by rolling the CalendarView window based on the last item returned, but unfortunately the logic behind the CalendarView and Appointment expansion make it impossible to get exactly what you need. Basically as it does the expansion, it's going to stop at "N" items, but you might have more than one appointment that starts at the exact same time and it may give you one, but not the rest. Also, any appointments that overlap the window will get included, so the below code would go into an infinite loop if you had 50 appointments on the calendar that all had the same start time.

    Add-Type -Path "C:\Program Files\Microsoft\Exchange\Web Services\1.2\Microsoft.Exchange.WebServices.dll"
    
    $service = New-Object Microsoft.Exchange.WebServices.Data.ExchangeService
    $cred = New-Object Microsoft.Exchange.WebServices.Data.WebCredentials ($user , $passwd)
    $service.UseDefaultCredentials = $false
    $service.Credentials = $cred
    $service.AutodiscoverUrl($user)
    
    $num=50
    $total=0
    $propsetfc = [Microsoft.Exchange.WebServices.Data.BasePropertySet]::FirstClassProperties
    $calfolder = [Microsoft.Exchange.WebServices.Data.WellKnownFolderName]::Calendar
    
    $service.UserAgent = "EWSCalViewTest"
    $calview = New-Object Microsoft.Exchange.WebServices.Data.CalendarView("1/1/2012","12/31/2012", $num)
    $calview.PropertySet = $propsetfc
    
    do {
        $findresults = $service.FindAppointments($calfolder,$calview)
        write-host  "Found:" $findresults.Items.Count "of" $findresults.TotalCount
        $calview.StartDate = $findresults.Items[$findresults.Items.Count-1].Start
        $total+=$findresults.Items.Count
    } while($findresults.MoreAvailable)
    write-host $total "total found (including dups)"
    

    Unfortunately the expansion and overlap logic mean you'll get duplicates this way, at least one duplicate for each call beyond the first.

    If I had to write code using CalendarView, I'd probably use a MaxItemsReturned of 1000 (this is also the limit that throws you into the error condition if you don't specify MaxItemsReturned). If you get them all in one call, you're good. If you have to make a second call, then you'll have to do some extra work to dedup the result set. I'd also try to limit the burden on the server by using as narrow of a date window as possible in the CalendarView since you're asking Exchange to calculate the expansion of recurring appointments across the entire time span. It can be a fairly expensive operation for the server.

提交回复
热议问题