Java:ews-java-api获取Exchange Web Services (EWS)会议日程

简介: Java:ews-java-api获取Exchange Web Services (EWS)会议日程

文档

打开官方文档,发现全是C#的示例

我们的实际情况是:

  • 开发端在MacOS
  • 调试在Windows虚拟机,通过虚拟机连接内网,获取内网部署的Exchange Web Services数据
  • 最终服务端需要运行在Linux上

为了多端兼容,而且对C#不熟悉,还好网上还能搜索到部分Java的示例,所以使用Java接口进行二次开发

主要开发功能是获取用户的日历数据,同步到公司系统,方便查看统计数据

Maven

<dependency>
    <groupId>com.microsoft.ews-java-api</groupId>
    <artifactId>ews-java-api</artifactId>
    <version>2.0</version>
</dependency>

部分代码

业务接口

package com.example.demo.service;
import microsoft.exchange.webservices.data.core.service.item.Appointment;
import microsoft.exchange.webservices.data.core.service.item.Contact;
import java.util.Date;
import java.util.List;
public interface OutlookService {
    /**
     * 翻页获取会议
     * @param emailName
     * @param pageNumber
     * @param pageSize
     * @return
     */
    List<Appointment> getAppointmentListForPage(String emailName, int pageNumber, int pageSize);
    /**
     * 按照时间区间获取会议
     * @param emailName
     * @param startDate
     * @param endDate
     * @return
     */
    List<Appointment> getAppointmentListForDate(String emailName, Date startDate, Date endDate);
    /**
     * 获取所有联系人列表
     * @param emailName
     * @return
     */
    List<Contact> getAllContactList(String emailName);
}

接口实现

package com.example.demo.service.impl;
import com.example.demo.service.OutlookService;
import lombok.extern.slf4j.Slf4j;
import microsoft.exchange.webservices.data.core.ExchangeService;
import microsoft.exchange.webservices.data.core.PropertySet;
import microsoft.exchange.webservices.data.core.enumeration.misc.ExchangeVersion;
import microsoft.exchange.webservices.data.core.enumeration.property.WellKnownFolderName;
import microsoft.exchange.webservices.data.core.service.item.Appointment;
import microsoft.exchange.webservices.data.core.service.item.Contact;
import microsoft.exchange.webservices.data.core.service.item.Item;
import microsoft.exchange.webservices.data.credential.ExchangeCredentials;
import microsoft.exchange.webservices.data.credential.WebCredentials;
import microsoft.exchange.webservices.data.property.complex.FolderId;
import microsoft.exchange.webservices.data.property.complex.Mailbox;
import microsoft.exchange.webservices.data.search.CalendarView;
import microsoft.exchange.webservices.data.search.FindItemsResults;
import microsoft.exchange.webservices.data.search.ItemView;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
/**
 * exchange 版本 2019
 */
