ALF-9156 Convert the last of the Calendar CRUD webscripts to Java (except for recurring parts)

git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@28866 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
Nick Burch
2011-07-07 20:57:31 +00:00
parent dfa5971d86
commit 9bdeee7986
9 changed files with 284 additions and 239 deletions

View File

@@ -11,12 +11,12 @@
"where": "${result.where?html}",
"startAt": {
"iso8601": "${xmldate(result.from)}",
"legacyDate": "${result.from?string("M/d/yyyy")}",
"legacyDate": "${result.from?string("yyyy-MM-dd")}",
"legacyTime": "${result.from?string("HH:mm")}",
},
"endAt": {
"iso8601": "${xmldate(result.to)}",
"legacyDate": "${result.to?string("M/d/yyyy")}",
"legacyDate": "${result.to?string("yyyy-MM-dd")}",
"legacyTime": "${result.to?string("HH:mm")}",
},
"allday": "${result.allday?string}",

View File

@@ -7,8 +7,24 @@
"summary" : "${result.summary?js_string}",
"location" : "${result.location?js_string}",
"description" : "${result.description?js_string}",
"dtstart" : "${result.dtstart}",
"dtend" : "${result.dtend}",
"startAt": {
"iso8601": "${xmldate(result.dtstart)}",
"legacyDateTime": "${result.dtstart?string("yyyy-MM-dd")}T${result.dtstart?string("HH:mm")}",
"legacyDate": "${result.dtstart?string("yyyy-MM-dd")}",
"legacyTime": "${result.dtstart?string("HH:mm")}",
},
"endAt": {
"iso8601": "${xmldate(result.dtend)}",
"legacyDateTime": "${result.dtend?string("yyyy-MM-dd")}T${result.dtend?string("HH:mm")}",
"legacyDate": "${result.dtend?string("yyyy-MM-dd")}",
"legacyTime": "${result.dtend?string("HH:mm")}",
},
<#-- These are the old ones we'll get rid of soon -->
"dtstart" : "${result.dtstart?string("yyyy-MM-dd")}T${result.dtstart?string("HH:mm")}",
"dtend" : "${result.dtend?string("yyyy-MM-dd")}T${result.dtend?string("HH:mm")}",
"uri" : "${result.uri}",
"allday" : "${result.allday?string}",
"tags" : "${result.tags}",

View File

