Skip to content

Commit

Permalink
Fix URL IDs rules (#446)
Browse files Browse the repository at this point in the history
  • Loading branch information
wazolab authored Dec 17, 2024
1 parent 611d13f commit a375ab3
Show file tree
Hide file tree
Showing 12 changed files with 116 additions and 113 deletions.
6 changes: 6 additions & 0 deletions composables/useIdsResolver.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
export const regexForCategoryIds = /^(?:(?<cartocode>cartocode:\w{2})|(?<reference>ref:[\w-]+:\w+)|(?<osm>osm:[nwr]\d+)|\d+(?:,\d+)*)$/
export const regexForPOIIds = /^(?:cartocode:\w{2}|ref:[\w-]+:\w+|osm:[nwr]\d+|\d+)$/

export function useIdsResolver() {

}
4 changes: 3 additions & 1 deletion pages/category/[id].vue
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
<script setup lang="ts">
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return typeof params.id === 'string' && /^[-\w:,]+$/.test(params.id)
return !!params.id && regexForCategoryIds.test(params.id.toString())
},
})
</script>
57 changes: 33 additions & 24 deletions pages/embedded.vue
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,12 @@ import Embedded from '~/components/Home/Embedded.vue'
import type { ApiPoi } from '~/lib/apiPois'
import { siteStore as useSiteStore } from '~/stores/site'
import { mapStore as useMapStore } from '~/stores/map'
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Composables
//
const { params, query, path } = useRoute()
const route = useRoute()
const siteStore = useSiteStore()
const mapStore = useMapStore()
const { config, settings } = storeToRefs(siteStore)
Expand All @@ -20,7 +21,6 @@ const { $trackingInit } = useNuxtApp()
// Data
//
const boundaryGeojson = ref<Polygon | MultiPolygon>()
const categoryIdsJoin = ref<string>()
const poiId = ref<string>()
const categoryIds = ref<number[]>()
Expand All @@ -31,7 +31,7 @@ onBeforeMount(() => {
$trackingInit(config.value!)
})
const { boundary } = query
const { boundary } = route.query
if (boundary && typeof boundary === 'string' && settings.value!.polygons_extra) {
const boundaryObject = settings.value!.polygons_extra[boundary]
if (boundaryObject) {
Expand All @@ -51,33 +51,42 @@ if (boundary && typeof boundary === 'string' && settings.value!.polygons_extra)
}
}
// Workaround Nuxt missing feature to simple respect trialling slash meaning
if (params.poiId) {
categoryIdsJoin.value = params.p1 as string
poiId.value = params.poiId as string
}
else if (path.endsWith('/')) {
categoryIdsJoin.value = params.p1 as string
poiId.value = undefined
}
else {
categoryIdsJoin.value = undefined
poiId.value = params.p1 as string
// Get category IDs from URL
if (route.params.p1) {
const match = route.params.p1.toString().match(regexForCategoryIds)
if (!match || (!route.path.endsWith('/') && match.groups && (match.groups.cartocode || match.groups.reference || match.groups.osm)))
throw createError({ statusCode: 400, message: `No match for category ID: ${route.params.p1}` })
categoryIds.value = match.input?.split(',').map(id => Number.parseInt(id))
}
categoryIds.value = categoryIdsJoin.value?.split(',').map(id => Number.parseInt(id))
// Get POI ID from URL
if (categoryIds.value?.length === 1 && route.name === 'index-p1' && route.path.endsWith('/')) {
poiId.value = route.params.p1?.toString()
categoryIds.value = undefined
}
const { data, error } = await useFetch<ApiPoi>(`${API_ENDPOINT}/${API_PROJECT}/${API_THEME}/poi/${poiId.value}.geojson?geometry_as=bbox&short_description=false`)
if (route.params.poiId)
poiId.value = route.params.poiId.toString()
if (categoryIds.value && poiId.value) {
if (error.value)
throw createError(error.value)
// Fetch inital POI
const { data, error, status } = await useFetch<ApiPoi>(
() => `${API_ENDPOINT}/${API_PROJECT}/${API_THEME}/poi/${poiId.value}.geojson`,
{
query: {
geometry_as: 'bbox',
short_description: false,
},
immediate: !!poiId.value,
},
)
if (!data.value)
throw createError({ statusCode: 404, message: 'Initial POI not found !' })
if (error.value)
createError(error.value)
mapStore.setSelectedFeature(data.value!)
}
if (status.value === 'success' && data.value)
mapStore.setSelectedFeature(data.value)
</script>

<template>
Expand Down
14 changes: 0 additions & 14 deletions pages/embedded/[p1].vue

This file was deleted.

17 changes: 10 additions & 7 deletions pages/embedded/[p1]/[poiId].vue
Original file line number Diff line number Diff line change
@@ -1,15 +1,18 @@
<script setup lang="ts">
import { regexForCategoryIds, regexForPOIIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
typeof params.p1 === 'string'
&& typeof params.poiId === 'string'
&& /^[0-9,]+$/.test(params.p1)
&& /^[-\w:]+$/.test(params.poiId)
)
validate: ({ params }) => {
const match = params.p1.toString().match(regexForCategoryIds)
if (match?.groups && (match.groups.cartocode || match.groups.reference || match.groups.osm)) {
return false
}
return !!params.p1 && !!params.poiId && regexForCategoryIds.test(params.p1.toString()) && regexForPOIIds.test(params.poiId.toString())
},
})
</script>
Expand Down
9 changes: 4 additions & 5 deletions pages/embedded/[p1]/index.vue
Original file line number Diff line number Diff line change
@@ -1,13 +1,12 @@
<script setup lang="ts">
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
params.p1 === undefined
|| (typeof params.p1 === 'string' && /^[0-9,]+$/.test(params.p1))
)
validate: ({ params }) => {
return !!params.p1 && regexForCategoryIds.test(params.p1.toString())
},
})
</script>
Expand Down
66 changes: 42 additions & 24 deletions pages/index.vue
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,12 @@ import Home from '~/components/Home/Home.vue'
import type { ApiPoi } from '~/lib/apiPois'
import { siteStore as useSiteStore } from '~/stores/site'
import { mapStore as useMapStore } from '~/stores/map'
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Composables
//
const { params, query, path } = useRoute()
const route = useRoute()
const siteStore = useSiteStore()
const mapStore = useMapStore()
const { config, settings } = storeToRefs(siteStore)
Expand All @@ -20,7 +21,6 @@ const { $trackingInit } = useNuxtApp()
// Data
//
const boundaryGeojson = ref<Polygon | MultiPolygon>()
const categoryIdsJoin = ref<string>()
const poiId = ref<string>()
const categoryIds = ref<number[]>()
Expand All @@ -31,7 +31,7 @@ onBeforeMount(() => {
$trackingInit(config.value!)
})
const { boundary } = query
const { boundary } = route.query
if (boundary && typeof boundary === 'string' && settings.value!.polygons_extra) {
const boundaryObject = settings.value!.polygons_extra[boundary]
if (boundaryObject) {
Expand All @@ -51,38 +51,56 @@ if (boundary && typeof boundary === 'string' && settings.value!.polygons_extra)
}
}
// Workaround Nuxt missing feature to simple respect trialling slash meaning
if (params.poiId) {
categoryIdsJoin.value = params.p1 as string
poiId.value = params.poiId as string
}
else if (path.endsWith('/')) {
categoryIdsJoin.value = params.p1 as string
poiId.value = undefined
// Get category IDs from URL
if (route.params.p1) {
const match = route.params.p1.toString().match(regexForCategoryIds)
if (!match || (!route.path.endsWith('/') && match.groups && (match.groups.cartocode || match.groups.reference || match.groups.osm)))
throw createError({ statusCode: 400, message: `No match for category ID: ${route.params.p1}` })
categoryIds.value = match.input?.split(',').map(id => Number.parseInt(id))
}
else {
categoryIdsJoin.value = undefined
poiId.value = params.p1 as string
// Get POI ID from URL
if (categoryIds.value?.length === 1 && route.name === 'index-p1' && route.path.endsWith('/')) {
poiId.value = route.params.p1?.toString()
categoryIds.value = undefined
}
categoryIds.value = categoryIdsJoin.value?.split(',').map(id => Number.parseInt(id))
if (route.params.poiId)
poiId.value = route.params.poiId.toString()
// Fetch inital POI
const { data, error } = await useFetch<ApiPoi>(`${API_ENDPOINT}/${API_PROJECT}/${API_THEME}/poi/${poiId.value}.geojson?geometry_as=bbox&short_description=true`)
const { data, error, status } = await useFetch<ApiPoi>(
() => `${API_ENDPOINT}/${API_PROJECT}/${API_THEME}/poi/${poiId.value}.geojson`,
{
query: {
geometry_as: 'bbox',
short_description: true,
},
immediate: !!poiId.value,
},
)
if (categoryIds.value && poiId.value) {
if (error.value)
throw createError(error.value)
if (error.value)
createError(error.value)
if (!data.value)
throw createError({ statusCode: 404, message: 'Initial POI not found !' })
mapStore.setSelectedFeature(data.value!)
}
if (status.value === 'success' && data.value)
mapStore.setSelectedFeature(data.value)
</script>

<template>
<VApp>
<VAlert
v-if="error"
:closable="true"
:style="{ zIndex: 999 }"
:text="error.message"
location="top center"
position="fixed"
type="error"
variant="elevated"
/>
<Home
:boundary-area="boundaryGeojson"
:initial-category-ids="categoryIds"
Expand Down
18 changes: 0 additions & 18 deletions pages/index/[p1].vue

This file was deleted.

17 changes: 10 additions & 7 deletions pages/index/[p1]/[poiId].vue
Original file line number Diff line number Diff line change
@@ -1,15 +1,18 @@
<script setup lang="ts">
import { regexForCategoryIds, regexForPOIIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
typeof params.p1 === 'string'
&& typeof params.poiId === 'string'
&& /^[-\w:,]/.test(params.p1)
&& /^[-\w:]+$/.test(params.poiId)
)
validate: ({ params }) => {
const match = params.p1.toString().match(regexForCategoryIds)
if (match?.groups && (match.groups.cartocode || match.groups.reference || match.groups.osm)) {
return false
}
return !!params.p1 && !!params.poiId && regexForCategoryIds.test(params.p1.toString()) && regexForPOIIds.test(params.poiId.toString())
},
})
</script>
Expand Down
10 changes: 4 additions & 6 deletions pages/index/[p1]/index.vue
Original file line number Diff line number Diff line change
@@ -1,14 +1,12 @@
<script setup lang="ts">
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
params.p1 === undefined
|| (typeof params.p1 === 'string'
&& /^[-\w:,]+$/.test(params.p1))
)
validate: ({ params }) => {
return !!params.p1 && regexForCategoryIds.test(params.p1.toString())
},
})
</script>
Expand Down
5 changes: 2 additions & 3 deletions pages/poi/[id]/details.vue
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,14 @@ import type { ApiPoi } from '~/lib/apiPois'
import { headerFromSettings } from '~/lib/apiSettings'
import { getAsyncDataOrThrows } from '~/lib/getAsyncData'
import { siteStore as useSiteStore } from '~/stores/site'
import { regexForPOIIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
typeof params.id === 'string' && /^[-\w:]+$/.test(params.id)
)
return !!params.id && regexForPOIIds.test(params.id.toString())
},
})
Expand Down
6 changes: 2 additions & 4 deletions pages/pois/[ids]/map.vue
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,14 @@ import MapPois from '~/components/Map/MapPois.vue'
import { type ApiPoiId, type ApiPois, getPois } from '~/lib/apiPois'
import { getAsyncDataOrThrows } from '~/lib/getAsyncData'
import { siteStore as useSiteStore } from '~/stores/site'
import { regexForCategoryIds } from '~/composables/useIdsResolver'
//
// Validators
//
definePageMeta({
validate({ params }) {
return (
typeof params.ids === 'string'
&& /^[-\w:,]+$/.test(params.ids)
)
return !!params.ids && regexForCategoryIds.test(params.ids.toString())
},
})
Expand Down

0 comments on commit a375ab3

Please sign in to comment.