@Service
@Slf4j
public class OutlookServiceImpl implements OutlookService {
    @Value("${outlook.exchangeUrl}")
    private String exchangeUrl;
    @Value("${outlook.username}")
    private String username;
    @Value("${outlook.password}")
    private String password;
    /**
     * 获取服务
     *
     * @return
     */
    private ExchangeService getExchangeService() {
        ExchangeService service = new ExchangeService(ExchangeVersion.Exchange2010_SP2);
        ExchangeCredentials credentials = new WebCredentials(username, password);
        service.setCredentials(credentials);
        try {
            service.setUrl(new URI(exchangeUrl));
        } catch (URISyntaxException e) {
            e.printStackTrace();
        }
        service.setTraceEnabled(true);
        return service;
    }
    /**
     * 在 Exchange 中使用 EWS 获取约会和会议
     * https://learn.microsoft.com/zh-cn/exchange/client-developer/exchange-web-services/how-to-get-appointments-and-meetings-by-using-ews-in-exchange
     * <p>
     * EWS Java API 的基本使用
     * https://blog.csdn.net/m0_37972348/article/details/83960690
     */
    @Override
    public List<Appointment> getAppointmentListForPage(String emailName, int pageNumber, int pageSize) {
        List<Appointment> list = new ArrayList<>();
        List<Item> items = this.getItemsForPage(emailName, WellKnownFolderName.Calendar, pageNumber, pageSize);
        for (Item item : items) {
            if (item instanceof Appointment) {
                Appointment appointment = ((Appointment) item);
                try {
                    appointment.load(PropertySet.FirstClassProperties);
                } catch (Exception e) {
                    e.printStackTrace();
                }
                list.add(appointment);
            }
        }
        return list;
    }
    /**
     * 获取所有联系人列表
     */
    @Override
    public List<Contact> getAllContactList(String emailName) {
        List<Contact> list = new ArrayList<>();
        int pageNumber = 1;
        int pageSize = 50;
        while (true) {
            List<Item> items = this.getItemsForPage(emailName, WellKnownFolderName.Contacts, pageNumber, pageSize);
            if (items == null || items.size() == 0) {
                break;
            }
            for (Item item : items) {
                if (item instanceof Contact) {
                    Contact contactItem = ((Contact) item);
                    list.add(contactItem);
                }
            }
            if(items.size() < pageSize){
                break;
            }
            pageNumber++;
        }
        return list;
    }
    public List<Item> getItemsForPage(String emailName, WellKnownFolderName folderName, int pageNumber, int pageSize) {
        ExchangeService service = this.getExchangeService();
        List<Item> list = new ArrayList<>();
        int offset = (pageNumber - 1) * pageSize;
        System.out.println("pageNumber: " + pageNumber);
        System.out.println("offset: " + offset);
        ItemView view = new ItemView(pageSize, offset);
        FolderId folderId = new FolderId(folderName, new Mailbox(emailName));
        FindItemsResults<Item> findResults = null;
        try {
            findResults = service.findItems(folderId, view);
        } catch (Exception e) {
            log.info("items is empty");
        }
        if (findResults == null) {
            return list;
        }
        int totalCount = findResults.getTotalCount();
        System.out.println("totalCount: " + totalCount);
        //MOOOOOOST IMPORTANT: load messages' properties before
        try {
            service.loadPropertiesForItems(findResults, PropertySet.FirstClassProperties);
        } catch (Exception e) {
            log.info("service.loadPropertiesForItems error");
        }
        list = findResults.getItems();
        return list;
    }
    @Override
    public List<Appointment> getAppointmentListForDate(String emailName, Date startDate, Date endDate) {
        ExchangeService service = this.getExchangeService();
        List<Appointment> list = new ArrayList<>();
        CalendarView calendarView = new CalendarView(startDate, endDate);
        FolderId folderId = new FolderId(WellKnownFolderName.Calendar, new Mailbox(emailName));
        FindItemsResults<Appointment> findResults = null;
        try {
            findResults = service.findAppointments(folderId, calendarView);
        } catch (Exception e) {
            log.info("Appointments is empty");
        }
        if (findResults == null) {
            return list;
        }
        int totalCount = findResults.getTotalCount();
        System.out.println("totalCount: " + totalCount);
        list = findResults.getItems();
        for (Appointment appointment : list) {
            try {
                appointment.load(PropertySet.FirstClassProperties);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return list;
    }
}

参考


相关文章
Understanding RESTful API and Web Services: Key Differences and Use Cases
在现代软件开发中,RESTful API和Web服务均用于实现系统间通信,但各有特点。RESTful API遵循REST原则,主要使用HTTP/HTTPS协议,数据格式多为JSON或XML,适用于无状态通信;而Web服务包括SOAP和REST,常用于基于网络的API,采用标准化方法如WSDL或OpenAPI。理解两者区别有助于选择适合应用需求的解决方案,构建高效、可扩展的应用程序。
Python 高级编程与实战:深入理解 Web 开发与 API 设计
在前几篇文章中,我们探讨了 Python 的基础语法、面向对象编程、函数式编程、元编程、性能优化、调试技巧以及数据科学和机器学习。本文将深入探讨 Python 在 Web 开发和 API 设计中的应用,并通过实战项目帮助你掌握这些技术。
Gemini Coder:基于 Google Gemini API 的开源 Web 应用生成工具,支持实时编辑和预览
Gemini Coder 是一款基于 Google Gemini API 的 AI 应用生成工具,支持通过文本描述快速生成代码,并提供实时代码编辑和预览功能,简化开发流程。
187 38
Gemini Coder:基于 Google Gemini API 的开源 Web 应用生成工具,支持实时编辑和预览
java spring 项目若依框架启动失败,启动不了服务提示端口8080占用escription: Web server failed to start. Port 8080 was already in use. Action: Identify and stop the process that’s listening on port 8080 or configure this application to listen on another port-优雅草卓伊凡解决方案
java spring 项目若依框架启动失败,启动不了服务提示端口8080占用escription: Web server failed to start. Port 8080 was already in use. Action: Identify and stop the process that’s listening on port 8080 or configure this application to listen on another port-优雅草卓伊凡解决方案
96 7
小团队 CI/CD 实践:无需运维,Java Web应用的自动化部署
本文介绍如何使用GitHub Actions和阿里云Kubernetes(ACK)实现Java Web应用的自动化部署。通过CI/CD流程,开发人员无需手动处理复杂的运维任务,从而提高效率并减少错误。文中详细讲解了Docker与Kubernetes的概念,并演示了从创建Kubernetes集群、配置容器镜像服务到设置GitHub仓库Secrets及编写GitHub Actions工作流的具体步骤。最终实现了代码提交后自动构建、推送镜像并部署到Kubernetes集群的功能。整个过程不仅简化了部署流程,还确保了应用在不同环境中的稳定运行。
128 9
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
121 10
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
127 7
Spring Boot 入门:简化 Java Web 开发的强大工具
Java Web 应用中,资源文件的位置和加载方式
在Java Web应用中,资源文件如配置文件、静态文件等通常放置在特定目录下,如WEB-INF或classes。通过类加载器或Servlet上下文路径可实现资源的加载与访问。正确管理资源位置与加载方式对应用的稳定性和可维护性至关重要。
113 7
理解Session和Cookie:Java Web开发中的用户状态管理
理解Session和Cookie:Java Web开发中的用户状态管理
138 4
Python Web开发者必看!AJAX、Fetch API实战技巧,让前后端交互如丝般顺滑!
在Web开发中,前后端的高效交互是提升用户体验的关键。本文通过一个基于Flask框架的博客系统实战案例,详细介绍了如何使用AJAX和Fetch API实现不刷新页面查看评论的功能。从后端路由设置到前端请求处理,全面展示了这两种技术的应用技巧,帮助Python Web开发者提升项目质量和开发效率。
106 1
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等