diff --git a/components/mdx/page-components.js b/components/mdx/page-components.js index 9c55c58..a65556a 100644 --- a/components/mdx/page-components.js +++ b/components/mdx/page-components.js @@ -3,6 +3,23 @@ import dynamic from 'next/dynamic' // NOTE: This is a dynamically generated file based on the config specified under the // `components` key in each post's frontmatter. const components = { + 'bigcoast-project-boundary': { + Map: dynamic(() => + import('../../posts/bigcoast-project-boundary/map.js').then( + (mod) => mod.Map || mod.default + ) + ), + Screenshot: dynamic(() => + import('../../posts/bigcoast-project-boundary/screenshot.js').then( + (mod) => mod.Screenshot || mod.default + ) + ), + BreakAll: dynamic(() => + import('../../posts/bigcoast-project-boundary/break-all.js').then( + (mod) => mod.BreakAll || mod.default + ) + ), + }, 'geochemical-cdr-measurements': {}, 'open-risk-data': {}, 'forest-offsets-firms': {}, diff --git a/posts/bigcoast-project-boundary.md b/posts/bigcoast-project-boundary.md new file mode 100644 index 0000000..d64670d --- /dev/null +++ b/posts/bigcoast-project-boundary.md @@ -0,0 +1,68 @@ +--- +version: 1.0.0 +title: To know if an offset project is burning, first you have to find it +authors: + - Grayson Badgley +date: 10-02-2023 +summary: Figuring out the location of an offset project shouldn't involve a scavenger hunt. +card: bigcoast-project-boundary +components: + - name: Map + src: ./map.js + - name: Screenshot + src: ./screenshot.js + - name: BreakAll + src: ./break-all.js +--- + +Wildfires across Canada have [burned millions of acres this year](https://en.wikipedia.org/wiki/2023_Canadian_wildfires). As the fire season started, I sat down to see if any Canadian forest offset projects were burning. Forests release carbon when they burn, which can undo the carbon storage promised by forest carbon offset projects. CarbonPlan has been [tracking US-based offset projects](https://carbonplan.org/research/forest-offsets-fires) threatened or destroyed by wildfires for several years now and I wanted to see if we could do the same for Canadian projects. + +But it was nearly impossible to figure out if any of Canada's offset projects were on fire. That’s because the rules governing the global carbon market don't always require a project to divulge its exact borders. Often only the province or country is documented. In other words, no one knows the precise locations of all the world’s active offset projects, a startling reality that prevents systematic oversight of the growing voluntary carbon market. + +One way to find out what’s on fire is to call up offset developers and ask. [That’s what reporters from Bloomberg did](https://www.bloomberg.com/news/articles/2023-06-26/canada-wildfire-season-burns-forest-set-aside-for-carbon-offsets) this spring to confirm that part of BigCoast, a forest offset project in British Columbia, was burning. In an ideal world, you shouldn't need to get someone on the phone to figure this out. And that made me curious – could I replicate what those reporters discovered using only publicly available data? + +It seemed straightforward at first. BigCoast's project documentation includes a few static, PDF maps of the project area. Unfortunately, those turned out to be such low-resolution that it was all but impossible to compare them to fire data. + +My next step was to visit [the official BigCoast project page at the Verra offset registry](https://registry.verra.org/app/projectDetail/VCS/3018), where there's a clearly labeled boundary file named `ProjectArea_BigCoast.kml`. But as soon as I loaded the data, it was clear I had a problem. Rather than delineating specific bits of forest protected by the project, the data indicated that BigCoast encompassed an area of over 50 million acres – nearly 20 percent of the total land area of British Columbia. The posted boundary even included large areas of the Pacific Ocean, a place that certainly doesn't have any forest protection going on. After digging deeper on the Internet, I convinced myself that this posted boundary data has nothing to do with the BigCoast project, but is instead the boundary of [the British Columbia Coastal Fire Centre](https://www2.gov.bc.ca/gov/content/safety/wildfire-status/about-bcws/fire-centres), a provincial administrative district used for managing wildfires. + +
+ + + The uploaded boundary of the BigCoast offset project, provided by the Verra + offset registry. The boundary's overlap with the Pacific ocean suggests the + data does not depict the actual areas of forest protected by the project. + +
+ +Back at the drawing board, I next found a document entitled `VCS_Joint_Prjct_Description_Monitoring_Report_BigCoast.pdf` that contained an appendix labeled “Project area polygon location.” I was hoping this appendix would be the jackpot. Instead, it turned out to be one the most baffling pieces of paperwork I've ever seen. + +Rather than post machine-readable boundary data – an ESRI shapefile, KML, or GeoJSON – the project developer elected to provide more than 70,000 latitude/longitude coordinate pairs in table form, spread across 688 pages of a PDF. I can’t imagine a less user-friendly format. But, somewhat perversely, the sheer absurdity of the whole thing became a challenge – I felt [compelled](https://xkcd.com/356/) to figure out if I could make something sensible of it. + +
+ + + Screenshot of the first page of the “Project area polygon location” + appendix. The appendix includes an additional 687 pages with data, which we + used to infer the boundary of the BigCoast offset project. + +
+ +Transforming the data into something useful required some experimentation. Coordinates alone aren't sufficient to recreate complex geometries. You also need to know how coordinates relate to each other. Typically, you’d lay out connected points in “rings,” starting with a single point and moving clockwise until you come back to where you started. This strategy allows you to draw even the most complicated shapes. But it gets trickier when the shape you're describing has more than one ring, because you also need to know where one ring ends and the next begins. Needless to say, these sorts of subtle details didn't come across in PDF format. + +Ultimately, I appealed to brute force. Rather than treating each point as a coordinate, I just drew a 100x100 meter square around each point, an approach known as buffering. From there, you _just_ need to take the intersection of the tens of thousands of squares to create a single geometry. + +If you squint and tilt your head, our reconstructed boundary looks sort of similar to the various, blurry static maps included in BigCoast's project documents. Our effort is at least more credible than the boundary uploaded to Verra. As further validation, our reconstruction has an area of 106,567 acres, which is remarkably close to 108,780 acres, the project's official acreage as listed in BigCoast's project documentation. And when I overlaid the inferred boundary against [ British Columbia's wildfire dataset](https://catalogue.data.gov.bc.ca/dataset/fire-perimeters-current), I calculated that about 276 acres – or 112 hectares – of the project burned in the [Cameron Bluffs Fire](https://wildfiresituation.nrs.gov.bc.ca/incidents?fireYear=2023&incidentNumber=V70600). That's pretty close to the “[a]bout 100 hectares” of burned forest the owner of the project told Bloomberg about. The time it took me to figure this out: upwards of 5 hours. + +
+ + + Our reconstructed boundary of the BigCoast offset project (light green) + overlaid against the much larger and inaccurate boundary data uploaded to + the Verra offset registry (green). Toggle the `+` in the upper left of the + figure to zoom in for more detail. + +
+ +It shouldn’t be this way and it doesn’t have to be. Some offset protocols, like the Climate Action Reserve’s [Mexico Forest Protocol](https://www.climateactionreserve.org/how/protocols/ncs/mexico-forest/) already require projects to share machine-readable boundary data. California’s forest offset program [does the same](https://webmaps.arb.ca.gov/ARBOCIssuanceMap/). It's exactly these data that have allowed us to track [the](https://carbonplan.org/research/offset-project-fire) [many](https://www.frontiersin.org/articles/10.3389/ffgc.2022.930426/full) [wildfires](https://carbonplan.org/blog/buffer-update-two) that have burned projects enrolled in California's offset program. Figuring out if an offset project is burning – or doing [any other analysis](https://www.nature.com/articles/s43247-023-00984-2) that requires exact borders – shouldn't involve phone calls or sifting through PDFs. All offset registries, offset rating agencies, and offset quality initiatives, like the Integrity Council for the Voluntary Carbon Market, should explicitly require that projects disclose analysis-ready boundary data. If these offset programs are operating as intended, there shouldn't be anything to hide. + +In the meantime, we've [open sourced](https://github.com/carbonplan/bigcoast-project-boundary) our version of the BigCoast project boundary and the code we used to generate it. We'd love to hear from other folks who have struggled working with offset project boundary data – drop us a line at [hello@carbonplan.org](mailto:hello@carbonplan.org). diff --git a/posts/bigcoast-project-boundary/break-all.js b/posts/bigcoast-project-boundary/break-all.js new file mode 100644 index 0000000..8f68eff --- /dev/null +++ b/posts/bigcoast-project-boundary/break-all.js @@ -0,0 +1,11 @@ +import { Box } from 'theme-ui' + +const BreakAll = ({ children }) => { + return ( + + {children} + + ) +} + +export default BreakAll diff --git a/posts/bigcoast-project-boundary/map.js b/posts/bigcoast-project-boundary/map.js new file mode 100644 index 0000000..0781016 --- /dev/null +++ b/posts/bigcoast-project-boundary/map.js @@ -0,0 +1,209 @@ +import { useEffect, useState } from 'react' +import { Box, useThemeUI } from 'theme-ui' +import { geoPath, geoAlbers } from 'd3-geo' +import { useSpring, animated } from '@react-spring/web' +import { feature as topoFeature } from 'topojson-client' +import Zoom from './zoom' + +const Point = ({ scale, position, sx, projection }) => { + const [x, y] = projection(position) + + const path = `M${x} ${y} A0 0 0 0 1 ${x + 0.0001 * scale} ${ + y + 0.0001 * scale + }` + + return ( + + ) +} + +const AnimatedPoint = animated(Point) + +const Label = ({ position, children, projection, scale, translate }) => { + const [x, y] = projection(position) + + const effectiveX = (x + translate[0]) * scale + const effectiveY = (y + translate[1]) * scale + + return ( + + {children} + + ) +} +const AnimatedLabel = animated(Label) + +const Map = ({ showInferred = false, showZoom = false }) => { + const [zoomed, setZoomed] = useState(false) + const [projection, setProjection] = useState(null) + const [paths, setPaths] = useState(null) + const { theme } = useThemeUI() + const { transform } = useSpring({ + config: { duration: 500, mass: 1, tension: 280, friction: 120 }, + transform: zoomed + ? `scale(3.7) translate(-160, -126)` + : 'scale(1) translate(0,0)', + }) + const { scale, translate } = useSpring({ + config: { duration: 500, mass: 1, tension: 280, friction: 120 }, + scale: zoomed ? 3.7 : 1, + translate: zoomed ? [-160, -126] : [0, 0], + }) + + useEffect(() => { + Promise.all( + [ + 'https://carbonplan-forest-offsets.s3.us-west-1.amazonaws.com/offsets-project-boundaries/inferred_boundary.json', + 'https://carbonplan-forest-offsets.s3.us-west-1.amazonaws.com/offsets-project-boundaries/provided_boundary.json', + 'https://cdn.jsdelivr.net/npm/world-atlas@2/land-50m.json', + 'https://cdn.jsdelivr.net/npm/world-atlas@2/countries-50m.json', + ].map((d) => fetch(d)) + ) + .then((res) => Promise.all(res.map((r) => r.json()))) + .then(([inferred, provided, basemap, countries]) => { + provided.features[0].geometry.coordinates[0] = + provided.features[0].geometry.coordinates[0].reverse() + + inferred.geometries[0].coordinates.forEach((c, i) => { + inferred.geometries[0].coordinates[i][0] = c[0].reverse() + }) + + const p = geoAlbers().fitExtent( + [ + [0, 0], + [400, 180], + ], + provided + ) + + const generator = geoPath(p) + setProjection(() => p) + + let countriesFiltered = topoFeature( + countries, + countries.objects['countries'] + ) + countriesFiltered.features = countriesFiltered.features.filter( + (d) => d.properties.name === 'United States of America' + ) + + setPaths({ + inferred: generator(inferred), + provided: generator(provided), + basemap: generator(topoFeature(basemap, basemap.objects['land'])), + countries: generator(countriesFiltered), + }) + }) + }, []) + + return ( + + + + + {paths && ( + <> + + + {showInferred && ( + + )} + + + )} + {projection && ( + <> + + + + )} + + + {projection && ( + <> + + Vancouver + + + Victoria + + + )} + {showZoom && } + + + ) +} + +export default Map diff --git a/posts/bigcoast-project-boundary/screenshot.js b/posts/bigcoast-project-boundary/screenshot.js new file mode 100644 index 0000000..11bcb4b --- /dev/null +++ b/posts/bigcoast-project-boundary/screenshot.js @@ -0,0 +1,115 @@ +import { alpha } from '@theme-ui/color' +import { useState } from 'react' +import { Box, Flex } from 'theme-ui' +import { FadeIn } from '@carbonplan/components' +import { Left, Right } from '@carbonplan/icons' + +const Screenshot = () => { + const [expanded, setExpanded] = useState(false) + + return ( + + + + setExpanded(true)} + sx={{ + position: 'absolute', + top: 0, + left: 0, + width: '100%', + height: '100%', + cursor: 'zoom-in', + transition: '0.2s background', + '@media (hover: hover) and (pointer: fine)': { + '&:hover': { + background: alpha('muted', 0.2), + }, + '&:hover div': { + opacity: 1, + }, + }, + }} + > + + + + + + + + {expanded && ( + setExpanded(false)} + sx={{ + position: 'fixed', + top: 0, + left: 0, + width: '100%', + height: '100%', + zIndex: 9999, + cursor: 'zoom-out', + alignItems: 'center', + justifyContent: 'center', + background: alpha('background', 0.9), + '& div': { + maxWidth: '100%', + maxHeight: '100%', + overflowY: 'scroll', + }, + }} + > + + setExpanded(false)} + src='https://images.carbonplan.org/blog/bigcoast-project-boundary/screenshot.png' + sx={{ + maxWidth: '100%', + maxHeight: '100%', + display: 'block', + userSelect: 'none', + }} + /> + + + )} + + + ) +} + +export default Screenshot diff --git a/posts/bigcoast-project-boundary/zoom.js b/posts/bigcoast-project-boundary/zoom.js new file mode 100644 index 0000000..48fbf1a --- /dev/null +++ b/posts/bigcoast-project-boundary/zoom.js @@ -0,0 +1,61 @@ +import { Box } from 'theme-ui' + +const Zoom = ({ setZoomed }) => { + return ( + + setZoomed(false)} + sx={{ + color: 'primary', + border: 'none', + bg: 'transparent', + m: [0], + p: [0], + cursor: 'pointer', + textTransform: 'uppercase', + fontFamily: 'body', + letterSpacing: 'smallcaps', + mr: [3], + fontSize: [5, 5, 5, 6], + transition: 'color 0.15s', + '@media (hover: hover) and (pointer: fine)': { + '&:hover': { + color: 'secondary', + }, + }, + }} + > + - + + setZoomed(true)} + sx={{ + color: 'primary', + border: 'none', + bg: 'transparent', + m: [0], + p: [0], + cursor: 'pointer', + textTransform: 'uppercase', + fontFamily: 'body', + letterSpacing: 'smallcaps', + fontSize: [5, 5, 5, 6], + transition: 'color 0.15s', + '@media (hover: hover) and (pointer: fine)': { + '&:hover': { + color: 'secondary', + }, + }, + }} + > + + + + + ) +} + +export default Zoom