@@ -1,198 +0,0 @@
<import resource="classpath:/alfresco/templates/webscripts/org/alfresco/slingshot/calendar/lib/calendar.lib.js">
/**
* Update event properties
* @method PUT
* @param uri {string} /calendar/event/{siteid}/{eventname}
*/
function getTemplateParams()
{
// Grab the URI parameters
var siteid = "" + url.templateArgs.siteid;
var eventname = "" + url.templateArgs.eventname;
var date = args["date"];
if (siteid === null || siteid.length === 0)
{
return null;
}
if (eventname === null || eventname.length === 0)
{
return null;
}
return {
"siteid": siteid,
"eventname": eventname,
"date": date
};
}
function main()
{
var params = getTemplateParams();
if (params === null)
{
return {
"error": "Invalid parameters"
};
}
// Get the site
var site = siteService.getSite(params.siteid);
if (site === null)
{
return {
"error": "Could find not specified site"
};
}
var eventsFolder = getCalendarContainer(site);
if (eventsFolder === null)
{
return {
"error": "Could not find specified calendar"
};
}
var editedEvent = eventsFolder.childByNamePath(params.eventname);
var event = editedEvent;
if (event === null)
{
return {
"error": "Could not find specified event to update"
};
}
if (editedEvent.properties["ia:recurrenceRule"] != null)
{
var prop = new Array();
var fromParts = params.date.split("-");
prop["ia:date"] = new Date(fromParts[0],fromParts[1] - 1,fromParts[2]);
editedEvent.createNode(null, "ia:ignoreEvent", prop, "ia:ignoreEventList");
var timestamp = new Date().getTime();
var random = Math.round(Math.random() * 10000);
event = eventsFolder.createNode(timestamp + "-" + random + ".ics", "ia:calendarEvent");
event.properties["ia:isOutlook"] = true;
}
if (json.get("docfolder")=='*NOT_CHANGE*')
{
if (editedEvent.properties["ia:docFolder"] != "")
event.properties["ia:docFolder"] = editedEvent.properties["ia:docFolder"];
}
else
{
if (json.get("docfolder") != "")
event.properties["ia:docFolder"] = json.get("docfolder");
}
var props = [
"what",
"desc",
"where"
];
var propsmap =
{
"what" : "ia:whatEvent",
"desc" : "ia:descriptionEvent",
"where" : "ia:whereEvent"
};
for (var i=0; i < props.length; i++)
{
var prop = props[i], value;
if (!json.isNull(prop))
{
value = json.get(prop);
event.properties[ propsmap[prop] ] = value;
}
}
if (!json.isNull("tags"))
{
var tags = String(json.get("tags")); // space delimited string
if (tags !== "")
{
var tagsArray = tags.split(" ");
event.tags = tagsArray;
}
else
{
event.tags = []; // reset
}
}
try
{
// Handle date formatting as a separate case
var from = json.get("from");
var to = json.get("to");
if (json.isNull("allday"))
{
from += " " + json.get("start");
to += " " + json.get("end");
allday = ''
}
from = new Date(from);
to = new Date(to);
event.properties["ia:fromDate"] = from;
event.properties["ia:toDate"] = to;
var pad = function (value, length)
{
value = String(value);
length = parseInt(length) || 2;
while (value.length < length)
{
value = "0" + value;
}
return value;
};
var fromIsoDate = from.getFullYear() + "-" + pad(from.getMonth() + 1) + "-" + pad(from.getDate());
var toIsoDate = to.getFullYear() + "-" + pad(to.getMonth() + 1) + "-" + pad(to.getDate());
var data =
{
title: json.get("what"),
page: json.get("page") + "?date=" + fromIsoDate
}
activities.postActivity("org.alfresco.calendar.event-updated", params.siteid, "calendar", jsonUtils.toJSONString(data));
}
catch(e)
{
if (logger.isLoggingEnabled())
{
logger.log(e);
}
}
// Saved data
// {"site":"testSite","page":"calendar","from":"Tuesday, 4 November 2008","to":"Tuesday, 4 November 2008"
// ,"what":"big lunchie","where":"somewhere","desc":"","fromdate":"Tuesday, 4 November 2008","start":"12
// :00","todate":"Tuesday, 4 November 2008","end":"13:00","tags":""}
event.save();
var savedData =
{
summary: json.get('what'),
location: json.get('where'),
description: json.get('desc'),
dtstart: fromIsoDate+ 'T' + json.get('start'),
dtend: toIsoDate + 'T' + json.get('end'),
allday: (json.isNull("allday")) ? "" : (json.get('allday') == 'on') ? true : "",
uri: "calendar/event/" + params.siteid + "/" + event.name + "?date=" + params.date,
tags: tags,
docfolder: event.properties["ia:docFolder"] == null ? "" : event.properties["ia:docFolder"]
}
return savedData;
}
model.result = main();

View File

@@ -1466,6 +1466,12 @@
parent="abstractCalendarWebScript">
</bean>
<!-- Edits an existing Calendar Event from the site calendar -->
<bean id="webscript.org.alfresco.slingshot.calendar.event.put"
class="org.alfresco.repo.web.scripts.calendar.CalendarEntryPut"
parent="abstractCalendarWebScript">
</bean>
<!-- Removes a Calendar Event from the site calendar -->
<bean id="webscript.org.alfresco.slingshot.calendar.event.delete"
class="org.alfresco.repo.web.scripts.calendar.CalendarEntryDelete"

View File

