我认为答案是肯定的,但我不知道该怎么做。我一直在使用API从设备上存储的日历中获取事件。 e.g。
public static ArrayList<MINCalendarEvent> queryEvents(long startMillis, long endMillis) throws MINPermissionException
{
if ( ContextCompat.checkSelfPermission(MINMainActivity.getSharedInstance(), Manifest.permission.WRITE_CALENDAR) != PackageManager.PERMISSION_GRANTED )
{
throw new MINPermissionException(NO_PERMISSION);
}
else
{
ArrayList<MINCalendarEvent> eventArray = new ArrayList<MINCalendarEvent>();
Cursor cur = CalendarContract.Instances.query(MINMainActivity.getSharedInstance().getContentResolver(), EVENT_PROJECTION, startMillis, endMillis);
int numItems = cur.getCount();
Log.d("MINCalendarUtils.queryEvents", "Number of events retrieved: " + numItems);
while (cur.moveToNext())
{
MINCalendarEvent event = new MINCalendarEvent();
event.calendarID = cur.getLong(EVENT_PROJECTION_CALENDAR_ID);
event.organizer = cur.getString(EVENT_PROJECTION_ORGANIZER);
event.title = cur.getString(EVENT_PROJECTION_TITLE);
event.eventLocation = cur.getString(EVENT_PROJECTION_EVENT_LOCATION);
event.description = cur.getString(EVENT_PROJECTION_DESCRIPTION);
event.dtStart = cur.getLong(EVENT_PROJECTION_DTSTART);
event.dtEnd = cur.getLong(EVENT_PROJECTION_DTEND);
event.eventTimeZone = cur.getString(EVENT_PROJECTION_EVENT_TIMEZONE);
event.eventEndTimeZone = cur.getString(EVENT_PROJECTION_EVENT_END_TIMEZONE);
event.duration = cur.getString(EVENT_PROJECTION_DURATION);
event.allDay = (cur.getInt(EVENT_PROJECTION_ALL_DAY) != 0);
event.rRule = cur.getString(EVENT_PROJECTION_RRULE);
event.rDate = cur.getString(EVENT_PROJECTION_RDATE);
event.availability = cur.getInt(EVENT_PROJECTION_AVAILABILITY);
event.guestsCanModify = (cur.getInt(EVENT_PROJECTION_GUESTS_CAN_MODIFY) != 0);
event.guestsCanInviteOthers = (cur.getInt(EVENT_PROJECTION_GUESTS_CAN_INVITE_OTHERS) != 0);
event.guestsCanSeeGuests = (cur.getInt(EVENT_PROJECTION_GUESTS_CAN_SEE_GUESTS) != 0);
eventArray.add(event);
}
return eventArray;
}
}
这很有效。问题是我需要查询存储在服务器上的共享日历,用户也没有权限。我的应用创建了一个LOCAL日历。我需要访问用户也没有权限的共享,并将存储在共享日历中的事件与LOCAL日历(非同步)同步。我假设我可以使用服务帐户访问共享日历。
我已成功创建了一个服务帐户,并将该帐户添加到共享日历中。现在是什么????
似乎有几种使用服务帐户访问日历活动的方法,但我完全感到困惑。显然,我想使用我一直在使用的API,但我认为它只适用于与设备同步的日历。
我使用&#34; GoogleCredentials&#34;进行了调查但我需要一些示例源代码来完成这项工作。首先,当我创建服务帐户时,我使用JSON而不是p12导出它。我不知道如何使用它。 API似乎需要p12。我对它们如何使用凭证完全感到困惑。这是我开始使用的示例源:
//String emailAddress = "123456789000-abc123def456@developer.gserviceaccount.com";
GoogleCredential credential = null;
JsonFactory JSON_FACTORY = JacksonFactory.getDefaultInstance();
HttpTransport httpTransport = null;
try
{
httpTransport = GoogleNetHttpTransport.newTrustedTransport();
credential = new GoogleCredential.Builder()
.setTransport(httpTransport)
.setJsonFactory(JSON_FACTORY)
.setServiceAccountId(emailAddress)
.setServiceAccountPrivateKeyFromP12File(new File("MyProject.p12"))
.setServiceAccountScopes(Collections.singleton(SQLAdminScopes.SQLSERVICE_ADMIN))
.build();
}
catch (GeneralSecurityException e)
{
e.printStackTrace();
}
catch (IOException e)
{
e.printStackTrace();
}
有几个问题。 1)我无法通过事件来编译。我似乎无法让导入工作:
import com.google.api.services.sqladmin.SQLAdminScopes;
假设我可以过去,现在是什么。我不确定如何使用凭据访问远程日历。我需要的是从共享日历中获取事件列表。
我也一直在查看以下链接的源代码以获取路线,但它没有使用服务帐户:https://developers.google.com/google-apps/calendar/quickstart/android
此外,有没有办法将更改挂钩到基于服务器的共享日历,以便在共享日历发生更改时收到ping?
任何帮助????
根据Andres的回答,我将以下代码放在一起:
public static void calendarAuthenticate()
{
Thread thread = new Thread(new Runnable()
{
@Override
public void run()
{
try
{
MINAppConfiguration appConfig = MINAppConfiguration.getSharedInstance();
// Application Name
appName = appConfig.getCurrentAppInfo().appName;
// Directory to store user credentials for this application;
//dataStoreDir = new File(appConfig.appDirectoryOnDevice);
// Instance of the JSON factory
jsonFactory = JacksonFactory.getDefaultInstance();
// Instance of the scopes required
scopes = new ArrayList<String>();
scopes.add(CalendarScopes.CALENDAR_READONLY);
// Http transport creation
httpTransport = AndroidHttp.newCompatibleTransport();
java.io.File licenseFile = getSecretFile();
GoogleCredential credential = new GoogleCredential.Builder()
.setTransport(httpTransport)
.setJsonFactory(jsonFactory)
.setServiceAccountId("account-1@handy-contact-762.iam.gserviceaccount.com")
.setServiceAccountScopes(scopes)
.setServiceAccountPrivateKeyFromP12File(licenseFile)
.build();
com.google.api.services.calendar.Calendar.Builder builder = new com.google.api.services.calendar.Calendar.Builder(httpTransport, jsonFactory, credential);
builder.setApplicationName(appName);
com.google.api.services.calendar.Calendar client = builder.build();
// List the next 10 events from the target calendar.
DateTime now = new DateTime(System.currentTimeMillis());
com.google.api.services.calendar.Calendar.Events.List list = client.events().list("mobilityinitiative.com_qfvbdrk368f9la06hacb4bduos@group.calendar.google.com");
list.setMaxAttendees(10);
list.setTimeMin(now);
list.setOrderBy("startTime");
list.setSingleEvents(true);
Events events = list.execute();
List<Event> items = events.getItems();
if (items.size() == 0)
{
Log.d(TAG, "No upcoming events found.");
}
else
{
Log.d(TAG, "Upcoming events");
for (Event event : items) {
DateTime start = event.getStart().getDateTime();
if (start == null) {
start = event.getStart().getDate();
}
Log.d(TAG, event.getSummary() + " (" + start + ")\n");
}
}
}
catch (GeneralSecurityException e)
{
e.printStackTrace();
}
catch (IOException e)
{
e.printStackTrace();
}
}
});
thread.start();
}
public static java.io.File getSecretFile()
{
File f = new File(MINMainActivity.getSharedInstance().getCacheDir()+ "/" +"google_calendar_secret.p12");
if (f.exists())
{
f.delete();
}
try
{
InputStream is = MINMainActivity.getSharedInstance().getAssets().open("google_calendar_secret.p12");
int size = is.available();
byte[] buffer = new byte[size];
is.read(buffer);
is.close();
FileOutputStream fos = new FileOutputStream(f);
fos.write(buffer);
fos.close();
}
catch (Exception e)
{
throw new RuntimeException(e);
}
return f;
}
几点说明:
答案 0 :(得分:4)
我不是Android专家,但以下任何一种方法都适用于您的用例。
您的方法:使用服务帐户,将避免让用户对您的应用程序进行身份验证。您可能希望阅读有关“Delegating domain-wide authority”的信息,这样您的应用就可以作为域中的用户(也称为“模仿”用户)进行API调用。我还发现这个SO有帮助。
另一种方法:使用Acl.Insert资源。这将需要每次登录到您的应用程序时进行用户身份验证。这是关于如何实现这一点的example。
从上面的示例源中,将范围设置为Calendar scope而不是SQL Admin,如下所示:
GoogleCredential credential = new GoogleCredential.Builder()
...
.setServiceAccountScopes(CalendarScopes.CALENDAR)
.setServiceAccountPrivateKeyFromP12File(xxxx)
.build();
希望这会有所帮助并祝你好运!