devonportcity.rb 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. # Devonport City Council — Advertised Planning Permit Applications (WP File Download)
  2. require "date"
  3. require "nokogiri"
  4. require "fileutils"
  5. require "net/http"
  6. require "uri"
  7. require_relative "../lib/http"
  8. require_relative "../lib/db"
  9. require_relative "../lib/util"
  10. require_relative "../lib/geocode"
  11. require_relative "../lib/enrich"
  12. TABLE = ENV.fetch("TABLE_NAME") # run_all.sh -> da_devonportcity
  13. URL = "https://www.devonport.tas.gov.au/building-development/planning/advertised-planning-permit-applications/"
  14. DOWNLOAD_ATTACHMENTS = ENV["DOWNLOAD_ATTACHMENTS"] == "1"
  15. DOWNLOAD_DIR = ENV["DOWNLOAD_DIR"] || "/app/downloads"
  16. DB.ensure_table!(TABLE)
  17. ensure_extra_columns!(TABLE)
  18. def abs_url(base, href)
  19. return "" if href.to_s.strip.empty?
  20. URI.join(base, href).to_s rescue href.to_s
  21. end
  22. def sanitize_filename(s)
  23. s.to_s.gsub(/[^\w.\-]+/, "_")[0, 180]
  24. end
  25. # ---------- Reference + date helpers ----------
  26. # Accepts PA/DA with separators like ".", "/", "-" and optional spaces
  27. def normalize_ref(str)
  28. s = str.to_s.strip
  29. if (m = s.match(/\b(PA|DA)\s*([12]\d{3})[.\-\/\s]+([A-Za-z0-9]{3,})\b/i))
  30. "#{m[1].upcase} #{m[2]} / #{m[3]}"
  31. elsif (m = s.match(/\bpa([12]\d{3})[.\-]([A-Za-z0-9]{3,})\b/i))
  32. "PA #{m[1]} / #{m[2]}"
  33. elsif (m = s.match(/\bda([12]\d{3})[.\-]([A-Za-z0-9]{3,})\b/i))
  34. "DA #{m[1]} / #{m[2]}"
  35. else
  36. nil
  37. end
  38. end
  39. def extract_date_token(str)
  40. s = str.to_s
  41. return s[/\b\d{1,2}\/\d{1,2}\/\d{2,4}\b/] ||
  42. s[/\b\d{1,2}\s+[A-Za-z]{3,}\s+\d{4}\b/] ||
  43. s[/\b[A-Za-z]{3,}\s+\d{1,2},?\s+\d{4}\b/] ||
  44. s[/\b\d{1,2}-\d{1,2}-\d{4}\b/] ||
  45. ""
  46. end
  47. def parse_date_any(s)
  48. return nil if s.to_s.strip.empty?
  49. Util.parse_aus_date(s) rescue nil
  50. end
  51. def extract_on_notice_to_from_title(title)
  52. # Prefer explicit phrase
  53. if (m = title.to_s.match(/advertising\s+period\s+ends?\s+(.+?)\s*(?:$|\(|-)/i))
  54. tkn = extract_date_token(m[1])
  55. return parse_date_any(tkn), tkn unless tkn.empty?
  56. end
  57. # Fallback: any date token in the title
  58. tkn = extract_date_token(title)
  59. [parse_date_any(tkn), tkn]
  60. end
  61. # ---------- Simple PDF downloader ----------
  62. def download_pdf(url, council_reference)
  63. return if url.to_s.strip.empty?
  64. return unless DOWNLOAD_ATTACHMENTS
  65. uri = URI(url)
  66. out_dir = File.join(DOWNLOAD_DIR, TABLE)
  67. FileUtils.mkdir_p(out_dir)
  68. base = sanitize_filename(File.basename(uri.path))
  69. prefix = sanitize_filename(council_reference.to_s.gsub(" / ", "-"))
  70. out_path = File.join(out_dir, "#{prefix}__#{base}")
  71. Net::HTTP.start(uri.host, uri.port, use_ssl: uri.scheme == "https") do |http|
  72. req = Net::HTTP::Get.new(uri)
  73. req["User-Agent"] = "Mozilla/5.0"
  74. req["Accept"] = "application/pdf,*/*;q=0.8"
  75. req["Accept-Encoding"] = "identity"
  76. http.request(req) do |resp|
  77. if resp.code.to_i == 200
  78. File.open(out_path, "wb") do |f|
  79. resp.read_body { |chunk| f.write(chunk) }
  80. end
  81. puts "Saved PDF to #{out_path}"
  82. else
  83. warn "PDF fetch failed (#{resp.code} #{resp.message}) for #{url}"
  84. end
  85. end
  86. end
  87. rescue => e
  88. warn "PDF save error for #{url}: #{e.class} #{e.message}"
  89. end
  90. # ---------- Fetch + parse ----------
  91. html = Http.get(URL)
  92. doc = Nokogiri::HTML(html)
  93. # Devonport uses WP File Download. Rows live under .wpfd-search-result > table > tbody > tr
  94. rows = doc.css(".wpfd-search-result tbody tr")
  95. puts "Found #{rows.length} row(s) for #{TABLE}"
  96. saved = 0
  97. rows.each_with_index do |row, idx|
  98. link = row.at_css("a.wpfd_downloadlink")
  99. next unless link
  100. title_reference = link["title"].to_s.strip
  101. href = link["href"].to_s.strip
  102. document_url = abs_url(URL, href)
  103. # Typical title:
  104. # "PA2025.0103 - 11-17 Stewart Street Devonport - Signage - Advertising period ends 2 September 2025"
  105. parts = title_reference.split(" - ").map(&:strip)
  106. raw_ref = parts[0].to_s
  107. address = parts[1].to_s
  108. # Description: all middle parts until the last one (often the date/notice bit)
  109. middle = (parts[2..-1] || [])
  110. # Pull out on-notice first so we can filter
  111. on_notice_to, on_notice_to_raw = extract_on_notice_to_from_title(title_reference)
  112. middle.reject! { |p| p =~ /advertising\s+period\s+ends?/i || p == on_notice_to_raw }
  113. description = middle.join(" - ").strip
  114. description = "Development Application" if description.empty?
  115. # Date added column -> date_received
  116. date_received_raw = row.at_css(".file_created")&.text&.strip.to_s
  117. date_received = parse_date_any(date_received_raw)
  118. if date_received.nil? && !date_received_raw.empty?
  119. # handle 19-08-2025
  120. begin
  121. date_received = Date.strptime(date_received_raw, "%d-%m-%Y")
  122. rescue ArgumentError, Date::Error
  123. date_received = parse_date_any(date_received_raw)
  124. end
  125. end
  126. # Normalize / derive council_reference
  127. council_reference = normalize_ref(raw_ref) ||
  128. normalize_ref(title_reference) ||
  129. normalize_ref(document_url) ||
  130. raw_ref # last resort (raw)
  131. # Fallback address if missing
  132. if address.to_s.empty?
  133. # Try data-filetitle on hidden input (same text as title)
  134. hidden_title = row.at_css("input.wpfd_file_preview_link_download")&.[]("data-filetitle").to_s
  135. parts2 = hidden_title.split(" - ").map(&:strip)
  136. address = parts2[1].to_s unless parts2.empty?
  137. address = title_reference[0, 140] if address.to_s.empty?
  138. end
  139. next if council_reference.to_s.empty? || address.to_s.empty?
  140. # Download PDF if requested
  141. download_pdf(document_url, council_reference)
  142. DB.upsert(TABLE, {
  143. description: description,
  144. date_received: date_received,
  145. date_received_raw: date_received_raw,
  146. on_notice_to: on_notice_to,
  147. on_notice_to_raw: on_notice_to_raw,
  148. address: address,
  149. council_reference: council_reference,
  150. applicant: "",
  151. owner: ""
  152. })
  153. enrich_after_upsert!(
  154. table: TABLE,
  155. council_reference: council_reference,
  156. address: address
  157. )
  158. # Store extras if columns exist
  159. begin
  160. upd = DB.client.prepare("UPDATE `#{DB.client.escape(TABLE)}` SET document_url = ?, title_reference = ? WHERE council_reference = ? AND address = ?")
  161. upd.execute(document_url, title_reference, council_reference, address)
  162. rescue Mysql2::Error => e
  163. warn "[devonportcity] db update skipped for #{council_reference}: #{e.message}"
  164. end
  165. puts "Upserted #{council_reference} -> #{address}"
  166. saved += 1
  167. end
  168. puts "Done #{TABLE}. Saved #{saved} item(s)."