@@ -0,0 +1,8 @@ | |||||
/.bundle/ | |||||
/.yardoc | |||||
/_yardoc/ | |||||
/coverage/ | |||||
/doc/ | |||||
/pkg/ | |||||
/spec/reports/ | |||||
/tmp/ |
@@ -0,0 +1,9 @@ | |||||
# Changelog | |||||
All notable changes to this project will be documented in this file. | |||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | |||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | |||||
## 0.1 - 2020-04-30 | |||||
First release. |
@@ -0,0 +1,6 @@ | |||||
source "https://rubygems.org" | |||||
# Specify your gem's dependencies in merton_planning_formatter.gemspec | |||||
gemspec | |||||
gem "rake", "~> 12.0" |
@@ -0,0 +1,29 @@ | |||||
PATH | |||||
remote: . | |||||
specs: | |||||
merton_planning_formatter (0.1.0) | |||||
pragmatic_segmenter | |||||
GEM | |||||
remote: https://rubygems.org/ | |||||
specs: | |||||
coderay (1.1.2) | |||||
method_source (1.0.0) | |||||
pragmatic_segmenter (0.3.22) | |||||
unicode | |||||
pry (0.13.1) | |||||
coderay (~> 1.1) | |||||
method_source (~> 1.0) | |||||
rake (12.3.3) | |||||
unicode (0.4.4.4) | |||||
PLATFORMS | |||||
ruby | |||||
DEPENDENCIES | |||||
merton_planning_formatter! | |||||
pry (~> 0.11) | |||||
rake (~> 12.0) | |||||
BUNDLED WITH | |||||
2.1.4 |
@@ -0,0 +1,67 @@ | |||||
# Merton Planning Formatter | |||||
Formats Merton Council planning application descriptions/proposals nicely. | |||||
Converts uppercase descriptions to a readable mixed case format while preserving capitalisation for some proper nouns (eg Merton) and abbreviations (eg LBM, WC), [use classes](https://www.planningportal.co.uk/info/200130/common_projects/9/change_of_use) (eg A1, B1(c), D2), trees (eg TPO, M123, T20) and application reference numbers (eg 19/P1234). | |||||
## Installation | |||||
Add this line to your application's Gemfile: | |||||
```ruby | |||||
gem 'merton_planning_formatter' | |||||
``` | |||||
And then execute: | |||||
$ bundle install | |||||
Or install it yourself as: | |||||
$ gem install merton_planning_formatter | |||||
## Usage | |||||
Some real examples: | |||||
```ruby | |||||
require 'merton_planning_formatter' | |||||
description = "APPLICATION TO DISCHARGE CONDITION 22 (CAR LIFT) ATTACHED TO LBM PLANNING PERMISSION 19/P0140 (PART DEMOLITION AND ERECTION OF A REPLACEMENT DWELLINGHOUSE)" | |||||
puts MertonPlanningFormatter::format(description) | |||||
# => Application to discharge condition 22 (car lift) attached to LBM planning permission 19/P0140 (part demolition and erection of a replacement dwellinghouse) | |||||
description = "APPLICATION TO DETERMINE WHETHER PRIOR APPROVAL IS REQUIRED IN RESPECT OF THE PROPOSED CHANGE OF USE FROM OFFICE SPACE (CLASS B1a) TO RESIDENTIAL (CLASS C3) TO CREATE 1 RESIDENTIAL UNIT (CLASS C3)" | |||||
puts MertonPlanningFormatter::format(description) | |||||
# => Application to determine whether prior approval is required in respect of the proposed change of use from office space (class B1a) to residential (class C3) to create 1 residential unit (class C3) | |||||
description = "APPLICATION FOR DISCHARGE OF CONDITION 25 ATTACHED LBM PLANNING PERMISSION 18/P4447 RELATING TO THE DEMOLITION OF EXISTING BUILDINGS AND STRUCTURES, AND REDEVELOPMENT FOR A NEW 8 - STOREY BUILDING (PLUS ADDITIONAL PLANT AT ROOF LEVEL) COMPRISING OF A HOTEL (USE CLASS C1) AND THREE COMMERCIAL UNITS (A FLEXIBLE USE WITHIN CLASSES A1, A2, A3 AND / OR A4); SUBSTATION; ALTERATIONS TO EXISTING ACCESS AND CREATION OF NEW ACCESS ON GRAHAM ROAD; HARD AND SOFT LANDSCAPING, GROUND WORKS AND ASSOCIATED INFRASTRUCTURE." | |||||
puts MertonPlanningFormatter::format(description) | |||||
# => Application for discharge of condition 25 attached LBM planning permission 18/P4447 relating to the demolition of existing buildings and structures, and redevelopment for a new 8 - storey building (plus additional plant at roof level) comprising of a hotel (use class C1) and three commercial units (a flexible use within classes A1, A2, A3 and / or A4); substation; alterations to existing access and creation of new access on graham road; hard and soft landscaping, ground works and associated infrastructure. | |||||
# Note that "Graham Road" is not capitalised. | |||||
description = "TPO M529: FRONT GARDEN - MATURE BEECH TREE (T2) TO BE REMOVED DUE TO STRUCTURAL WEAKNESS IN STEM AND UPPER CANOPY (REPLACEMENT PLANTING PROPOSED)." | |||||
puts MertonPlanningFormatter::format(description) | |||||
# => TPO M529: front garden - mature beech tree (T2) to be removed due to structural weakness in stem and upper canopy (replacement planting proposed). | |||||
``` | |||||
## Development | |||||
After checking out the repo, run `bin/setup` to install dependencies. You can also run `bin/console` for an interactive prompt that will allow you to experiment. | |||||
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org). | |||||
## Contributing | |||||
Bug reports and pull requests are welcome on GitHub at https://github.com/adrianshort/merton_planning_formatter. | |||||
@@ -0,0 +1,2 @@ | |||||
require "bundler/gem_tasks" | |||||
task :default => :spec |
@@ -0,0 +1,14 @@ | |||||
#!/usr/bin/env ruby | |||||
require "bundler/setup" | |||||
require "merton_planning_formatter" | |||||
# You can add fixtures and/or initialization code here to make experimenting | |||||
# with your gem easier. You can also use a different console, if you like. | |||||
# (If you use this, don't forget to add pry to your Gemfile!) | |||||
# require "pry" | |||||
# Pry.start | |||||
require "pry" | |||||
pry.start |
@@ -0,0 +1,8 @@ | |||||
#!/usr/bin/env bash | |||||
set -euo pipefail | |||||
IFS=$'\n\t' | |||||
set -vx | |||||
bundle install | |||||
# Do any other automated setup that you need to do here |
@@ -0,0 +1,71 @@ | |||||
require "merton_planning_formatter/version" | |||||
require 'pragmatic_segmenter' | |||||
module MertonPlanningFormatter | |||||
class Error < StandardError; end | |||||
def self.format(s) | |||||
ps = PragmaticSegmenter::Segmenter.new(text: s, language: 'en') | |||||
sentences = ps.segment | |||||
output = [] | |||||
sentences.each do |sentence| | |||||
sentence = sentence.downcase.capitalize | |||||
# Patterns have uppercase and lowercase alertnatives for the first letter to ensure that they match at the start of a sentence when the first letter will already be capitalised. | |||||
# Abbreviations | |||||
sentence.gsub! /\b[lL]bm\b/, 'LBM' | |||||
sentence.gsub! /\b[lL]b\b/, 'LB' | |||||
sentence.gsub! /\b[wW]c\b/, 'WC' | |||||
sentence.gsub! /\b[lL]ondon borough of merton\b/, 'London Borough of Merton' | |||||
sentence.gsub! /\b[mM]erton\b/, 'Merton' | |||||
# Districts | |||||
# https://en.wikipedia.org/wiki/London_Borough_of_Merton#Districts | |||||
sentence.gsub! /\b[bB]ushey mead\b/, 'Bushey Mead' | |||||
sentence.gsub! /\b[cC]olliers wood\b/, 'Colliers Wood' | |||||
sentence.gsub! /\b[cC]opse hill\b/, 'Copse Hill' | |||||
sentence.gsub! /\b[cC]ottenham park\b/, 'Cottenham Park' | |||||
sentence.gsub! /\b[cC]rooked bill?ett?\b/, 'Crooked Billet' | |||||
sentence.gsub! /\b[lL]ower morden\b/, 'Lower Morden' | |||||
sentence.gsub! /\b[mM]erton park\b/, 'Merton Park' | |||||
sentence.gsub! /\b[mM]itcham\b/, 'Mitcham' | |||||
sentence.gsub! /\b[mM]itcham common\b/, 'Mitcham Common' | |||||
sentence.gsub! /\b[mM]orden\b/, 'Morden' | |||||
sentence.gsub! /\b[mM]orden park\b/, 'Morden Park' | |||||
sentence.gsub! /\b[mM]otspur park\b/, 'Motspur Park' | |||||
sentence.gsub! /\b[nN]ew malden\b/, 'New Malden' | |||||
sentence.gsub! /\b[nN]orbury\b/, 'Norbury' | |||||
sentence.gsub! /\b[pP]ollards hill\b/, 'Pollards Hill' | |||||
sentence.gsub! /\b[rR]aynes park\b/, 'Raynes Park' | |||||
sentence.gsub! /\b[sS]t\.? Helier\b/, 'St. Helier' | |||||
sentence.gsub! /\b[sS]outh Wimbledon\b/, 'South Wimbledon' | |||||
sentence.gsub! /\b[sS]ummerstown\b/, 'Summerstown' | |||||
sentence.gsub! /\b[wW]imbledon\b/, 'Wimbledon' | |||||
sentence.gsub! /\b[wW]imbledon park\b/, 'Wimbledon Park' | |||||
# Use classes | |||||
# https://www.planningportal.co.uk/info/200130/common_projects/9/change_of_use | |||||
sentence.gsub! /\b[aA](\d)\b/, 'A\1' | |||||
sentence.gsub! /\b[bB](\d)([abc]?)\b/, 'B\1\2' | |||||
sentence.gsub! /\b[cC](\d)([abc]?)\b/, 'C\1\2' | |||||
sentence.gsub! /\b[dD](\d)\b/, 'D\1' | |||||
# Trees | |||||
sentence.gsub! /\b[tT]po\b/, 'TPO' | |||||
sentence.gsub! /\b[tT](\d+)\b/, 'T\1' | |||||
sentence.gsub! /\b[mM]er(\d+)\b/, 'MER\1' | |||||
sentence.gsub! /\b[mM](\d+)\b/, 'M\1' | |||||
# Case reference numbers, eg 18/P1234 | |||||
sentence.gsub! /\b(\d{2})\/p(\d+)\b/, '\1/P\2' | |||||
output << sentence | |||||
end | |||||
output.join(' ') | |||||
end | |||||
end |
@@ -0,0 +1,3 @@ | |||||
module MertonPlanningFormatter | |||||
VERSION = "0.1.0" | |||||
end |
@@ -0,0 +1,29 @@ | |||||
require_relative 'lib/merton_planning_formatter/version' | |||||
Gem::Specification.new do |spec| | |||||
spec.name = "merton_planning_formatter" | |||||
spec.version = MertonPlanningFormatter::VERSION | |||||
spec.authors = ["Adrian Short"] | |||||
spec.email = ["adrian@adrianshort.org"] | |||||
spec.summary = %q{Formats planning application descriptions/proposals nicely.} | |||||
# spec.description = %q{TODO: Write a longer description or delete this line.} | |||||
spec.homepage = "https://github.com/adrianshort/merton_planning_formatter" | |||||
spec.required_ruby_version = Gem::Requirement.new(">= 2.3.0") | |||||
spec.licenses = ['LGPL-3.0'] | |||||
spec.metadata["homepage_uri"] = spec.homepage | |||||
spec.metadata["source_code_uri"] = spec.homepage | |||||
spec.metadata["changelog_uri"] = "https://github.com/adrianshort/merton_planning_formatter/CHANGELOG.md" | |||||
# Specify which files should be added to the gem when it is released. | |||||
# The `git ls-files -z` loads the files in the RubyGem that have been added into git. | |||||
spec.files = Dir.chdir(File.expand_path('..', __FILE__)) do | |||||
`git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) } | |||||
end | |||||
spec.bindir = "exe" | |||||
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) } | |||||
spec.require_paths = ["lib"] | |||||
spec.add_development_dependency "pry", "~> 0.11" | |||||
spec.add_runtime_dependency "pragmatic_segmenter" | |||||
end |