@@ -27,6 +27,7 @@ import java.util.Map;
import org.alfresco.repo.content.MimetypeMap;
import org.alfresco.service.cmr.activities.ActivityService;
import org.alfresco.service.cmr.calendar.CalendarEntry;
import org.alfresco.service.cmr.calendar.CalendarService;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.cmr.site.SiteInfo;
@@ -79,7 +80,7 @@ public abstract class AbstractCalendarWebScript extends DeclarativeWebScript
* Gets the date from the String, trying the various formats
* (New and Legacy) until one works...
*/
protected Date extractDate(String date)
protected Date parseDate(String date)
{
// Is there one at all?
if(date == null || date.length() == 0)
@@ -126,6 +127,51 @@ public abstract class AbstractCalendarWebScript extends DeclarativeWebScript
throw new WebScriptException(Status.STATUS_BAD_REQUEST, "Invalid date '" + date + "'");
}
/**
* Extracts the Start and End details, along with the All Day flag
* from the JSON, and returns if the event is all day or not
*/
protected boolean extractDates(CalendarEntry entry, JSONObject json) throws JSONException
{
boolean isAllDay = false;
if(json.has("startAt") && json.has("endAt"))
{
// New style ISO8601 dates
entry.setStart(parseDate(json.getString("startAt")));
entry.setEnd(parseDate(json.getString("endAt")));
if(json.has("allday"))
{
// TODO Handle All Day events properly, including timezones
isAllDay = true;
}
}
else if(json.has("allday"))
{
// Old style all-day event
entry.setStart(parseDate(getOrNull(json, "from")));
entry.setEnd(parseDate(getOrNull(json, "to")));
isAllDay = true;
}
else
{
// Old style regular event
entry.setStart(parseDate(json.getString("from") + " " + json.getString("start")));
entry.setEnd(parseDate(json.getString("to") + " " + json.getString("end")));
}
return isAllDay;
}
protected String getOrNull(JSONObject json, String key) throws JSONException
{
if(json.has(key))
{
return json.getString(key);
}
return null;
}
/**
* Normally the Calendar webscripts return a 200 with JSON
* containing the error message. Override this to switch to

View File

@@ -71,7 +71,7 @@ public class CalendarEntryDelete extends AbstractCalendarWebScript
{
// Get the date to be ignored
Map<QName,Serializable> props = new HashMap<QName, Serializable>();
Date date = extractDate(req.getParameter("date"));
Date date = parseDate(req.getParameter("date"));
props.put(CalendarModel.PROP_IGNORE_EVENT_DATE, date);
// Create a child node of the event

View File

@@ -61,30 +61,7 @@ public class CalendarEntryPost extends AbstractCalendarWebScript
entry.setSharePointDocFolder(getOrNull(json, "docfolder"));
// Handle the dates
if(json.has("startAt") && json.has("endAt"))
{
// New style ISO8601 dates
entry.setStart(extractDate(json.getString("startAt")));
entry.setEnd(extractDate(json.getString("endAt")));
if(json.has("allday"))
{
// TODO Handle All Day events properly, including timezones
isAllDay = true;
}
}
else if(json.has("allday"))
{
// Old style all-day event
entry.setStart(extractDate(getOrNull(json, "from")));
entry.setEnd(extractDate(getOrNull(json, "to")));
isAllDay = true;
}
else
{
// Old style regular event
entry.setStart(extractDate(json.getString("from") + " " + json.getString("start")));
entry.setEnd(extractDate(json.getString("to") + " " + json.getString("end")));
}
isAllDay = extractDates(entry, json);
// Handle tags
if(json.has("tags"))
@@ -162,13 +139,4 @@ public class CalendarEntryPost extends AbstractCalendarWebScript
model.put("result", result);
return model;
}
private String getOrNull(JSONObject json, String key) throws JSONException
{
if(json.has(key))
{
return json.getString(key);
}
return null;
}
}

View File

@@ -0,0 +1,207 @@
/*
* Copyright (C) 2005-2011 Alfresco Software Limited.
*
* This file is part of Alfresco
*
* Alfresco is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Alfresco is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
*/
package org.alfresco.repo.web.scripts.calendar;
import java.text.SimpleDateFormat;
import java.util.HashMap;
import java.util.Map;
import java.util.StringTokenizer;
import org.alfresco.service.cmr.calendar.CalendarEntry;
import org.alfresco.service.cmr.site.SiteInfo;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.json.JSONException;
import org.json.JSONObject;
import org.springframework.extensions.webscripts.Cache;
import org.springframework.extensions.webscripts.Status;
import org.springframework.extensions.webscripts.WebScriptRequest;
/**
* This class is the controller for the slingshot calendar event.put webscript.
*
* @author Nick Burch
* @since 4.0
*/
public class CalendarEntryPut extends AbstractCalendarWebScript
{
private static Log logger = LogFactory.getLog(CalendarEntryPut.class);
@Override
protected Map<String, Object> executeImpl(SiteInfo site, String eventName,
WebScriptRequest req, JSONObject json, Status status, Cache cache) {
CalendarEntry entry = calendarService.getCalendarEntry(
site.getShortName(), eventName
);
if(entry == null)
{
return buildError("Could not find event: " + eventName);
}
// TODO Handle All Day events properly, including timezones
boolean isAllDay = false;
try
{
// Doc folder is a bit special
String docFolder = json.getString("docfolder");
if(entry.getRecurrenceRule() != null)
{
// TODO Handle editing recurring rules
// Needs stuff with ignored events
/*
var prop = new Array();
var fromParts = params.date.split("-");
prop["ia:date"] = new Date(fromParts[0],fromParts[1] - 1,fromParts[2]);
editedEvent.createNode(null, "ia:ignoreEvent", prop, "ia:ignoreEventList");
var timestamp = new Date().getTime();
var random = Math.round(Math.random() * 10000);
event = eventsFolder.createNode(timestamp + "-" + random + ".ics", "ia:calendarEvent");
event.properties["ia:isOutlook"] = true;
*/
// TODO Special doc folder stuff
if("*NOT_CHANGE*".equals(docFolder))
{
// TODO
}
}
// Doc folder is a bit special
if("*NOT_CHANGE*".equals(docFolder))
{
// Nothing to change
}
else
{
entry.setSharePointDocFolder(docFolder);
}
// Grab the properties
entry.setTitle(getOrNull(json, "what"));
entry.setDescription(getOrNull(json, "desc"));
entry.setLocation(getOrNull(json, "where"));
// Handle the dates
isAllDay = extractDates(entry, json);
// Handle tags
if(json.has("tags"))
{
entry.getTags().clear();
StringTokenizer st = new StringTokenizer(json.getString("tags"), " ");
while(st.hasMoreTokens())
{
entry.getTags().add(st.nextToken());
}
}
}
catch(JSONException je)
{
return buildError("Invalid JSON: " + je.getMessage());
}
if(entry == null)
{
return buildError("Could not find event: " + eventName);
}
// Have it edited
entry = calendarService.updateCalendarEntry(entry);
// Generate the activity feed for this
SimpleDateFormat fmt = new SimpleDateFormat("yyyy-MM-dd");
String dateOpt = "?date=" + fmt.format(entry.getStart());
try
{
JSONObject activity = new JSONObject();
activity.put("title", entry.getTitle());
activity.put("page", req.getParameter("page") + dateOpt);
activityService.postActivity(
"org.alfresco.calendar.event-updated",
site.getShortName(),
CALENDAR_SERVICE_ACTIVITY_APP_NAME,
activity.toString()
);
}
catch(Exception e)
{
// Warn, but carry on
logger.warn("Error adding event deletion to activities feed", e);
}
// Build the return object
Map<String, Object> result = new HashMap<String, Object>();
result.put("summary", entry.getTitle());
result.put("description", entry.getDescription());
result.put("location", entry.getLocation());
result.put("dtstart", entry.getStart());
result.put("dtend", entry.getEnd());
result.put("uri", "calendar/event/" + site.getShortName() + "/" +
entry.getSystemName() + dateOpt);
result.put("tags", generateTagString(entry));
result.put("allday", isAllDay);
result.put("docfolder", entry.getSharePointDocFolder());
// Replace nulls with blank strings for the JSON
for(String key : result.keySet())
{
if(result.get(key) == null)
{
result.put(key, "");
}
}
// All done
Map<String, Object> model = new HashMap<String, Object>();
model.put("result", result);
return model;
}
/**
* We use lists for tags internally, and the other webscripts
* return arrays too. This one is different, and it needs to
* a single space separated string. This does the conversion
*/
protected String generateTagString(CalendarEntry entry)
{
StringBuffer sb = new StringBuffer();
if(entry.getTags() != null)
{
for(String tag : entry.getTags())
{
if(sb.length() > 0) sb.append(' ');
sb.append(tag);
}
}
return sb.toString();
}
}

View File

@@ -379,7 +379,7 @@ public class CalendarRestApiTest extends BaseWebScriptTest
// No from/to/start/end, does dtstart and dtend instead
assertEquals("2011-06-28T11:30", entry.getString("dtstart"));
assertEquals("2011-06-28T13:30", entry.getString("dtend"));
assertEquals("", entry.getString("allday"));
assertEquals("false", entry.getString("allday"));
// No isoutlook on create/edit