Skip to content

Commit

Permalink
Merge branch 'staging' of github.com:digital-land/lpa-data-validator-…
Browse files Browse the repository at this point in the history
…frontend into staging
  • Loading branch information
GeorgeGoodall committed Aug 2, 2024
2 parents 7d2ca81 + 4167d95 commit ba1e318
Show file tree
Hide file tree
Showing 33 changed files with 445 additions and 175 deletions.
8 changes: 5 additions & 3 deletions index.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,16 +11,18 @@ import { setupErrorHandlers } from './src/serverSetup/errorHandlers.js'
import { setupSession } from './src/serverSetup/session.js'
import { setupNunjucks } from './src/serverSetup/nunjucks.js'
import { setupSentry } from './src/serverSetup/sentry.js'

import { dataSubjects } from './src/utils/utils.js'
import { getDatasetSlugNameMapping } from './src/utils/datasetteQueries/getDatasetSlugNameMapping.js'

dotenv.config()

const app = express()

setupMiddlewares(app)
setupSession(app)
setupNunjucks({ app, dataSubjects })
setupNunjucks({
app,
datasetNameMapping: await getDatasetSlugNameMapping()
})
setupRoutes(app)
setupSentry(app)
setupErrorHandlers(app)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ const availableDatasets = Object.values(dataSubjects)
.map(dataset => dataset.value)
)

