<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="en">
	<id>https://www.rabbibob.com/index.php?action=history&amp;feed=atom&amp;title=VoYD_Gig_Calendar</id>
	<title>VoYD Gig Calendar - Revision history</title>
	<link rel="self" type="application/atom+xml" href="https://www.rabbibob.com/index.php?action=history&amp;feed=atom&amp;title=VoYD_Gig_Calendar"/>
	<link rel="alternate" type="text/html" href="https://www.rabbibob.com/index.php?title=VoYD_Gig_Calendar&amp;action=history"/>
	<updated>2026-05-19T07:37:54Z</updated>
	<subtitle>Revision history for this page on the wiki</subtitle>
	<generator>MediaWiki 1.39.5</generator>
	<entry>
		<id>https://www.rabbibob.com/index.php?title=VoYD_Gig_Calendar&amp;diff=1926&amp;oldid=prev</id>
		<title>Rabbi Bob at 13:25, 17 May 2026</title>
		<link rel="alternate" type="text/html" href="https://www.rabbibob.com/index.php?title=VoYD_Gig_Calendar&amp;diff=1926&amp;oldid=prev"/>
		<updated>2026-05-17T13:25:44Z</updated>

		<summary type="html">&lt;p&gt;&lt;/p&gt;
&lt;table style=&quot;background-color: #fff; color: #202122;&quot; data-mw=&quot;interface&quot;&gt;
				&lt;col class=&quot;diff-marker&quot; /&gt;
				&lt;col class=&quot;diff-content&quot; /&gt;
				&lt;col class=&quot;diff-marker&quot; /&gt;
				&lt;col class=&quot;diff-content&quot; /&gt;
				&lt;tr class=&quot;diff-title&quot; lang=&quot;en&quot;&gt;
				&lt;td colspan=&quot;2&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;← Older revision&lt;/td&gt;
				&lt;td colspan=&quot;2&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;Revision as of 13:25, 17 May 2026&lt;/td&gt;
				&lt;/tr&gt;&lt;tr&gt;&lt;td colspan=&quot;2&quot; class=&quot;diff-lineno&quot; id=&quot;mw-diff-left-l2&quot;&gt;Line 2:&lt;/td&gt;
&lt;td colspan=&quot;2&quot; class=&quot;diff-lineno&quot;&gt;Line 2:&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;I had a project at work recently combining 10+ Google calendars into one primary calendar.  It had to include a prefix on each event title so the event could be traced back to the proper department area and it had to add\update\delete depending on how the source calendar was updated.&lt;/div&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;I had a project at work recently combining 10+ Google calendars into one primary calendar.  It had to include a prefix on each event title so the event could be traced back to the proper department area and it had to add\update\delete depending on how the source calendar was updated.&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;br/&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;br/&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot; data-marker=&quot;−&quot;&gt;&lt;/td&gt;&lt;td style=&quot;color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #ffe49c; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;I keep a spreadsheet going for band related things and one of them is a calendar tab, but it isn&amp;#039;t very portable.  The recent project got me to thinking about the spreadsheet: I have dates, venue name, some notes, the venue location, some approximation of the time we usually go live, and we have a FB page... let&amp;#039;s have Google &lt;del style=&quot;font-weight: bold; text-decoration: none;&quot;&gt;App Scripts &lt;/del&gt;update a calendar based on the spreadsheet.  Let&amp;#039;s have it do it hourly.&lt;/div&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot; data-marker=&quot;+&quot;&gt;&lt;/td&gt;&lt;td style=&quot;color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #a3d3ff; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;I keep a spreadsheet going for band related things and one of them is a calendar tab, but it isn&amp;#039;t very portable.  The recent project got me to thinking about the spreadsheet: I have dates, venue name, some notes, the venue location, some approximation of the time we usually go live, and we have a FB page... let&amp;#039;s have Google &lt;ins style=&quot;font-weight: bold; text-decoration: none;&quot;&gt;Apps Script &lt;/ins&gt;update a calendar based on the spreadsheet.  Let&amp;#039;s have it do it hourly.&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;br/&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;br/&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;And it does!&lt;/div&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;And it does!&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td colspan=&quot;2&quot; class=&quot;diff-lineno&quot; id=&quot;mw-diff-left-l225&quot;&gt;Line 225:&lt;/td&gt;
&lt;td colspan=&quot;2&quot; class=&quot;diff-lineno&quot;&gt;Line 225:&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;}&lt;/div&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;}&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;&amp;lt;/code&amp;gt;&lt;/div&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot;&gt;&lt;/td&gt;&lt;td style=&quot;background-color: #f8f9fa; color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #eaecf0; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;&amp;lt;/code&amp;gt;&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td colspan=&quot;2&quot; class=&quot;diff-side-deleted&quot;&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot; data-marker=&quot;+&quot;&gt;&lt;/td&gt;&lt;td style=&quot;color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #a3d3ff; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;&lt;ins style=&quot;font-weight: bold; text-decoration: none;&quot;&gt;[[Category:VoYD]]&lt;/ins&gt;&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;tr&gt;&lt;td colspan=&quot;2&quot; class=&quot;diff-side-deleted&quot;&gt;&lt;/td&gt;&lt;td class=&quot;diff-marker&quot; data-marker=&quot;+&quot;&gt;&lt;/td&gt;&lt;td style=&quot;color: #202122; font-size: 88%; border-style: solid; border-width: 1px 1px 1px 4px; border-radius: 0.33em; border-color: #a3d3ff; vertical-align: top; white-space: pre-wrap;&quot;&gt;&lt;div&gt;&lt;ins style=&quot;font-weight: bold; text-decoration: none;&quot;&gt;[[Category:Google Apps Script]]&lt;/ins&gt;&lt;/div&gt;&lt;/td&gt;&lt;/tr&gt;

