@@ -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 |