2016-10-19 03:11:36 +00:00
|
|
|
# Copyright (C) 2012-2016 Zammad Foundation, http://zammad-foundation.org/
|
2015-09-09 06:52:05 +00:00
|
|
|
|
|
|
|
class Calendar < ApplicationModel
|
|
|
|
store :business_hours
|
|
|
|
store :public_holidays
|
|
|
|
|
2015-09-25 07:21:55 +00:00
|
|
|
before_create :validate_public_holidays, :fetch_ical
|
|
|
|
before_update :validate_public_holidays, :fetch_ical
|
2015-09-10 19:09:50 +00:00
|
|
|
after_create :sync_default, :min_one_check
|
|
|
|
after_update :sync_default, :min_one_check
|
|
|
|
after_destroy :min_one_check
|
|
|
|
|
2016-03-20 19:09:52 +00:00
|
|
|
notify_clients_support
|
|
|
|
|
2015-09-09 06:52:05 +00:00
|
|
|
=begin
|
|
|
|
|
2015-09-22 14:48:43 +00:00
|
|
|
set inital default calendar
|
|
|
|
|
|
|
|
calendar = Calendar.init_setup
|
|
|
|
|
|
|
|
returns calendar object
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
|
|
|
def self.init_setup(ip = nil)
|
|
|
|
|
2015-09-29 08:23:13 +00:00
|
|
|
# ignore client ip if not public ip
|
|
|
|
if ip && ip =~ /^(::1|127\.|10\.|172\.1[6-9]\.|172\.2[0-9]\.|172\.3[0-1]\.|192\.168\.)/
|
|
|
|
ip = nil
|
|
|
|
end
|
|
|
|
|
2016-01-13 13:12:05 +00:00
|
|
|
# prevent multible setups for same ip
|
|
|
|
cache = Cache.get('Calendar.init_setup.done')
|
|
|
|
return if cache && cache[:ip] == ip
|
|
|
|
Cache.write('Calendar.init_setup.done', { ip: ip }, { expires_in: 1.hour })
|
|
|
|
|
2015-09-22 14:48:43 +00:00
|
|
|
# call for calendar suggestion
|
|
|
|
calendar_details = Service::GeoCalendar.location(ip)
|
|
|
|
return if !calendar_details
|
|
|
|
|
|
|
|
calendar_details['name'] = Calendar.genrate_uniq_name(calendar_details['name'])
|
|
|
|
calendar_details['default'] = true
|
|
|
|
calendar_details['created_by_id'] = 1
|
|
|
|
calendar_details['updated_by_id'] = 1
|
|
|
|
|
|
|
|
# find if auto generated calendar exists
|
|
|
|
calendar = Calendar.find_by(default: true, updated_by_id: 1, created_by_id: 1)
|
|
|
|
if calendar
|
|
|
|
calendar.update_attributes(calendar_details)
|
|
|
|
return calendar
|
|
|
|
end
|
|
|
|
create(calendar_details)
|
|
|
|
end
|
|
|
|
|
|
|
|
=begin
|
|
|
|
|
2015-09-09 06:52:05 +00:00
|
|
|
get default calendar
|
|
|
|
|
|
|
|
calendar = Calendar.default
|
|
|
|
|
|
|
|
returns calendar object
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
|
|
|
def self.default
|
|
|
|
find_by(default: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
=begin
|
|
|
|
|
2016-12-13 13:58:13 +00:00
|
|
|
returns preset of ical feeds
|
2015-09-09 06:52:05 +00:00
|
|
|
|
|
|
|
feeds = Calendar.ical_feeds
|
|
|
|
|
|
|
|
returns
|
|
|
|
|
|
|
|
{
|
2016-12-02 11:24:00 +00:00
|
|
|
'http://www.google.com/calendar/ical/en.usa%23holiday%40group.v.calendar.google.com/public/basic.ics' => 'US',
|
2015-09-09 06:52:05 +00:00
|
|
|
...
|
|
|
|
}
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
|
|
|
def self.ical_feeds
|
2016-12-02 11:24:00 +00:00
|
|
|
data = YAML.load_file(Rails.root.join('config/holiday_calendars.yml'))
|
|
|
|
url = data['url']
|
|
|
|
|
|
|
|
data['countries'].map do |country, domain|
|
|
|
|
[(url % { domain: domain }), country]
|
|
|
|
end.to_h
|
2015-09-09 06:52:05 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
=begin
|
|
|
|
|
|
|
|
get list of available timezones and UTC offsets
|
|
|
|
|
|
|
|
list = Calendar.timezones
|
|
|
|
|
|
|
|
returns
|
|
|
|
|
|
|
|
{
|
|
|
|
'America/Los_Angeles' => -7
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
|
|
|
def self.timezones
|
|
|
|
list = {}
|
|
|
|
TZInfo::Timezone.all_country_zone_identifiers.each { |timezone|
|
|
|
|
t = TZInfo::Timezone.get(timezone)
|
|
|
|
diff = t.current_period.utc_total_offset / 60 / 60
|
|
|
|
list[ timezone ] = diff
|
|
|
|
}
|
|
|
|
list
|
|
|
|
end
|
|
|
|
|
|
|
|
=begin
|
|
|
|
|
|
|
|
syn all calendars with ical feeds
|
|
|
|
|
|
|
|
success = Calendar.sync
|
|
|
|
|
|
|
|
returns
|
|
|
|
|
|
|
|
true # or false
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
|
|
|
def self.sync
|
2016-12-02 11:24:00 +00:00
|
|
|
Calendar.find_each(&:sync)
|
2015-09-09 06:52:05 +00:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
=begin
|
|
|
|
|
|
|
|
syn one calendars with ical feed
|
|
|
|
|
|
|
|
calendar = Calendar.find(4711)
|
|
|
|
success = calendar.sync
|
|
|
|
|
|
|
|
returns
|
|
|
|
|
|
|
|
true # or false
|
|
|
|
|
|
|
|
=end
|
|
|
|
|
2015-09-23 07:10:07 +00:00
|
|
|
def sync(without_save = nil)
|
2015-09-09 06:52:05 +00:00
|
|
|
return if !ical_url
|
2016-11-10 15:11:59 +00:00
|
|
|
|
|
|
|
# only sync every 5 days
|
|
|
|
cache_key = "CalendarIcal::#{id}"
|
|
|
|
cache = Cache.get(cache_key)
|
|
|
|
return if !last_log && cache && cache[:ical_url] == ical_url
|
|
|
|
|
2015-09-23 07:10:07 +00:00
|
|
|
begin
|
2015-09-25 09:48:14 +00:00
|
|
|
events = {}
|
|
|
|
if ical_url && !ical_url.empty?
|
|
|
|
events = Calendar.parse(ical_url)
|
|
|
|
end
|
2015-09-09 06:52:05 +00:00
|
|
|
|
2015-09-23 07:10:07 +00:00
|
|
|
# sync with public_holidays
|
|
|
|
if !public_holidays
|
|
|
|
self.public_holidays = {}
|
2015-09-09 06:52:05 +00:00
|
|
|
end
|
2015-09-25 07:21:55 +00:00
|
|
|
|
|
|
|
# remove old ical entries if feed has changed
|
2016-06-30 20:04:48 +00:00
|
|
|
public_holidays.each { |day, meta|
|
2015-09-25 07:21:55 +00:00
|
|
|
next if !public_holidays[day]['feed']
|
|
|
|
next if meta['feed'] == Digest::MD5.hexdigest(ical_url)
|
|
|
|
public_holidays.delete(day)
|
|
|
|
}
|
|
|
|
|
|
|
|
# sync new ical feed dates
|
2016-06-30 20:04:48 +00:00
|
|
|
events.each { |day, summary|
|
2015-09-23 07:10:07 +00:00
|
|
|
if !public_holidays[day]
|
|
|
|
public_holidays[day] = {}
|
|
|
|
end
|
|
|
|
|
|
|
|
# ignore if already added or changed
|
|
|
|
next if public_holidays[day].key?('active')
|
|
|
|
|
|
|
|
# create new entry
|
|
|
|
public_holidays[day] = {
|
|
|
|
active: true,
|
|
|
|
summary: summary,
|
2015-09-25 07:21:55 +00:00
|
|
|
feed: Digest::MD5.hexdigest(ical_url)
|
2015-09-23 07:10:07 +00:00
|
|
|
}
|
2015-09-09 06:52:05 +00:00
|
|
|
}
|
2015-09-23 07:10:07 +00:00
|
|
|
self.last_log = nil
|
2016-11-10 15:11:59 +00:00
|
|
|
cache = Cache.write(
|
|
|
|
cache_key,
|
|
|
|
{ public_holidays: public_holidays, ical_url: ical_url },
|
|
|
|
{ expires_in: 5.days },
|
|
|
|
)
|
2015-09-23 07:10:07 +00:00
|
|
|
rescue => e
|
|
|
|
self.last_log = e.inspect
|
|
|
|
end
|
|
|
|
|
2015-09-09 06:52:05 +00:00
|
|
|
self.last_sync = Time.zone.now
|
2015-09-23 07:10:07 +00:00
|
|
|
if !without_save
|
|
|
|
save
|
|
|
|
end
|
2015-09-09 06:52:05 +00:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.parse(location)
|
|
|
|
if location =~ /^http/i
|
|
|
|
result = UserAgent.get(location)
|
2015-09-23 07:10:07 +00:00
|
|
|
if !result.success?
|
2016-03-01 14:26:46 +00:00
|
|
|
raise result.error
|
2015-09-23 07:10:07 +00:00
|
|
|
end
|
2015-09-09 06:52:05 +00:00
|
|
|
cal_file = result.body
|
|
|
|
else
|
|
|
|
cal_file = File.open(location)
|
|
|
|
end
|
|
|
|
|
2016-07-17 23:05:40 +00:00
|
|
|
cals = Icalendar::Calendar.parse(cal_file)
|
2015-09-09 06:52:05 +00:00
|
|
|
cal = cals.first
|
|
|
|
events = {}
|
2016-06-30 20:04:48 +00:00
|
|
|
cal.events.each { |event|
|
2015-09-09 06:52:05 +00:00
|
|
|
next if event.dtstart < Time.zone.now - 1.year
|
2015-11-30 13:29:23 +00:00
|
|
|
next if event.dtstart > Time.zone.now + 3.years
|
2015-09-09 06:52:05 +00:00
|
|
|
day = "#{event.dtstart.year}-#{format('%02d', event.dtstart.month)}-#{format('%02d', event.dtstart.day)}"
|
|
|
|
comment = event.summary || event.description
|
|
|
|
comment = Encode.conv( 'utf8', comment.to_s.force_encoding('utf-8') )
|
|
|
|
if !comment.valid_encoding?
|
|
|
|
comment = comment.encode('utf-8', 'binary', invalid: :replace, undef: :replace, replace: '?')
|
|
|
|
end
|
2015-09-16 08:40:18 +00:00
|
|
|
|
|
|
|
# ignore daylight saving time entries
|
|
|
|
next if comment =~ /(daylight saving|sommerzeit|summertime)/i
|
2015-09-09 06:52:05 +00:00
|
|
|
events[day] = comment
|
|
|
|
}
|
|
|
|
events.sort.to_h
|
|
|
|
end
|
2015-09-10 19:09:50 +00:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
# if changed calendar is default, set all others default to false
|
|
|
|
def sync_default
|
|
|
|
return if !default
|
2016-12-02 11:24:00 +00:00
|
|
|
Calendar.find_each { |calendar|
|
2015-09-10 19:09:50 +00:00
|
|
|
next if calendar.id == id
|
|
|
|
next if !calendar.default
|
|
|
|
calendar.default = false
|
|
|
|
calendar.save
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
# check if min one is set to default true
|
|
|
|
def min_one_check
|
2016-11-10 15:11:59 +00:00
|
|
|
if !Calendar.find_by(default: true)
|
|
|
|
first = Calendar.order(:created_at, :id).limit(1).first
|
|
|
|
first.default = true
|
|
|
|
first.save
|
|
|
|
end
|
2015-09-22 23:22:45 +00:00
|
|
|
|
|
|
|
# check if sla's are refer to an existing calendar
|
2016-11-10 15:11:59 +00:00
|
|
|
default_calendar = Calendar.find_by(default: true)
|
2016-12-02 11:24:00 +00:00
|
|
|
Sla.find_each { |sla|
|
2015-09-22 23:22:45 +00:00
|
|
|
if !sla.calendar_id
|
2016-11-10 15:11:59 +00:00
|
|
|
sla.calendar_id = default_calendar.id
|
|
|
|
sla.save!
|
2015-09-22 23:22:45 +00:00
|
|
|
next
|
|
|
|
end
|
|
|
|
if !Calendar.find_by(id: sla.calendar_id)
|
2016-11-10 15:11:59 +00:00
|
|
|
sla.calendar_id = default_calendar.id
|
|
|
|
sla.save!
|
2015-09-22 23:22:45 +00:00
|
|
|
end
|
|
|
|
}
|
2015-09-10 19:09:50 +00:00
|
|
|
end
|
2015-09-23 07:10:07 +00:00
|
|
|
|
|
|
|
# fetch ical feed
|
|
|
|
def fetch_ical
|
|
|
|
sync(true)
|
|
|
|
end
|
2015-09-25 07:21:55 +00:00
|
|
|
|
|
|
|
# validate format of public holidays
|
|
|
|
def validate_public_holidays
|
|
|
|
|
|
|
|
# fillup feed info
|
2016-11-10 15:11:59 +00:00
|
|
|
before = public_holidays_was
|
2016-06-30 20:04:48 +00:00
|
|
|
public_holidays.each { |day, meta|
|
2016-11-10 15:11:59 +00:00
|
|
|
if before && before[day] && before[day]['feed']
|
|
|
|
meta['feed'] = before[day]['feed']
|
2015-09-25 07:21:55 +00:00
|
|
|
end
|
2016-01-15 17:22:57 +00:00
|
|
|
meta['active'] = if meta['active']
|
|
|
|
true
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
2015-09-25 07:21:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
end
|
2015-09-09 06:52:05 +00:00
|
|
|
end
|