&lt;!-- diff cache key mw_RabbiBob_139-wiki_:diff::1.12:old-1925:rev-1926 --&gt;
&lt;/table&gt;</summary>
		<author><name>Rabbi Bob</name></author>
	</entry>
	<entry>
		<id>https://www.rabbibob.com/index.php?title=VoYD_Gig_Calendar&amp;diff=1925&amp;oldid=prev</id>
		<title>Rabbi Bob: Created page with &quot;==History== I had a project at work recently combining 10+ Google calendars into one primary calendar.  It had to include a prefix on each event title so the event could be traced back to the proper department area and it had to add\update\delete depending on how the source calendar was updated.  I keep a spreadsheet going for band related things and one of them is a calendar tab, but it isn&#039;t very portable.  The recent project got me to thinking about the spreadsheet: I...&quot;</title>
		<link rel="alternate" type="text/html" href="https://www.rabbibob.com/index.php?title=VoYD_Gig_Calendar&amp;diff=1925&amp;oldid=prev"/>
		<updated>2026-05-17T13:25:03Z</updated>

		<summary type="html">&lt;p&gt;Created page with &amp;quot;==History== I had a project at work recently combining 10+ Google calendars into one primary calendar.  It had to include a prefix on each event title so the event could be traced back to the proper department area and it had to add\update\delete depending on how the source calendar was updated.  I keep a spreadsheet going for band related things and one of them is a calendar tab, but it isn&amp;#039;t very portable.  The recent project got me to thinking about the spreadsheet: I...&amp;quot;&lt;/p&gt;