const LpaOverviewController = {
const organisationsController = {
/**
* Get LPA overview data and render the overview page
* @param {Request} req - Express request object
Expand Down Expand Up @@ -63,12 +63,16 @@ const LpaOverviewController = {
datasetsWithErrors
}

res.render('manage/lpa-overview.html', params)
res.render('organisations/overview.html', params)
} catch (error) {
logger.error(error)
next(error)
}
},

async getOrganisations (req, res, next) {
res.render('organisations/find.html')
}
}

export default LpaOverviewController
export default organisationsController
2 changes: 1 addition & 1 deletion src/controllers/lpaDetailsController.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import PageController from './pageController.js'
import { fetchLocalAuthorities } from '../services/fetchLocalAuthorities.js'
import { fetchLocalAuthorities } from '../utils/datasetteQueries/fetchLocalAuthorities.js'

class LpaDetailsController extends PageController {
async locals (req, res, next) {
Expand Down
5 changes: 2 additions & 3 deletions src/filters/filters.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,11 @@ import validationMessageLookup from './validationMessageLookup.js'
import toErrorList from './toErrorList.js'
import prettifyColumnName from './prettifyColumnName.js'
import getFullServiceName from './getFullServiceName.js'
import { makeDatasetSlugToReadableNameFilter, createDatasetMapping } from './makeDatasetSlugToReadableNameFilter.js'
import { makeDatasetSlugToReadableNameFilter } from './makeDatasetSlugToReadableNameFilter.js'

const { govukMarkdown } = xGovFilters

const addFilters = (nunjucksEnv, { dataSubjects }) => {
const datasetNameMapping = createDatasetMapping(dataSubjects)
const addFilters = (nunjucksEnv, { datasetNameMapping }) => {
const datasetSlugToReadableName = makeDatasetSlugToReadableNameFilter(datasetNameMapping)
nunjucksEnv.addFilter('datasetSlugToReadableName', datasetSlugToReadableName)

Expand Down
11 changes: 7 additions & 4 deletions src/filters/getFullServiceName.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,10 @@
import config from '../../config/index.js'

export default (service) => {
const serviceName = config.serviceName

return serviceName.replace('Provide', service)
const getFullServiceName = (service) => {
if (!service || typeof service !== 'string') {
throw new Error('Service name must be a non-empty string')
}
return config.serviceName.replace('Provide', service)
}

export default getFullServiceName
15 changes: 0 additions & 15 deletions src/filters/makeDatasetSlugToReadableNameFilter.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,18 +26,3 @@ export const makeDatasetSlugToReadableNameFilter = (datasetNameMapping) => {
return name
}
}

/**
*
* @param {*} dataSubjects
* @returns {Map<string,string>}
*/
export const createDatasetMapping = (dataSubjects) => {
const mapping = new Map()
for (const data of Object.values(dataSubjects)) {
for (const dataset of data.dataSets) {
mapping.set(dataset.value, dataset.text)
}
}
return mapping
}
8 changes: 0 additions & 8 deletions src/routes/manage.js

This file was deleted.

10 changes: 10 additions & 0 deletions src/routes/organisations.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
import express from 'express'
import OrganisationsController from '../controllers/OrganisationsController.js'

const router = express.Router()

router.get('/', OrganisationsController.getOrganisations)

router.get('/:lpa/overview', OrganisationsController.getOverview)

export default router
4 changes: 2 additions & 2 deletions src/serverSetup/nunjucks.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import nunjucks from 'nunjucks'
import config from '../../config/index.js'
import addFilters from '../filters/filters.js'

export function setupNunjucks ({ app, dataSubjects }) {
export function setupNunjucks ({ app, datasetNameMapping }) {
if (app) {
app.set('view engine', 'html')
}
Expand Down Expand Up @@ -33,7 +33,7 @@ export function setupNunjucks ({ app, dataSubjects }) {
Object.keys(globalValues).forEach((key) => {
nunjucksEnv.addGlobal(key, globalValues[key])
})
addFilters(nunjucksEnv, { dataSubjects })
addFilters(nunjucksEnv, { datasetNameMapping })

return nunjucks
}
10 changes: 6 additions & 4 deletions src/serverSetup/routes.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,17 +3,19 @@ import endpointSubmissionFormFormWisard from '../routes/form-wizard/endpoint-sub
import accessibility from '../routes/accessibility.js'
import polling from '../routes/api.js'
import health from '../routes/health.js'
import manage from '../routes/manage.js'
import organisations from '../routes/organisations.js'
import privacy from '../routes/privacy.js'
import cookies from '../routes/cookies.js'

export function setupRoutes (app) {
app.use('/', checkFormWizard)
app.use('/submit', endpointSubmissionFormFormWisard)
app.use('/accessibility', accessibility)
app.use('/organisations', organisations)

app.use('/api', polling)
app.use('/health', health)
app.use('/manage', manage)

app.use('/accessibility', accessibility)
app.use('/privacy-notice', privacy)
app.use('/cookies', cookies)
app.use('/health', health)
}
31 changes: 30 additions & 1 deletion src/services/datasette.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,44 @@ const datasetteUrl = 'https://datasette.planning.data.gov.uk'
const database = 'digital-land'

export default {
/**
* Executes a SQL query on the Datasette instance and returns the results.
*
* @param {string} query - The SQL query to execute.
* @returns {Promise<{data: object, formattedData: object}>} - A promise that resolves to an object with the following properties:
* - `data`: The raw data returned by Datasette.
* - `formattedData`: The formatted data, with columns and rows parsed into a usable format.
* @throws {Error} If the query fails or there is an error communicating with Datasette.
*/
runQuery: async (query) => {
const encodedQuery = encodeURIComponent(query)
const url = `${datasetteUrl}/${database}.json?sql=${encodedQuery}`
try {
const response = await axios.get(url)
return response.data
return {
...response.data,
formattedData: formatData(response.data.columns, response.data.rows)
}
} catch (error) {
logger.warn(error)
throw error
}
}
}

/**
* Formats an array of rows into an easier to access format, where each row is an object with column names as keys.
*
* @param {string[]} columns - An array of column names
* @param {any[][]} rows - A 2D array of row data, where each inner array represents a row
* @returns {object[]} - An array of objects, where each object represents a row with column names as keys
*/
export function formatData (columns, rows) {
// convert the rows into an easier to access format
return rows.map((row) => {
return row.reduce((acc, val, index) => {
acc[columns[index]] = val
return acc
}, {})
})
}
17 changes: 4 additions & 13 deletions src/services/performanceDbApi.js
Original file line number Diff line number Diff line change
Expand Up @@ -86,19 +86,10 @@ ORDER BY

const result = await datasette.runQuery(query)

// convert the rows into an easier to access format
const columns = result.columns
const rows = result.rows.map((row) => {
return row.reduce((acc, val, index) => {
acc[columns[index]] = val
return acc
}, {})
})

const datasets = rows.reduce((accumulator, row) => {
const datasets = result.formattedData.reduce((accumulator, row) => {
let error
if (row.http_status !== '200' || row.exception !== '') {
error = row.exception !== '' ? row.exception : `endpoint returned with a status of ${row.http_status}`
if (row.http_status !== '200' || row.exception) {
error = row.exception ? row.exception : `endpoint returned with a status of ${row.http_status}`
}

let issue
Expand All @@ -115,7 +106,7 @@ ORDER BY
}, {})

return {
name: result.rows[0][1],
name: result.formattedData[0].name,
datasets
}
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import axios from 'axios'
import logger from '../../src/utils/logger.js'
import datasette from '../../services/datasette.js'
import logger from '../logger.js'

/**
* Fetches a list of local authority names from a specified dataset.
Expand All @@ -24,15 +24,14 @@ export const fetchLocalAuthorities = async () => {
order by
provision.organisation`

const url = `https://datasette.planning.data.gov.uk/digital-land.json?sql=${encodeURIComponent(sql)}`
try {
const response = await axios.get(url)
const names = response.data.rows.map(row => {
if (row[1] === null) {
const response = await datasette.runQuery(sql)
const names = response.formattedData.map(row => {
if (row.name == null) {
logger.debug('Null value found in response:', row)
return null
} else {
return row[1]
return row.name
}
}).filter(name => name !== null) // Filter out null values
return names
Expand Down
11 changes: 11 additions & 0 deletions src/utils/datasetteQueries/getDatasetSlugNameMapping.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import datasette from '../../services/datasette.js'

export const getDatasetSlugNameMapping = async () => {
const datasetSlugNameTable = await datasette.runQuery('select dataset, name from dataset')

const datasetMapping = new Map()
datasetSlugNameTable.rows.forEach(([slug, name]) => {
datasetMapping.set(slug, name)
})
return datasetMapping
}
26 changes: 26 additions & 0 deletions src/views/organisations/find.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
{% extends "layouts/main.html" %}

{% from "govuk/components/breadcrumbs/macro.njk" import govukBreadcrumbs %}
{% from "govuk/components/tag/macro.njk" import govukTag %}

{% block beforeContent %}
{{ super() }}

{% endblock %}

{% block content %}

<div class="govuk-grid-row">
<div class="govuk-grid-column-two-thirds">

<h1 class="govuk-heading-xl">
{{ pageName }}
</h1>
</div>
</div>

<h1>Find page placeholder</h1>



{% endblock %}
File renamed without changes.
5 changes: 2 additions & 3 deletions test/unit/check-answers.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ import { setupNunjucks } from '../../src/serverSetup/nunjucks.js'
import { runGenericPageTests } from './generic-page.js'
import config from '../../config/index.js'
import { stripWhitespace } from '../utils/stripWhiteSpace.js'
import { mockDataSubjects } from './data.js'

describe('check-answers View', async () => {
const params = {
Expand All @@ -19,7 +18,7 @@ describe('check-answers View', async () => {
hasLicence: 'true'
}
}
const nunjucks = setupNunjucks({ dataSubjects: mockDataSubjects })
const nunjucks = setupNunjucks({ datasetNameMapping: new Map() })
const html = stripWhitespace(nunjucks.render('check-answers.html', params))

runGenericPageTests(html, {
Expand All @@ -43,7 +42,7 @@ describe('check-answers View', async () => {
})

it('should render the dataset entered', () => {
const datasetRegex = new RegExp('<div class="govuk-summary-list__row">.*Dataset.*A Mock dataset.*Change.*</div>', 'g')
const datasetRegex = new RegExp('<div class="govuk-summary-list__row">.*Dataset.*mockDataset.*Change.*</div>', 'g')
expect(html).toMatch(datasetRegex)
})

Expand Down
7 changes: 3 additions & 4 deletions test/unit/check/confirmationPage.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,8 @@ import { setupNunjucks } from '../../../src/serverSetup/nunjucks.js'
import { runGenericPageTests } from '../generic-page.js'
import config from '../../../config/index.js'
import { stripWhitespace } from '../../utils/stripWhiteSpace.js'
import { mockDataSubjects } from '../data.js'

const nunjucks = setupNunjucks({ dataSubjects: mockDataSubjects })
const nunjucks = setupNunjucks({ datasetNameMapping: new Map() })

describe('Check confirmation View', () => {
const params = {
Expand All @@ -18,12 +17,12 @@ describe('Check confirmation View', () => {
const html = stripWhitespace(nunjucks.render('submit/confirmation.html', params))

runGenericPageTests(html, {
pageTitle: 'A Mock dataset submitted - Submit planning and housing data for England',
pageTitle: 'mockDataset submitted - Submit planning and housing data for England',
serviceName: config.serviceName
})

it('should render the gov uk panel', () => {
const regex = new RegExp('<h1 class="govuk-panel__title".*A Mock dataset submitted.*</h1>', 'g')
const regex = new RegExp('<h1 class="govuk-panel__title".*mockDataset submitted.*</h1>', 'g')
expect(html).toMatch(regex)
})
})
2 changes: 1 addition & 1 deletion test/unit/choose-datasetPage.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { runGenericPageTests } from './generic-page.js'
import config from '../../config/index.js'
import { testValidationErrorMessage } from './validation-tests.js'

const nunjucks = setupNunjucks({ dataSubjects: {} })
const nunjucks = setupNunjucks({ datasetNameMapping: new Map() })

describe('choose dataset View', () => {
const params = {
Expand Down
4 changes: 2 additions & 2 deletions test/unit/dataset-details.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import { stripWhitespace } from '../utils/stripWhiteSpace.js'
import { testValidationErrorMessage } from './validation-tests.js'
import { mockDataSubjects } from './data.js'

const nunjucks = setupNunjucks({ dataSubjects: mockDataSubjects })
const nunjucks = setupNunjucks({ datasetNameMapping: new Map() })

function errorTestFn ({
params,
Expand Down Expand Up @@ -41,7 +41,7 @@ describe('dataset details View', () => {
errors: {}
}
const html = stripWhitespace(nunjucks.render('dataset-details.html', params))
const datasetName = mockDataSubjects.mockDataset.dataSets[0].text
const datasetName = mockDataSubjects.mockDataset.dataSets[0].value
runGenericPageTests(html, {
pageTitle: `Enter ${datasetName.toLowerCase()} details - Submit planning and housing data for England`,
serviceName: config.serviceName
Expand Down
Loading

0 comments on commit ba1e318

Please sign in to comment.