Comparing version
@@ -1,13 +0,16 @@ | ||
var ical = require('ical') | ||
, months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'] | ||
'use strict'; | ||
const ical = require('ical'); | ||
const months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; | ||
ical.fromURL('http://lanyrd.com/topics/nodejs/nodejs.ics', {}, function(err, data){ | ||
for (var k in data){ | ||
if (data.hasOwnProperty(k)){ | ||
var ev = data[k] | ||
console.log("Conference", ev.summary, 'is in', ev.location, 'on the', ev.start.getDate(), 'of', months[ev.start.getMonth()] ); | ||
} | ||
} | ||
}) | ||
ical.fromURL('http://lanyrd.com/topics/nodejs/nodejs.ics', {}, function (err, data) { | ||
for (let k in data) { | ||
if (data.hasOwnProperty(k)) { | ||
var ev = data[k]; | ||
if (data[k].type == 'VEVENT') { | ||
console.log(`${ev.summary} is in ${ev.location} on the ${ev.start.getDate()} of ${months[ev.start.getMonth()]} at ${ev.start.toLocaleTimeString('en-GB')}`); | ||
} | ||
} | ||
} | ||
}); |
228
ical.js
@@ -36,5 +36,5 @@ (function(name, definition) { | ||
var segs = p[i].split('='); | ||
out[segs[0]] = parseValue(segs.slice(1).join('=')); | ||
} | ||
@@ -48,3 +48,3 @@ } | ||
return true; | ||
if ('FALSE' === val) | ||
@@ -60,32 +60,38 @@ return false; | ||
var storeParam = function(name){ | ||
return function(val, params, curr){ | ||
var data; | ||
if (params && params.length && !(params.length==1 && params[0]==='CHARSET=utf-8')){ | ||
data = {params:parseParams(params), val:text(val)} | ||
} | ||
else | ||
data = text(val) | ||
var storeValParam = function (name) { | ||
return function (val, curr) { | ||
var current = curr[name]; | ||
if (Array.isArray(current)) { | ||
current.push(val); | ||
return curr; | ||
} | ||
var current = curr[name]; | ||
if (Array.isArray(current)){ | ||
current.push(data); | ||
return curr; | ||
if (current != null) { | ||
curr[name] = [current, val]; | ||
return curr; | ||
} | ||
curr[name] = val; | ||
return curr | ||
} | ||
} | ||
if (current != null){ | ||
curr[name] = [current, data]; | ||
return curr; | ||
var storeParam = function (name) { | ||
return function (val, params, curr) { | ||
var data; | ||
if (params && params.length && !(params.length == 1 && params[0] === 'CHARSET=utf-8')) { | ||
data = { params: parseParams(params), val: text(val) } | ||
} | ||
else | ||
data = text(val) | ||
return storeValParam(name)(data, curr); | ||
} | ||
curr[name] = data; | ||
return curr | ||
} | ||
} | ||
var addTZ = function(dt, name, params){ | ||
var addTZ = function (dt, params) { | ||
var p = parseParams(params); | ||
if (params && p){ | ||
dt[name].tz = p.TZID | ||
dt.tz = p.TZID | ||
} | ||
@@ -96,9 +102,8 @@ | ||
var dateParam = function(name){ | ||
return function(val, params, curr){ | ||
return function (val, params, curr) { | ||
// Store as string - worst case scenario | ||
storeParam(name)(val, undefined, curr) | ||
var newDate = text(val); | ||
if (params && params[0] === "VALUE=DATE") { | ||
@@ -110,3 +115,3 @@ // Just Date | ||
// No TZ info - assume same timezone as this computer | ||
curr[name] = new Date( | ||
newDate = new Date( | ||
comps[1], | ||
@@ -117,3 +122,6 @@ parseInt(comps[2], 10)-1, | ||
return addTZ(curr, name, params); | ||
newDate = addTZ(newDate, params); | ||
// Store as string - worst case scenario | ||
return storeValParam(name)(newDate, curr) | ||
} | ||
@@ -127,3 +135,3 @@ } | ||
if (comps[7] == 'Z'){ // GMT | ||
curr[name] = new Date(Date.UTC( | ||
newDate = new Date(Date.UTC( | ||
parseInt(comps[1], 10), | ||
@@ -138,3 +146,3 @@ parseInt(comps[2], 10)-1, | ||
} else { | ||
curr[name] = new Date( | ||
newDate = new Date( | ||
parseInt(comps[1], 10), | ||
@@ -148,6 +156,10 @@ parseInt(comps[2], 10)-1, | ||
} | ||
} | ||
return addTZ(curr, name, params) | ||
newDate = addTZ(newDate, params); | ||
} | ||
// Store as string - worst case scenario | ||
return storeValParam(name)(newDate, curr) | ||
} | ||
} | ||
@@ -169,3 +181,7 @@ | ||
storeParam(val, params, curr) | ||
curr[name] = val ? val.split(separatorPattern) : [] | ||
if (curr[name] === undefined) | ||
curr[name] = val ? val.split(separatorPattern) : [] | ||
else | ||
if (val) | ||
curr[name] = curr[name].concat(val.split(separatorPattern)) | ||
return curr | ||
@@ -175,3 +191,48 @@ } | ||
var addFBType = function(fb, params){ | ||
// EXDATE is an entry that represents exceptions to a recurrence rule (ex: "repeat every day except on 7/4"). | ||
// The EXDATE entry itself can also contain a comma-separated list, so we make sure to parse each date out separately. | ||
// There can also be more than one EXDATE entries in a calendar record. | ||
// Since there can be multiple dates, we create an array of them. The index into the array is the ISO string of the date itself, for ease of use. | ||
// i.e. You can check if ((curr.exdate != undefined) && (curr.exdate[date iso string] != undefined)) to see if a date is an exception. | ||
// NOTE: This specifically uses date only, and not time. This is to avoid a few problems: | ||
// 1. The ISO string with time wouldn't work for "floating dates" (dates without timezones). | ||
// ex: "20171225T060000" - this is supposed to mean 6 AM in whatever timezone you're currently in | ||
// 2. Daylight savings time potentially affects the time you would need to look up | ||
// 3. Some EXDATE entries in the wild seem to have times different from the recurrence rule, but are still excluded by calendar programs. Not sure how or why. | ||
// These would fail any sort of sane time lookup, because the time literally doesn't match the event. So we'll ignore time and just use date. | ||
// ex: DTSTART:20170814T140000Z | ||
// RRULE:FREQ=WEEKLY;WKST=SU;INTERVAL=2;BYDAY=MO,TU | ||
// EXDATE:20171219T060000 | ||
// Even though "T060000" doesn't match or overlap "T1400000Z", it's still supposed to be excluded? Odd. :( | ||
// TODO: See if this causes any problems with events that recur multiple times a day. | ||
var exdateParam = function (name) { | ||
return function (val, params, curr) { | ||
var separatorPattern = /\s*,\s*/g; | ||
curr[name] = curr[name] || []; | ||
var dates = val ? val.split(separatorPattern) : []; | ||
dates.forEach(function (entry) { | ||
var exdate = new Array(); | ||
dateParam(name)(entry, params, exdate); | ||
if (exdate[name]) | ||
{ | ||
if (typeof exdate[name].toISOString === 'function') { | ||
curr[name][exdate[name].toISOString().substring(0, 10)] = exdate[name]; | ||
} else { | ||
console.error("No toISOString function in exdate[name]", exdate[name]); | ||
} | ||
} | ||
} | ||
) | ||
return curr; | ||
} | ||
} | ||
// RECURRENCE-ID is the ID of a specific recurrence within a recurrence rule. | ||
// TODO: It's also possible for it to have a range, like "THISANDPRIOR", "THISANDFUTURE". This isn't currently handled. | ||
var recurrenceParam = function (name) { | ||
return dateParam(name); | ||
} | ||
var addFBType = function (fb, params) { | ||
var p = parseParams(params); | ||
@@ -220,3 +281,3 @@ | ||
obj; | ||
for (key in curr) { | ||
@@ -230,10 +291,89 @@ if(curr.hasOwnProperty(key)) { | ||
} | ||
return curr | ||
} | ||
var par = stack.pop() | ||
if (curr.uid) | ||
par[curr.uid] = curr | ||
{ | ||
// If this is the first time we run into this UID, just save it. | ||
if (par[curr.uid] === undefined) | ||
{ | ||
par[curr.uid] = curr; | ||
} | ||
else | ||
{ | ||
// If we have multiple ical entries with the same UID, it's either going to be a | ||
// modification to a recurrence (RECURRENCE-ID), and/or a significant modification | ||
// to the entry (SEQUENCE). | ||
// TODO: Look into proper sequence logic. | ||
if (curr.recurrenceid === undefined) | ||
{ | ||
// If we have the same UID as an existing record, and it *isn't* a specific recurrence ID, | ||
// not quite sure what the correct behaviour should be. For now, just take the new information | ||
// and merge it with the old record by overwriting only the fields that appear in the new record. | ||
var key; | ||
for (key in curr) { | ||
par[curr.uid][key] = curr[key]; | ||
} | ||
} | ||
} | ||
// If we have recurrence-id entries, list them as an array of recurrences keyed off of recurrence-id. | ||
// To use - as you're running through the dates of an rrule, you can try looking it up in the recurrences | ||
// array. If it exists, then use the data from the calendar object in the recurrence instead of the parent | ||
// for that day. | ||
// NOTE: Sometimes the RECURRENCE-ID record will show up *before* the record with the RRULE entry. In that | ||
// case, what happens is that the RECURRENCE-ID record ends up becoming both the parent record and an entry | ||
// in the recurrences array, and then when we process the RRULE entry later it overwrites the appropriate | ||
// fields in the parent record. | ||
if (curr.recurrenceid != null) | ||
{ | ||
// TODO: Is there ever a case where we have to worry about overwriting an existing entry here? | ||
// Create a copy of the current object to save in our recurrences array. (We *could* just do par = curr, | ||
// except for the case that we get the RECURRENCE-ID record before the RRULE record. In that case, we | ||
// would end up with a shared reference that would cause us to overwrite *both* records at the point | ||
// that we try and fix up the parent record.) | ||
var recurrenceObj = new Object(); | ||
var key; | ||
for (key in curr) { | ||
recurrenceObj[key] = curr[key]; | ||
} | ||
if (recurrenceObj.recurrences != undefined) { | ||
delete recurrenceObj.recurrences; | ||
} | ||
// If we don't have an array to store recurrences in yet, create it. | ||
if (par[curr.uid].recurrences === undefined) { | ||
par[curr.uid].recurrences = new Array(); | ||
} | ||
// Save off our cloned recurrence object into the array, keyed by date but not time. | ||
// We key by date only to avoid timezone and "floating time" problems (where the time isn't associated with a timezone). | ||
// TODO: See if this causes a problem with events that have multiple recurrences per day. | ||
if (typeof curr.recurrenceid.toISOString === 'function') { | ||
par[curr.uid].recurrences[curr.recurrenceid.toISOString().substring(0,10)] = recurrenceObj; | ||
} else { | ||
console.error("No toISOString function in curr.recurrenceid", curr.recurrenceid); | ||
} | ||
} | ||
// One more specific fix - in the case that an RRULE entry shows up after a RECURRENCE-ID entry, | ||
// let's make sure to clear the recurrenceid off the parent field. | ||
if ((par[curr.uid].rrule != undefined) && (par[curr.uid].recurrenceid != undefined)) | ||
{ | ||
delete par[curr.uid].recurrenceid; | ||
} | ||
} | ||
else | ||
@@ -252,2 +392,3 @@ par[Math.random()*100000] = curr // Randomly assign ID : TODO - use true GUID | ||
, 'DTEND' : dateParam('end') | ||
, 'EXDATE' : exdateParam('exdate') | ||
,' CLASS' : storeParam('class') | ||
@@ -260,2 +401,7 @@ , 'TRANSP' : storeParam('transparency') | ||
, 'FREEBUSY': freebusyParam('freebusy') | ||
, 'DTSTAMP': dateParam('dtstamp') | ||
, 'CREATED': dateParam('created') | ||
, 'LAST-MODIFIED': dateParam('lastmodified') | ||
, 'RECURRENCE-ID': recurrenceParam('recurrenceid') | ||
}, | ||
@@ -276,3 +422,3 @@ | ||
} | ||
return storeParam(name.toLowerCase())(val, params, ctx); | ||
@@ -279,0 +425,0 @@ }, |
@@ -9,5 +9,12 @@ var ical = require('./ical') | ||
request(url, opts, function(err, r, data){ | ||
if (err) | ||
return cb(err, null); | ||
cb(undefined, ical.parseICS(data)); | ||
if (err) | ||
{ | ||
return cb(err, null); | ||
} | ||
else if (r.statusCode != 200) | ||
{ | ||
return cb(r.statusCode + ": " + r.statusMessage, null); | ||
} | ||
cb(undefined, ical.parseICS(data)); | ||
}) | ||
@@ -28,12 +35,34 @@ } | ||
var originalEnd = ical.objectHandlers['END']; | ||
ical.objectHandlers['END'] = function(val, params, curr, stack){ | ||
if (curr.rrule) { | ||
var rule = curr.rrule.replace('RRULE:', ''); | ||
if (rule.indexOf('DTSTART') === -1) { | ||
rule += ';DTSTART=' + curr.start.toISOString().replace(/[-:]/g, ''); | ||
rule = rule.replace(/\.[0-9]{3}/, ''); | ||
} | ||
curr.rrule = rrule.fromString(rule); | ||
} | ||
ical.objectHandlers['END'] = function (val, params, curr, stack) { | ||
// Recurrence rules are only valid for VEVENT, VTODO, and VJOURNAL. | ||
// More specifically, we need to filter the VCALENDAR type because we might end up with a defined rrule | ||
// due to the subtypes. | ||
if ((val === "VEVENT") || (val === "VTODO") || (val === "VJOURNAL")) { | ||
if (curr.rrule) { | ||
var rule = curr.rrule.replace('RRULE:', ''); | ||
if (rule.indexOf('DTSTART') === -1) { | ||
if (curr.start.length === 8) { | ||
var comps = /^(\d{4})(\d{2})(\d{2})$/.exec(curr.start); | ||
if (comps) { | ||
curr.start = new Date(comps[1], comps[2] - 1, comps[3]); | ||
} | ||
} | ||
if (typeof curr.start.toISOString === 'function') { | ||
try { | ||
rule += ';DTSTART=' + curr.start.toISOString().replace(/[-:]/g, ''); | ||
rule = rule.replace(/\.[0-9]{3}/, ''); | ||
} catch (error) { | ||
console.error("ERROR when trying to convert to ISOString", error); | ||
} | ||
} else { | ||
console.error("No toISOString function in curr.start", curr.start); | ||
} | ||
} | ||
curr.rrule = rrule.fromString(rule); | ||
} | ||
} | ||
return originalEnd.call(this, val, params, curr, stack); | ||
} |
{ | ||
"name": "ical", | ||
"version": "0.5.0", | ||
"version": "0.6.0", | ||
"main": "index.js", | ||
@@ -13,2 +13,3 @@ "description": "A tolerant, minimal icalendar parser", | ||
"author": "Peter Braden <peterbraden@peterbraden.co.uk> (peterbraden.co.uk)", | ||
"license": "Apache-2.0", | ||
"repository": { | ||
@@ -19,8 +20,8 @@ "type": "git", | ||
"dependencies": { | ||
"request": "2.68.0", | ||
"rrule": "2.0.0" | ||
"request": "^2.88.0", | ||
"rrule": "2.4.1" | ||
}, | ||
"devDependencies": { | ||
"vows": "0.7.0", | ||
"underscore": "1.3.0" | ||
"vows": "0.8.2", | ||
"underscore": "1.9.1" | ||
}, | ||
@@ -27,0 +28,0 @@ "scripts": { |
@@ -10,2 +10,3 @@ # ical.js # | ||
## Install - Node.js ## | ||
@@ -37,17 +38,27 @@ | ||
```javascript | ||
var ical = require('ical') | ||
, months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'] | ||
'use strict'; | ||
ical.fromURL('http://lanyrd.com/topics/nodejs/nodejs.ics', {}, function(err, data) { | ||
for (var k in data){ | ||
if (data.hasOwnProperty(k)) { | ||
var ev = data[k] | ||
console.log("Conference", | ||
ev.summary, | ||
'is in', | ||
ev.location, | ||
'on the', ev.start.getDate(), 'of', months[ev.start.getMonth()]); | ||
} | ||
} | ||
}); | ||
const ical = require('ical'); | ||
const months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; | ||
ical.fromURL('http://lanyrd.com/topics/nodejs/nodejs.ics', {}, function (err, data) { | ||
for (let k in data) { | ||
if (data.hasOwnProperty(k)) { | ||
var ev = data[k]; | ||
if (data[k].type == 'VEVENT') { | ||
console.log(`${ev.summary} is in ${ev.location} on the ${ev.start.getDate()} of ${months[ev.start.getMonth()]} at ${ev.start.toLocaleTimeString('en-GB')}`); | ||
} | ||
} | ||
} | ||
}); | ||
``` | ||
## Recurrences and Exceptions ## | ||
Calendar events with recurrence rules can be significantly more complicated to handle correctly. There are three parts to handling them: | ||
1. rrule - the recurrence rule specifying the pattern of recurring dates and times for the event. | ||
2. recurrences - an optional array of event data that can override specific occurrences of the event. | ||
3. exdate - an optional array of dates that should be excluded from the recurrence pattern. | ||
See example_rrule.js for an example of handling recurring calendar events. |
117
test/test.js
@@ -200,3 +200,3 @@ /**** | ||
, 'with test6.ics (testing assembly.org)' : { | ||
, 'with test6.ics (testing assembly.org)': { | ||
topic: function () { | ||
@@ -320,2 +320,10 @@ return ical.parseFile('./test/test6.ics') | ||
} | ||
}, | ||
'when categories present on multiple lines': { | ||
topic: function (t) {return _.values(t)[4]}, | ||
'should contain the category values in an array': function (e) { | ||
assert.deepEqual(e.categories, ['cat1', 'cat2', 'cat3']); | ||
} | ||
} | ||
@@ -367,7 +375,108 @@ }, | ||
} | ||
}, | ||
} | ||
'url request errors' : { | ||
, 'with test12.ics (testing recurrences and exdates)': { | ||
topic: function () { | ||
return ical.parseFile('./test/test12.ics') | ||
} | ||
, 'event with rrule': { | ||
topic: function (events) { | ||
return _.select(_.values(events), function (x) { | ||
return x.uid === '0000001'; | ||
})[0]; | ||
} | ||
, "Has an RRULE": function (topic) { | ||
assert.notEqual(topic.rrule, undefined); | ||
} | ||
, "Has summary Treasure Hunting": function (topic) { | ||
assert.equal(topic.summary, 'Treasure Hunting'); | ||
} | ||
, "Has two EXDATES": function (topic) { | ||
assert.notEqual(topic.exdate, undefined); | ||
assert.notEqual(topic.exdate[new Date(2015, 06, 08, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.notEqual(topic.exdate[new Date(2015, 06, 10, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
} | ||
, "Has a RECURRENCE-ID override": function (topic) { | ||
assert.notEqual(topic.recurrences, undefined); | ||
assert.notEqual(topic.recurrences[new Date(2015, 06, 07, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.equal(topic.recurrences[new Date(2015, 06, 07, 12, 0, 0).toISOString().substring(0, 10)].summary, 'More Treasure Hunting'); | ||
} | ||
} | ||
} | ||
, 'with test13.ics (testing recurrence-id before rrule)': { | ||
topic: function () { | ||
return ical.parseFile('./test/test13.ics') | ||
} | ||
, 'event with rrule': { | ||
topic: function (events) { | ||
return _.select(_.values(events), function (x) { | ||
return x.uid === '6m2q7kb2l02798oagemrcgm6pk@google.com'; | ||
})[0]; | ||
} | ||
, "Has an RRULE": function (topic) { | ||
assert.notEqual(topic.rrule, undefined); | ||
} | ||
, "Has summary 'repeated'": function (topic) { | ||
assert.equal(topic.summary, 'repeated'); | ||
} | ||
, "Has a RECURRENCE-ID override": function (topic) { | ||
assert.notEqual(topic.recurrences, undefined); | ||
assert.notEqual(topic.recurrences[new Date(2016, 7, 26, 14, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.equal(topic.recurrences[new Date(2016, 7, 26, 14, 0, 0).toISOString().substring(0, 10)].summary, 'bla bla'); | ||
} | ||
} | ||
} | ||
, 'with test14.ics (testing comma-separated exdates)': { | ||
topic: function () { | ||
return ical.parseFile('./test/test14.ics') | ||
} | ||
, 'event with comma-separated exdate': { | ||
topic: function (events) { | ||
return _.select(_.values(events), function (x) { | ||
return x.uid === '98765432-ABCD-DCBB-999A-987765432123'; | ||
})[0]; | ||
} | ||
, "Has summary 'Example of comma-separated exdates'": function (topic) { | ||
assert.equal(topic.summary, 'Example of comma-separated exdates'); | ||
} | ||
, "Has four comma-separated EXDATES": function (topic) { | ||
assert.notEqual(topic.exdate, undefined); | ||
// Verify the four comma-separated EXDATES are there | ||
assert.notEqual(topic.exdate[new Date(2017, 6, 6, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.notEqual(topic.exdate[new Date(2017, 6, 17, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.notEqual(topic.exdate[new Date(2017, 6, 20, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.notEqual(topic.exdate[new Date(2017, 7, 3, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
// Verify an arbitrary date isn't there | ||
assert.equal(topic.exdate[new Date(2017, 4, 5, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
} | ||
} | ||
} | ||
, 'with test14.ics (testing exdates with bad times)': { | ||
topic: function () { | ||
return ical.parseFile('./test/test14.ics') | ||
} | ||
, 'event with exdates with bad times': { | ||
topic: function (events) { | ||
return _.select(_.values(events), function (x) { | ||
return x.uid === '1234567-ABCD-ABCD-ABCD-123456789012'; | ||
})[0]; | ||
} | ||
, "Has summary 'Example of exdate with bad times'": function (topic) { | ||
assert.equal(topic.summary, 'Example of exdate with bad times'); | ||
} | ||
, "Has two EXDATES even though they have bad times": function (topic) { | ||
assert.notEqual(topic.exdate, undefined); | ||
// Verify the two EXDATES are there, even though they have bad times | ||
assert.notEqual(topic.exdate[new Date(2017, 11, 18, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
assert.notEqual(topic.exdate[new Date(2017, 11, 19, 12, 0, 0).toISOString().substring(0, 10)], undefined); | ||
} | ||
} | ||
} | ||
, 'url request errors': { | ||
topic : function () { | ||
ical.fromURL('http://not.exist/', {}, this.callback); | ||
ical.fromURL('http://255.255.255.255/', {}, this.callback); | ||
} | ||
@@ -374,0 +483,0 @@ , 'are passed back to the callback' : function (err, result) { |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
140251
13.32%26
4%990
54.69%63
21.15%+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
- Removed
Updated
Updated