&lt;p&gt;&lt;b&gt;New page&lt;/b&gt;&lt;/p&gt;&lt;div&gt;==History==&lt;br /&gt;
I had a project at work recently combining 10+ Google calendars into one primary calendar.  It had to include a prefix on each event title so the event could be traced back to the proper department area and it had to add\update\delete depending on how the source calendar was updated.&lt;br /&gt;
&lt;br /&gt;
I keep a spreadsheet going for band related things and one of them is a calendar tab, but it isn&amp;#039;t very portable.  The recent project got me to thinking about the spreadsheet: I have dates, venue name, some notes, the venue location, some approximation of the time we usually go live, and we have a FB page... let&amp;#039;s have Google App Scripts update a calendar based on the spreadsheet.  Let&amp;#039;s have it do it hourly.&lt;br /&gt;
&lt;br /&gt;
And it does!&lt;br /&gt;
&lt;br /&gt;
https://rabbibob.com/voydcalendar/&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
==The Script==&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
function syncVoYDGigCalendar() {&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  // CONFIGURATION &amp;amp; TOGGLES&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  var VoYDDebug = 0; // 1 = Show full row-by-row debugging logs, 0 = Quiet summary mode&lt;br /&gt;
  var CALENDAR_ID = &amp;quot;thatspecialstringooiuwr209348294ufrjaowiuf2@group.calendar.google.com&amp;quot;;&lt;br /&gt;
  var FACEBOOK_URL = &amp;quot;https://facebook.com/voydrocks&amp;quot;;&lt;br /&gt;
// ==========================================&lt;br /&gt;
  // INITIALIZATION &amp;amp; SCRIPT SETUP&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  var calendar = CalendarApp.getCalendarById(CALENDAR_ID);&lt;br /&gt;
  if (!calendar) {&lt;br /&gt;
    Logger.log(&amp;quot;❌ ERROR: Could not find or access calendar ID: &amp;quot; + CALENDAR_ID);&lt;br /&gt;
    return;&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  var ss = SpreadsheetApp.getActiveSpreadsheet(); &lt;br /&gt;
  var sheets = ss.getSheets();&lt;br /&gt;
  var sheet = null;&lt;br /&gt;
  &lt;br /&gt;
  // Target specifically by GID 729321447&lt;br /&gt;
  for (var i = 0; i &amp;lt; sheets.length; i++) {&lt;br /&gt;
    if (sheets[i].getSheetId() == 729321447) {&lt;br /&gt;
      sheet = sheets[i];&lt;br /&gt;
      break;&lt;br /&gt;
    }&lt;br /&gt;
  }&lt;br /&gt;
  if (!sheet) {&lt;br /&gt;
    sheet = ss.getSheetByName(&amp;quot;Calendar&amp;quot;) || ss.getSheets()[0];&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  var data = sheet.getDataRange().getValues();&lt;br /&gt;
  var timeZone = Session.getScriptTimeZone();&lt;br /&gt;
  &lt;br /&gt;
  if (VoYDDebug === 1) {&lt;br /&gt;
    Logger.log(&amp;quot;=========== STARTING SHEET SCAN ===========&amp;quot;);&lt;br /&gt;
    Logger.log(&amp;quot;📊 Total rows found in sheet (including header): &amp;quot; + data.length);&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  // Fetch existing calendar events for 2024 to 2026&lt;br /&gt;
  var startRange = new Date(&amp;quot;2024-01-01T00:00:00&amp;quot;);&lt;br /&gt;
  var endRange = new Date(&amp;quot;2036-12-31T23:59:59&amp;quot;);&lt;br /&gt;
  var calendarEvents = calendar.getEvents(startRange, endRange);&lt;br /&gt;
  &lt;br /&gt;
  var calEventsMap = {};&lt;br /&gt;
  for (var j = 0; j &amp;lt; calendarEvents.length; j++) {&lt;br /&gt;
    var ev = calendarEvents[j];&lt;br /&gt;
    var tagKey = ev.getTag(&amp;quot;GIG_KEY&amp;quot;);&lt;br /&gt;
    &lt;br /&gt;
    if (!tagKey) {&lt;br /&gt;
      var evDateStr = Utilities.formatDate(ev.getStartTime(), timeZone, &amp;quot;yyyy-MM-dd&amp;quot;);&lt;br /&gt;
      var evTitle = ev.getTitle();&lt;br /&gt;
      var extractedVenue = &amp;quot;&amp;quot;;&lt;br /&gt;
      if (evTitle.indexOf(&amp;quot;VoYD played &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD played &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      else if (evTitle.indexOf(&amp;quot;VoYD plays &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD plays &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      else if (evTitle.indexOf(&amp;quot;VoYD might play &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD might play &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      &lt;br /&gt;
      tagKey = extractedVenue ? (evDateStr + &amp;quot;_&amp;quot; + extractedVenue) : (evDateStr + &amp;quot;_&amp;quot; + ev.getLocation());&lt;br /&gt;
    }&lt;br /&gt;
    calEventsMap[tagKey] = ev;&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  var validSheetKeys = {};&lt;br /&gt;
  var metrics = { scanned: 0, accepted: 0, rejected: 0 };&lt;br /&gt;
  &lt;br /&gt;
  // ==========================================&lt;br /&gt;
  // PROCESS SPREADSHEET ROWS&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  for (var i = 1; i &amp;lt; data.length; i++) {&lt;br /&gt;
    var row = data[i];&lt;br /&gt;
    metrics.scanned++;&lt;br /&gt;
    &lt;br /&gt;
    var filterValue = String(row[0] || &amp;quot;&amp;quot;).toLowerCase().trim(); &lt;br /&gt;
    var dateRaw = row[1];                                       &lt;br /&gt;
    var venue = String(row[4] || &amp;quot;&amp;quot;).trim();                    &lt;br /&gt;
    var desc = String(row[5] || &amp;quot;&amp;quot;).trim();                     &lt;br /&gt;
    &lt;br /&gt;
    var humanRowNumber = i + 1;&lt;br /&gt;
    &lt;br /&gt;
    // Safe date alignment and year compilation&lt;br /&gt;
    var isDateValid = dateRaw &amp;amp;&amp;amp; !isNaN(new Date(dateRaw).getTime());&lt;br /&gt;
    var yearValue = 0;&lt;br /&gt;
    &lt;br /&gt;
    if (isDateValid) {&lt;br /&gt;
      var parsedDate = new Date(dateRaw);&lt;br /&gt;
      yearValue = parsedDate.getFullYear(); &lt;br /&gt;
    } else {&lt;br /&gt;
      var backupYear = Number(row[2]);&lt;br /&gt;
      if (!isNaN(backupYear)) {&lt;br /&gt;
        yearValue = backupYear;&lt;br /&gt;
      }&lt;br /&gt;
    }&lt;br /&gt;
    &lt;br /&gt;
    var isFilterValid = [&amp;quot;played&amp;quot;, &amp;quot;scheduled&amp;quot;, &amp;quot;potential&amp;quot;].indexOf(filterValue) !== -1;&lt;br /&gt;
    var isYearValid = [2024, 2025, 2026, 2027, 2028, 2029, 2030, 2031, 2032, 2033, 2034, 2035, 2036].indexOf(yearValue) !== -1;&lt;br /&gt;
    &lt;br /&gt;
    if (!isFilterValid || !isYearValid || !isDateValid) {&lt;br /&gt;
      metrics.rejected++;&lt;br /&gt;
      &lt;br /&gt;
      // Conditional Debug Logging for Skips&lt;br /&gt;
      if (VoYDDebug === 1) {&lt;br /&gt;
        var reasons = [];&lt;br /&gt;
        if (!isFilterValid) reasons.push(&amp;quot;Filter value &amp;#039;&amp;quot; + (row[0] || &amp;quot;BLANK&amp;quot;) + &amp;quot;&amp;#039; is not accepted&amp;quot;);&lt;br /&gt;
        if (!isDateValid) reasons.push(&amp;quot;Invalid date format in Column B&amp;quot;);&lt;br /&gt;
        if (isDateValid &amp;amp;&amp;amp; !isYearValid) reasons.push(&amp;quot;Extracted year &amp;#039;&amp;quot; + yearValue + &amp;quot;&amp;#039; is outside 2024-2026&amp;quot;);&lt;br /&gt;
        Logger.log(&amp;quot;❌ Row &amp;quot; + humanRowNumber + &amp;quot; SKIPPED. Reasons: &amp;quot; + reasons.join(&amp;quot; | &amp;quot;));&lt;br /&gt;
      }&lt;br /&gt;
      continue;&lt;br /&gt;
    }&lt;br /&gt;
    &lt;br /&gt;
    // Handled Accepted Rows&lt;br /&gt;
    metrics.accepted++;&lt;br /&gt;
    var dateObj = new Date(dateRaw);&lt;br /&gt;
    var dateStr = Utilities.formatDate(dateObj, timeZone, &amp;quot;yyyy-MM-dd&amp;quot;);&lt;br /&gt;
    &lt;br /&gt;
    if (VoYDDebug === 1) {&lt;br /&gt;
      Logger.log(&amp;quot;✅ Row &amp;quot; + humanRowNumber + &amp;quot; ACCEPTED: &amp;quot; + filterValue.toUpperCase() + &amp;quot; -&amp;gt; &amp;quot; + venue + &amp;quot; (&amp;quot; + dateStr + &amp;quot;)&amp;quot;);&lt;br /&gt;
    }&lt;br /&gt;
    &lt;br /&gt;
    var sheetKey = dateStr + &amp;quot;_&amp;quot; + venue;&lt;br /&gt;
    validSheetKeys[sheetKey] = true;&lt;br /&gt;
    &lt;br /&gt;
    var prefix = &amp;quot;&amp;quot;;&lt;br /&gt;
    if (filterValue === &amp;quot;played&amp;quot;) prefix = &amp;quot;VoYD played &amp;quot;;&lt;br /&gt;
    else if (filterValue === &amp;quot;scheduled&amp;quot;) prefix = &amp;quot;VoYD plays &amp;quot;;&lt;br /&gt;
    else if (filterValue === &amp;quot;potential&amp;quot;) prefix = &amp;quot;VoYD might play &amp;quot;;&lt;br /&gt;
    var title = prefix + venue;&lt;br /&gt;
    &lt;br /&gt;
    var location = (venue &amp;amp;&amp;amp; venue !== &amp;quot;.&amp;quot;) ? venue : &amp;quot;&amp;quot;;&lt;br /&gt;
    &lt;br /&gt;
    var descriptionText = &amp;quot;See VoYD at &amp;quot; + venue + &amp;quot;!\n\n&amp;quot;;&lt;br /&gt;
    if (desc &amp;amp;&amp;amp; desc !== &amp;quot;.&amp;quot;) {&lt;br /&gt;
      descriptionText += desc + &amp;quot;\n\n&amp;quot;;&lt;br /&gt;
    }&lt;br /&gt;
    descriptionText += &amp;quot;For specific information, check out our FB page at &amp;quot; + FACEBOOK_URL + &amp;quot;\n\nHope to see you there!&amp;quot;;&lt;br /&gt;
    &lt;br /&gt;
    // Evaluate additions/modifications&lt;br /&gt;
    if (calEventsMap[sheetKey]) {&lt;br /&gt;
      var existingEvent = calEventsMap[sheetKey];&lt;br /&gt;
      var hasChanged = false;&lt;br /&gt;
      &lt;br /&gt;
      if (existingEvent.getTitle() !== title) hasChanged = true;&lt;br /&gt;
      if (existingEvent.getLocation() !== location) hasChanged = true;&lt;br /&gt;
      if (existingEvent.getDescription() !== descriptionText) hasChanged = true;&lt;br /&gt;
      if (!existingEvent.isAllDayEvent()) hasChanged = true;&lt;br /&gt;
      &lt;br /&gt;
      if (hasChanged) {&lt;br /&gt;
        existingEvent.setTitle(title);&lt;br /&gt;
        existingEvent.setLocation(location);&lt;br /&gt;
        existingEvent.setDescription(descriptionText);&lt;br /&gt;
        existingEvent.setTag(&amp;quot;GIG_KEY&amp;quot;, sheetKey);&lt;br /&gt;
        existingEvent.setVisibility(CalendarApp.Visibility.PUBLIC);&lt;br /&gt;
        Logger.log(&amp;quot;   🔄 Calendar Updated: &amp;quot; + title + &amp;quot; (&amp;quot; + dateStr + &amp;quot;)&amp;quot;);&lt;br /&gt;
      } else if (VoYDDebug === 1) {&lt;br /&gt;
        Logger.log(&amp;quot;   💤 No changes needed for: &amp;quot; + title);&lt;br /&gt;
      }&lt;br /&gt;
    } else {&lt;br /&gt;
      var newEvent = calendar.createAllDayEvent(title, dateObj, {&lt;br /&gt;
        description: descriptionText,&lt;br /&gt;
        location: location&lt;br /&gt;
      });&lt;br /&gt;
      newEvent.setTag(&amp;quot;GIG_KEY&amp;quot;, sheetKey);&lt;br /&gt;
      newEvent.setVisibility(CalendarApp.Visibility.PUBLIC);&lt;br /&gt;
      Logger.log(&amp;quot;   ➕ Calendar Created: &amp;quot; + title + &amp;quot; (&amp;quot; + dateStr + &amp;quot;)&amp;quot;);&lt;br /&gt;
    }&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  if (VoYDDebug === 1) {&lt;br /&gt;
    Logger.log(&amp;quot;=========== STARTING CALENDAR CLEANUP ===========&amp;quot;);&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  // ==========================================&lt;br /&gt;
  // PURGE STALE CALENDAR EVENTS&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  var deletedCount = 0;&lt;br /&gt;
  for (var j = 0; j &amp;lt; calendarEvents.length; j++) {&lt;br /&gt;
    var ev = calendarEvents[j];&lt;br /&gt;
    var tagKey = ev.getTag(&amp;quot;GIG_KEY&amp;quot;);&lt;br /&gt;
    &lt;br /&gt;
    if (!tagKey) {&lt;br /&gt;
      var evDateStr = Utilities.formatDate(ev.getStartTime(), timeZone, &amp;quot;yyyy-MM-dd&amp;quot;);&lt;br /&gt;
      var evTitle = ev.getTitle();&lt;br /&gt;
      var extractedVenue = &amp;quot;&amp;quot;;&lt;br /&gt;
      if (evTitle.indexOf(&amp;quot;VoYD played &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD played &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      else if (evTitle.indexOf(&amp;quot;VoYD plays &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD plays &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      else if (evTitle.indexOf(&amp;quot;VoYD might play &amp;quot;) === 0) extractedVenue = evTitle.replace(&amp;quot;VoYD might play &amp;quot;, &amp;quot;&amp;quot;);&lt;br /&gt;
      &lt;br /&gt;
      tagKey = extractedVenue ? (evDateStr + &amp;quot;_&amp;quot; + extractedVenue) : (evDateStr + &amp;quot;_&amp;quot; + ev.getLocation());&lt;br /&gt;
    }&lt;br /&gt;
    &lt;br /&gt;
    var isManagedEvent = ev.getTag(&amp;quot;GIG_KEY&amp;quot;) || &lt;br /&gt;
                         ev.getTitle().indexOf(&amp;quot;VoYD played &amp;quot;) === 0 || &lt;br /&gt;
                         ev.getTitle().indexOf(&amp;quot;VoYD plays &amp;quot;) === 0 || &lt;br /&gt;
                         ev.getTitle().indexOf(&amp;quot;VoYD might play &amp;quot;) === 0;&lt;br /&gt;
    &lt;br /&gt;
    if (isManagedEvent &amp;amp;&amp;amp; !validSheetKeys[tagKey]) {&lt;br /&gt;
      Logger.log(&amp;quot;🗑️ Calendar Deleted (no longer in sheet/valid): &amp;quot; + ev.getTitle() + &amp;quot; on &amp;quot; + Utilities.formatDate(ev.getStartTime(), timeZone, &amp;quot;yyyy-MM-dd&amp;quot;));&lt;br /&gt;
      ev.deleteEvent();&lt;br /&gt;
      deletedCount++;&lt;br /&gt;
    }&lt;br /&gt;
  }&lt;br /&gt;
  &lt;br /&gt;
  // ==========================================&lt;br /&gt;
  // FINAL METRICS SUMMARY (Always Displays)&lt;br /&gt;
  // ==========================================&lt;br /&gt;
  Logger.log(&amp;quot;=========== FINAL PERFORMANCE SUMMARY ===========&amp;quot;);&lt;br /&gt;
  Logger.log(&amp;quot;📋 Total Data Rows Scanned: &amp;quot; + metrics.scanned);&lt;br /&gt;
  Logger.log(&amp;quot;🎉 Total Rows Accepted:     &amp;quot; + metrics.accepted);&lt;br /&gt;
  Logger.log(&amp;quot;🚫 Total Rows Filtered Out: &amp;quot; + metrics.rejected);&lt;br /&gt;
  Logger.log(&amp;quot;🗑️ Calendar Events Purged:  &amp;quot; + deletedCount);&lt;br /&gt;
  Logger.log(&amp;quot;=================================================&amp;quot;);&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;/div&gt;</summary>
		<author><name>Rabbi Bob</name></author>
	</entry>
</feed>