526 lines
24 KiB
HTML
526 lines
24 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
<title>IP Map</title>
|
|
<script src="https://unpkg.com/maplibre-gl@4.1.2/dist/maplibre-gl.js"></script>
|
|
<link rel="stylesheet" href="https://unpkg.com/maplibre-gl@4.1.2/dist/maplibre-gl.css" />
|
|
<link rel="shortcut icon" href="" />
|
|
<style>
|
|
body {
|
|
margin: 0;
|
|
padding: 0;
|
|
background-color: #111;
|
|
}
|
|
html, body {
|
|
width: 100%;
|
|
height: 100%;
|
|
}
|
|
main {
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
font-family: sans-serif;
|
|
}
|
|
#map {
|
|
width: min(90vw, 90vh);
|
|
height: min(90vw, 90vh);
|
|
}
|
|
.maplibregl-canvas {
|
|
cursor: pointer;
|
|
}
|
|
.maplibregl-popup {
|
|
max-width: unset !important;
|
|
}
|
|
.maplibregl-popup-content {
|
|
background-color: white;
|
|
color: black;
|
|
font-size: 0.9rem;
|
|
padding: 0.3rem 2.2rem 0.3rem 0.6rem;
|
|
border-radius: 4px;
|
|
}
|
|
.maplibregl-popup-close-button {
|
|
height: 100%;
|
|
width: 2rem;
|
|
margin: 0;
|
|
padding: 0;
|
|
padding-bottom: 3px;
|
|
font-size: 1.3rem;
|
|
color: black;
|
|
}
|
|
.maplibregl-ctrl-top-right {
|
|
display: flex;
|
|
}
|
|
.maplibregl-ctrl-top-right .custom-control {
|
|
align-self: flex-start;
|
|
}
|
|
.maplibregl-ctrl button.maplibregl-ctrl-subnets .maplibregl-ctrl-icon {
|
|
background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' stroke='%23333' stroke-width='1.5' fill='none' viewBox='0 0 29 29'%3E%3Cpath d='m 6 6 v 17 h 17 v -17 h -17 z m 0 8.5 h 17 m -8.5 -8.5 v 17'/%3E%3C/svg%3E");
|
|
}
|
|
.custom-control {
|
|
user-select: none;
|
|
}
|
|
details.custom-control > summary {
|
|
margin: 0;
|
|
padding: 0 0.5rem;
|
|
display: flex;
|
|
align-items: center;
|
|
height: 29px;
|
|
cursor: pointer;
|
|
}
|
|
details.custom-control[open] > summary {
|
|
border-bottom: 2px solid black;
|
|
}
|
|
details.custom-control > summary::marker {
|
|
content: "";
|
|
}
|
|
details.custom-control > summary {
|
|
list-style: none;
|
|
}
|
|
details.custom-control > summary::-webkit-details-marker {
|
|
display:none;
|
|
}
|
|
details.custom-control > summary h3 {
|
|
margin: 0;
|
|
padding: 0;
|
|
display: inline;
|
|
font-size: 0.9rem;
|
|
}
|
|
details.custom-control > .button-container {
|
|
max-height: 12rem;
|
|
overflow-y: scroll;
|
|
}
|
|
details.custom-control > .button-container button {
|
|
width: 100%;
|
|
padding: 0 0.5rem;
|
|
}
|
|
.maplibregl-ctrl button {
|
|
color: black;
|
|
font-size: 0.8rem;
|
|
}
|
|
.maplibregl-ctrl button.active {
|
|
background-color: rgb(0 0 0/15%);
|
|
}
|
|
.maplibregl-ctrl button:not(:disabled):hover {
|
|
background-color: rgb(0 0 0/10%);
|
|
}
|
|
.maplibregl-ctrl button:not(:disabled):active {
|
|
background-color: rgb(0 0 0/15%);
|
|
}
|
|
.maplibregl-ctrl p {
|
|
margin: 0.25rem 0.5rem;
|
|
font-size: 0.8rem;
|
|
}
|
|
.maplibregl-ctrl p:empty {
|
|
display: none;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<main>
|
|
<div id="map"></div>
|
|
</main>
|
|
<script>
|
|
const coordsToHilbert = ({ x, y }) => {
|
|
let rotation = 0
|
|
let reflection = 0
|
|
let index = 0
|
|
for (let b = 15; b >= 0; b--) {
|
|
let bits = reflection
|
|
reflection = (y >>> b) & 1
|
|
reflection |= ((x >>> b) & 1) << 1
|
|
bits = bits ^ reflection
|
|
bits = ((bits >>> rotation) | (bits << (2 - rotation))) & 3
|
|
index |= bits << (b << 1)
|
|
reflection ^= (1 << rotation)
|
|
bits = bits & (-bits) & 1
|
|
while (bits) {
|
|
++rotation
|
|
bits >>>= 1
|
|
}
|
|
if (++rotation >= 2)
|
|
rotation -= 2
|
|
}
|
|
index ^= 0x2aaaaaaa
|
|
for (let d = 1; d < 32; d *= 2) {
|
|
let t = index >>> d
|
|
if (!t) break
|
|
index ^= t
|
|
}
|
|
return index
|
|
}
|
|
|
|
const hilbertToCoords = index => {
|
|
let rotation = 0
|
|
let reflection = 0
|
|
let coord = { x: 0, y: 0 }
|
|
index ^= (index >>> 1) ^ 0x2aaaaaaa
|
|
for (let b = 15; b >= 0; b--) {
|
|
var bits = index >>> (2 * b) & 3
|
|
reflection ^= ((bits >>> (2 - rotation)) | (bits << rotation)) & 3
|
|
coord.y |= (reflection & 1) << b
|
|
coord.x |= ((reflection >>> 1) & 1) << b
|
|
reflection ^= (1 << rotation)
|
|
bits &= (-bits) & 1
|
|
while (bits) {
|
|
bits >>>= 1
|
|
++rotation
|
|
}
|
|
if (++rotation >= 2)
|
|
rotation -= 2
|
|
}
|
|
return coord
|
|
}
|
|
|
|
const ipToString = (ip, subnet = 32) => {
|
|
const x = (ip >> (32 - subnet)) << (32 - subnet)
|
|
return `${x >> 24 & 0xFF}.${x >> 16 & 0xFF}.${x >> 8 & 0xFF}.${x >> 0 & 0xFF}`
|
|
}
|
|
|
|
const rangeToCoords = range => {
|
|
const [a, b, c, d, subnet] = range.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)\/(\d+)$/).slice(1).map(Number)
|
|
const ip = a << 24 | b << 16 | c << 8 | d
|
|
if (subnet % 2 === 0) {
|
|
const subnetSize = 2 ** (32 - subnet)
|
|
const subnetSideSize = Math.sqrt(subnetSize)
|
|
const subnetsPerSide = 0x10000 / subnetSideSize
|
|
const { x: hx, y: hy } = hilbertToCoords(ip + subnetSize / 2)
|
|
const tx = Math.floor(hx / subnetSideSize)
|
|
const ty = Math.floor(hy / subnetSideSize)
|
|
const mxA = tx / subnetsPerSide
|
|
const mxB = (tx + 1) / subnetsPerSide
|
|
const myA = ty / subnetsPerSide
|
|
const myB = (ty + 1) / subnetsPerSide
|
|
return [
|
|
new maplibregl.MercatorCoordinate(mxA, myA, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxB, myA, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxB, myB, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxA, myB, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxA, myA, 0).toLngLat().toArray(),
|
|
]
|
|
} else {
|
|
const nearestSubnet = subnet + 1
|
|
const subnetSize = 2 ** (32 - nearestSubnet)
|
|
const subnetSideSize = Math.sqrt(subnetSize)
|
|
const subnetsPerSide = 0x10000 / subnetSideSize
|
|
const { x: hxA, y: hyA } = hilbertToCoords(ip + subnetSize / 2)
|
|
const { x: hxB, y: hyB } = hilbertToCoords(ip + subnetSize + subnetSize / 2)
|
|
const txA = Math.floor(hxA / subnetSideSize)
|
|
const tyA = Math.floor(hyA / subnetSideSize)
|
|
const txB = Math.floor(hxB / subnetSideSize)
|
|
const tyB = Math.floor(hyB / subnetSideSize)
|
|
const mxA = Math.min(txA, txB) / subnetsPerSide
|
|
const mxB = (Math.max(txA, txB) + 1) / subnetsPerSide
|
|
const myA = Math.min(tyA, tyB) / subnetsPerSide
|
|
const myB = (Math.max(tyA, tyB) + 1) / subnetsPerSide
|
|
return [
|
|
new maplibregl.MercatorCoordinate(mxA, myA, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxB, myA, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxB, myB, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxA, myB, 0).toLngLat().toArray(),
|
|
new maplibregl.MercatorCoordinate(mxA, myA, 0).toLngLat().toArray(),
|
|
]
|
|
}
|
|
}
|
|
|
|
const defaultVariant = "density"
|
|
const defaultColormap = "jet"
|
|
const tileSize = 256
|
|
const tilesDir = "assets/tiles"
|
|
const tilesId = "ipmap-tiles"
|
|
const privateRangesId = "private-ranges"
|
|
const subnetsId = "subnets"
|
|
|
|
const canvas = document.createElement("canvas")
|
|
const canvasScale = window.devicePixelRatio
|
|
canvas.width = tileSize * canvasScale
|
|
canvas.height = tileSize * canvasScale
|
|
const ctx = canvas.getContext("2d")
|
|
ctx.scale(canvasScale, canvasScale)
|
|
ctx.fillStyle = "white"
|
|
ctx.lineWidth = 3
|
|
ctx.font = "bold 20px sans-serif"
|
|
maplibregl.addProtocol(subnetsId, async (params, abortController) => {
|
|
const [z, y, x] = params.url.split("://")[1].split("/")
|
|
const width = 2 ** z
|
|
const subnet = 2 * z
|
|
const hy = Math.floor(0x10000 * (y / width))
|
|
const hx = Math.floor(0x10000 * (x / width))
|
|
const text = `${ipToString(coordsToHilbert({ x: hx, y: hy }), subnet)}/${subnet}`
|
|
const metrics = ctx.measureText(text)
|
|
const cy = tileSize / 2 + metrics.actualBoundingBoxAscent / 2
|
|
const cx = tileSize / 2 - metrics.actualBoundingBoxRight / 2
|
|
ctx.clearRect(0, 0, tileSize, tileSize)
|
|
ctx.moveTo(0, 0)
|
|
ctx.lineTo(tileSize, 0)
|
|
ctx.lineTo(tileSize, tileSize)
|
|
ctx.lineTo(0, tileSize)
|
|
ctx.lineTo(0, 0)
|
|
ctx.strokeStyle = "white"
|
|
ctx.stroke()
|
|
ctx.strokeStyle = "black"
|
|
ctx.strokeText(text, cx, cy)
|
|
ctx.fillText(text, cx, cy)
|
|
return { data: await new Promise((res, rej) => canvas.toBlob(b => b.arrayBuffer().then(res))) }
|
|
})
|
|
|
|
const map = new maplibregl.Map({
|
|
container: "map",
|
|
attributionControl: false,
|
|
renderWorldCopies: false,
|
|
doubleClickZoom: false,
|
|
dragRotate: false,
|
|
pitchWithRotate: false,
|
|
touchPitch: false,
|
|
style: {
|
|
version: 8,
|
|
sources: {},
|
|
layers: []
|
|
},
|
|
center: [0, 0],
|
|
minZoom: -2,
|
|
maxZoom: 11.499,
|
|
zoom: -2
|
|
})
|
|
map.painter.context.extTextureFilterAnisotropic = undefined
|
|
map.touchZoomRotate.disableRotation()
|
|
|
|
const dataP = fetch(`${tilesDir}/tiles.json`, { cache: "no-store" }).then(res => res.json())
|
|
const privateRangePatternP = map.loadImage("assets/private-range-pattern.png")
|
|
|
|
map.once("style.load", async () => {
|
|
const data = await dataP
|
|
|
|
let curDate, curVariant, curColormap
|
|
const dates = Object.keys(data).sort()
|
|
curDate = dates.find(date => {
|
|
const variantEntries = Object.entries(data[date])
|
|
if (variantEntries.length === 0) return false
|
|
const variantEntry = variantEntries.find(([variant, colormaps]) => variant === defaultVariant && colormaps.length > 0) ??
|
|
variantEntries.sort(([a], [b]) => a.localeCompare(b)).find(([_, colormaps]) => colormaps.length > 0)
|
|
if (!variantEntry) return false
|
|
const [variant, colormaps] = variantEntry
|
|
curVariant = variant
|
|
curColormap = colormaps.find(colormap => colormap === defaultColormap) ?? colormaps.sort()[0]
|
|
return true
|
|
})
|
|
if (!curDate) {
|
|
console.error("no data found")
|
|
return
|
|
}
|
|
|
|
map.addSource(tilesId, {
|
|
type: "raster",
|
|
tiles: [`${tilesDir}/${curDate}/${curVariant}/${curColormap}/{z}/{y}/{x}.png`],
|
|
tileSize,
|
|
minzoom: 0,
|
|
maxzoom: 8,
|
|
})
|
|
map.addLayer({
|
|
id: tilesId,
|
|
type: "raster",
|
|
source: tilesId,
|
|
paint: {
|
|
"raster-resampling": "nearest"
|
|
}
|
|
})
|
|
|
|
const privateRanges = [
|
|
{ range: "0.0.0.0/8", description: "Current Network Definition" },
|
|
{ range: "10.0.0.0/8", description: "Private Networks" },
|
|
{ range: "100.64.0.0/10", description: "CGNAT Transition" },
|
|
{ range: "127.0.0.0/8", description: "Loopback" },
|
|
{ range: "169.254.0.0/16", description: "Link-Local" },
|
|
{ range: "172.16.0.0/12", description: "Private Networks" },
|
|
{ range: "192.0.0.0/24", description: "DS-Lite Transition" },
|
|
{ range: "192.0.2.0/24", description: "Documentation" },
|
|
{ range: "192.88.99.0/24", description: "IPv4 to IPv6 Transition" },
|
|
{ range: "192.168.0.0/16", description: "Private Networks" },
|
|
{ range: "198.18.0.0/15", description: "Testing" },
|
|
{ range: "198.51.100.0/24", description: "Documentation" },
|
|
{ range: "203.0.113.0/24", description: "Documentation" },
|
|
{ range: "224.0.0.0/4", description: "Multicast (Class D)" },
|
|
{ range: "240.0.0.0/4", description: "Class E" },
|
|
]
|
|
|
|
map.addSource(privateRangesId, {
|
|
type: "geojson",
|
|
data: {
|
|
type: "FeatureCollection",
|
|
features: privateRanges.map(({ range, description }) => ({
|
|
type: "Feature",
|
|
geometry: {
|
|
type: "Polygon",
|
|
coordinates: [rangeToCoords(range)]
|
|
},
|
|
properties: {
|
|
range,
|
|
description
|
|
}
|
|
}))
|
|
}
|
|
})
|
|
|
|
const privateRangePatternId = "private-range-pattern"
|
|
map.addImage(privateRangePatternId, (await privateRangePatternP).data)
|
|
map.addLayer({
|
|
id: privateRangesId,
|
|
type: "fill",
|
|
source: privateRangesId,
|
|
paint: {
|
|
"fill-pattern": privateRangePatternId
|
|
}
|
|
})
|
|
|
|
let subnetsVisible = true
|
|
map.addSource(subnetsId, {
|
|
type: "raster",
|
|
tiles: [`${subnetsId}://{z}/{y}/{x}`],
|
|
tileSize,
|
|
minzoom: 0,
|
|
maxzoom: 12,
|
|
})
|
|
map.addLayer({
|
|
id: subnetsId,
|
|
type: "raster",
|
|
source: subnetsId,
|
|
})
|
|
|
|
const setStyle = (date, variant, colormap) => {
|
|
if (date === curDate && variant === curVariant && colormap === curColormap || !data[date]?.[variant]?.includes(colormap))
|
|
return false
|
|
const source = map.getSource(tilesId)
|
|
if (!source) return false
|
|
source.setTiles([`${tilesDir}/${date}/${variant}/${colormap}/{z}/{y}/{x}.png`])
|
|
curDate = date
|
|
curVariant = variant
|
|
curColormap = colormap
|
|
return true
|
|
}
|
|
|
|
const addControl = (elem, side) => map.addControl({
|
|
onAdd: () => elem,
|
|
onRemove: () => elem.parentNode.removeChild(elem)
|
|
}, side)
|
|
|
|
const detailsControl = (name) => {
|
|
const container = document.createElement("details")
|
|
container.className = "maplibregl-ctrl maplibregl-ctrl-group custom-control"
|
|
const header = document.createElement("h3")
|
|
header.textContent = name
|
|
const summary = document.createElement("summary")
|
|
summary.replaceChildren(header)
|
|
const buttonContainer = document.createElement("div")
|
|
buttonContainer.className = "button-container"
|
|
container.replaceChildren(summary, buttonContainer)
|
|
return {
|
|
container,
|
|
setButtons: buttons => buttonContainer.replaceChildren(...buttons.map(({ button }) => button)),
|
|
addControl: (side = "top-right") => addControl(container, side)
|
|
}
|
|
}
|
|
|
|
const dateControl = detailsControl("Date")
|
|
const variantControl = detailsControl("Variant")
|
|
const colormapControl = detailsControl("Colormap")
|
|
|
|
const dateButtons = dates.map(date => {
|
|
const button = document.createElement("button")
|
|
button.textContent = date
|
|
if (curDate === date) button.className = "active"
|
|
return { date, button }
|
|
})
|
|
dateButtons.forEach(({ date, button }) => button.addEventListener("click", () => {
|
|
if (setStyle(date, curVariant, curColormap)) {
|
|
dateButtons.forEach(({ button }) => button.className = "")
|
|
button.className = "active"
|
|
renderControls()
|
|
}
|
|
}))
|
|
dateControl.setButtons(dateButtons)
|
|
|
|
const renderControls = () => {
|
|
const variants = Object.keys(data[curDate]).sort()
|
|
const variantButtons = variants.map(variant => {
|
|
const button = document.createElement("button")
|
|
button.textContent = variant
|
|
if (curVariant === variant) button.className = "active"
|
|
return { variant, button }
|
|
})
|
|
variantButtons.forEach(({ variant, button }) => button.addEventListener("click", () => {
|
|
if (setStyle(curDate, variant, curColormap)) renderControls()
|
|
}))
|
|
variantControl.setButtons(variantButtons)
|
|
|
|
const colormaps = data[curDate][curVariant].sort()
|
|
const colormapButtons = colormaps.map(colormap => {
|
|
const button = document.createElement("button")
|
|
button.textContent = colormap
|
|
if (curColormap === colormap) button.className = "active"
|
|
return { colormap, button }
|
|
})
|
|
colormapButtons.forEach(({ colormap, button }) => button.addEventListener("click", () => {
|
|
if (setStyle(curDate, curVariant, colormap)) {
|
|
colormapButtons.forEach(({ button }) => button.className = "")
|
|
button.className = "active"
|
|
}
|
|
}))
|
|
colormapControl.setButtons(colormapButtons)
|
|
}
|
|
|
|
renderControls()
|
|
|
|
const subnetsControl = document.createElement("div")
|
|
subnetsControl.className = "maplibregl-ctrl maplibregl-ctrl-group custom-control"
|
|
const subnetsButton = document.createElement("button")
|
|
subnetsButton.className = "maplibregl-ctrl-subnets active"
|
|
const subnetsIcon = document.createElement("span")
|
|
subnetsIcon.className = "maplibregl-ctrl-icon"
|
|
subnetsButton.replaceChildren(subnetsIcon)
|
|
subnetsButton.addEventListener("click", () => {
|
|
subnetsVisible = !subnetsVisible
|
|
subnetsButton.classList.toggle("active")
|
|
map.setLayoutProperty(subnetsId, "visibility", subnetsVisible ? "visible" : "none")
|
|
})
|
|
subnetsControl.replaceChildren(subnetsButton)
|
|
addControl(subnetsControl, "top-right")
|
|
|
|
dateControl.addControl()
|
|
variantControl.addControl()
|
|
colormapControl.addControl()
|
|
|
|
const getIPAtMouse = (e, features, link) => {
|
|
const { x, y } = maplibregl.MercatorCoordinate.fromLngLat(e.lngLat, 0)
|
|
const ip = coordsToHilbert({ x: Math.floor(0x10000 * x), y: Math.floor(0x10000 * y) })
|
|
const subnet = Math.min(32, Math.round(map.getZoom()) * 2 + 18)
|
|
const ipStr = ipToString(ip, subnet)
|
|
const props = features && map.queryRenderedFeatures(e.point).find(f => f.layer.id === privateRangesId)?.properties
|
|
const propsText = props ? `<br>Part of private range ${props.range}<br>Used for ${props.description}` : ""
|
|
const name = subnet < 32 ? "Range" : "IP"
|
|
const ipText = subnet < 32 ? `${ipStr}/${subnet}` : ipStr
|
|
const ipLink = link ? `<a href="https://bgp.tools/prefix/${ipStr}" target="_blank">${ipText}</a>` : ipText
|
|
return `${name}: ${ipLink}${propsText}`
|
|
}
|
|
|
|
const hoverTextControl = document.createElement("div")
|
|
hoverTextControl.className = "maplibregl-ctrl maplibregl-ctrl-group"
|
|
const hoverTextP = document.createElement("p")
|
|
hoverTextControl.replaceChildren(hoverTextP)
|
|
map.on("mousemove", e => hoverTextP.textContent = getIPAtMouse(e, false, false))
|
|
addControl(hoverTextControl, "bottom-left")
|
|
|
|
map.addControl(new maplibregl.NavigationControl({ showCompass: false }), "top-left")
|
|
|
|
let curPopup
|
|
map.on("click", e => curPopup = new maplibregl.Popup().setHTML(getIPAtMouse(e, true, true)).setLngLat(e.lngLat).addTo(map))
|
|
|
|
const main = document.getElementsByTagName("main")[0]
|
|
main.addEventListener("click", e => {
|
|
if (e.target === main && curPopup) curPopup.remove()
|
|
})
|
|
})
|
|
</script>
|
|
</body>
|
|
</html>
|