Skip to content

Commit

Permalink
mosaic, marimekko
Browse files Browse the repository at this point in the history
  • Loading branch information
Fil committed Feb 25, 2024
1 parent 0b3defa commit 1d701e8
Show file tree
Hide file tree
Showing 9 changed files with 3,279 additions and 0 deletions.
2,981 changes: 2,981 additions & 0 deletions docs/data/airports.csv

Large diffs are not rendered by default.

Binary file added docs/data/flights-10m.parquet
Binary file not shown.
Binary file added docs/data/gaia-sample.parquet
Binary file not shown.
31 changes: 31 additions & 0 deletions docs/data/gdp-receipts.csv
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
year,country,receipts
1970,Sweden,46.9
1970,Netherlands,44
1970,Norway,43.5
1970,Britain,40.7
1970,France,39
1970,Germany,37.5
1970,Belgium,35.2
1970,Canada,35.2
1970,Finland,34.9
1970,Italy,30.4
1970,United States,30.3
1970,Greece,26.8
1970,Switzerland,26.5
1970,Spain,22.5
1970,Japan,20.7
1979,Sweden,57.4
1979,Netherlands,55.8
1979,Norway,52.2
1979,Britain,39
1979,France,43.4
1979,Germany,42.9
1979,Belgium,43.2
1979,Canada,35.8
1979,Finland,38.2
1979,Italy,35.7
1979,United States,32.5
1979,Greece,30.6
1979,Switzerland,33.2
1979,Spain,27.1
1979,Japan,26.6
17 changes: 17 additions & 0 deletions docs/data/sales.csv
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
market,segment,value
"Auburn, AL",Almond lovers,3840
"Auburn, AL",Berry buyers,1920
"Auburn, AL",Carrots-n-more,960
"Auburn, AL",Delicious-n-new,400
"Birmingham, AL",Almond lovers,1600
"Birmingham, AL",Berry buyers,1440
"Birmingham, AL",Carrots-n-more,960
"Birmingham, AL",Delicious-n-new,400
"Gainesville, FL",Almond lovers,640
"Gainesville, FL",Berry buyers,960
"Gainesville, FL",Carrots-n-more,640
"Gainesville, FL",Delicious-n-new,400
"Durham, NC",Almond lovers,320
"Durham, NC",Berry buyers,480
"Durham, NC",Carrots-n-more,640
"Durham, NC",Delicious-n-new,400
57 changes: 57 additions & 0 deletions docs/party/arrow2parquet.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
# Convert Arrow files to parquet

A simple app that converts your arrow files to the parquet format, using DuckDB-wasm under the hood.

```js
const file = view(Inputs.file());
```

```js
const table = file?.name.replace(".arrow", "");
```
```js
const db = file && DuckDBClient.of({[table]: file});
```
```js
display(file
? html`<button onclick=${async function() {
this.disabled = true;
download(await toParquet(db, {table}));
this.disabled = false;
}}>Download ${table}.parquet`
: html`<button disabled>…`
);
```
```js
// Exports a DuckDB table to parquet.
async function toParquet(duckDbClient, {table = "data", name = `${table}.parquet`} = {}) {
const tmp = (Math.random()*1e16).toString(16);
const db = duckDbClient._db;
// https://duckdb.org/docs/sql/statements/copy
console.log("start COPY", {table, name, tmp});
await duckDbClient.query(`COPY ${duckDbClient.escape(table)} TO '${tmp}' (FORMAT PARQUET, COMPRESSION GZIP)`);
console.log("start copyFileToBuffer");
const buffer = await db.copyFileToBuffer(tmp);
//db.dropFile(tmp);

return new File([buffer], name, {
// https://issues.apache.org/jira/browse/PARQUET-1889
type: "application/vnd.apache.parquet"
});
}

// Triggers a download. Needs to be invoked via a user input.
function download(file) {
const a = document.createElement("a");
a.download = file.name;
a.href = URL.createObjectURL(file);
a.click();
URL.revokeObjectURL(a.href);
}
```
<div style="height: 40vh"></div>
54 changes: 54 additions & 0 deletions docs/party/duckdb-spatial.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
# DuckDB spatial

```js echo
display(
Plot.plot({
projection: "identity",
width: 975,
height: 610,
marks: [
Plot.geo(counties, {
stroke: "var(--theme-background-alt)",
strokeWidth: 0.25,
fill: (d) => d.properties.area
})
]
})
);
```

<p class=warning>This page only works if we upgrade DuckDBClient to import duckdb-wasm@1.28.1-dev106.0 or later, coupled with apache-arrow@14 or later (like I did here manually). See issues <a href=https://github.com/duckdb/duckdb-wasm/issues/1561>duckdb-wasm#1561</a>; <a href=https://github.com/observablehq/framework/issues/750>framework#750</a> and <a href=https://github.com/observablehq/framework/issues/733>framework#733</a>.</p>

---

Start by creating an empty DuckDB database with its [spatial extension](https://duckdb.org/docs/extensions/spatial.html), and load the TopoJSON [US Atlas](https://github.com/topojson/us-atlas):

```js echo
const db = await DuckDBClient.of();
await db.sql`INSTALL spatial`;
await db.sql`LOAD spatial`;
await db.sql`CREATE TABLE us AS (
SELECT * FROM ST_Read('https://cdn.jsdelivr.net/npm/us-atlas/us/10m.json')
)`;
```

Now we can work on this file as a database:

```js echo
const features = await db.sql`SELECT * FROM us`;
```

${Inputs.table(features)}

The **geom** field is a ST_GEOMETRY, the internal format that DuckDB spatial uses to represent geometries. We can convert it to GeoJSON:

```js echo
const counties = Array.from(
await db.sql`SELECT ST_AsGeoJSON(geom) AS county, ST_Area(geom) as area FROM us`,
({county, area}) => Object.assign(JSON.parse(county), {properties: {area}})
);
```

This is just the beginning.

(Note: this exploration is all happening client-side.)
81 changes: 81 additions & 0 deletions docs/party/duckdb.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
# DuckDB

DuckDB is “an in-process SQL OLAP Database Management System. [DuckDB-Wasm](https://github.com/duckdb/duckdb-wasm) brings DuckDB to every browser thanks to WebAssembly.” DuckDB-Wasm is available by default as `duckdb` in Markdown, but you can explicitly import it as:

```js echo
import * as duckdb from "npm:@duckdb/duckdb-wasm";
```

For convenience, we provide a [`DatabaseClient`](https://observablehq.com/@observablehq/database-client-specification) implementation on top of DuckDB-Wasm, `DuckDBClient`. This is also available by default in Markdown, but you can explicitly import it like so:

```js echo
import {DuckDBClient} from "npm:@observablehq/duckdb";
```

To get a DuckDB client, pass zero or more named tables to `DuckDBClient.of`. Each table can be expressed as a [`FileAttachment`](../javascript/files), [Arquero table](./arquero), [Arrow table](./arrow), an array of objects, or a promise to the same. For example, below we load a sample of 250,000 stars from the [Gaia Star Catalog](https://observablehq.com/@cmudig/peeking-into-the-gaia-star-catalog) as a [Apache Parquet](https://parquet.apache.org/) file:

```js echo
const db = DuckDBClient.of({gaia: FileAttachment("../data/gaia-sample.parquet")});
```

Now we can run a query using `db.sql` to bin the stars by [right ascension](https://en.wikipedia.org/wiki/Right_ascension) (`ra`) and [declination](https://en.wikipedia.org/wiki/Declination) (`dec`):

```js echo
const bins = db.sql`SELECT
floor(ra / 2) * 2 + 1 AS ra,
floor(dec / 2) * 2 + 1 AS dec,
count() AS count
FROM
gaia
GROUP BY
1,
2`;
```

These bins can quickly be turned into a heatmap with [Plot’s raster mark](https://observablehq.com/plot/marks/raster), showing the milky way.

```js echo
display(
Plot.plot({
aspectRatio: 1,
x: {domain: [0, 360]},
y: {domain: [-90, 90]},
marks: [
Plot.frame({fill: 0}),
Plot.raster(bins, {
x: "ra",
y: "dec",
fill: "count",
width: 360 / 2,
height: 180 / 2,
imageRendering: "pixelated"
})
]
})
);
```

For externally-hosted data, you can create an empty `DuckDBClient` and load a table from a SQL query, say using [`read_parquet`](https://duckdb.org/docs/guides/import/parquet_import) or [`read_csv`](https://duckdb.org/docs/guides/import/csv_import).

```js run=false
const db = await DuckDBClient.of();

await db.sql`CREATE TABLE addresses
AS SELECT *
FROM read_parquet('https://static.data.gouv.fr/resources/bureaux-de-vote-et-adresses-de-leurs-electeurs/20230626-135723/table-adresses-reu.parquet')
LIMIT 100`;
```

As an alternative to `db.sql`, there’s also `db.query`:

```js echo
display(await db.query("SELECT * FROM gaia LIMIT 10"));
```

And `db.queryRow`:

```js echo
display(await db.queryRow("SELECT count() AS count FROM gaia"));
```

See the [DatabaseClient Specification](https://observablehq.com/@observablehq/database-client-specification) for more details.
58 changes: 58 additions & 0 deletions docs/party/mosaic-flights.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
---
index: true
---

# Mosaic Cross-Filter Flights 10M

An example using [Mosaic vgplot](https://uwdata.github.io/mosaic/vgplot/) to interactively cross-filter 10 million flight records.
You may need to wait a few seconds for the dataset to load.

<div style="display: flex; flex-wrap: wrap">
${makePlot("delay")}
${makePlot("time")}
${makePlot("distance")}
</div>

_Try selecting delayed flights. How much more likely are they to leave later in the day?_

```js
// load flights data from external parquet file
const datasource = new URL(await FileAttachment("../data/flights-10m.parquet").url(), document.location).href;
await vg.coordinator().exec(`CREATE TABLE IF NOT EXISTS flights10m AS
SELECT
GREATEST(-60, LEAST(ARR_DELAY, 180))::DOUBLE AS delay,
DISTANCE AS distance,
DEP_TIME AS time
FROM '${datasource}'`);

// create a selection with crossfilter resolution
const brush = vg.Selection.crossfilter();

// helper method to generate a binned plot filtered by brush
// a plot contains a rectY mark for a histogram, as well as
// an intervalX interactor to populate the brush selection
const makePlot = (column) =>
vg.plot(
vg.rectY(
vg.from("flights10m", {filterBy: brush}), // data set and filter selection
{x: vg.bin(column), y: vg.count(), fill: "steelblue", inset: 0.5}
),
vg.intervalX({as: brush}), // create an interval selection brush
vg.xDomain(vg.Fixed), // don't change the x-axis domain across updates
vg.marginLeft(75),
vg.width(350),
vg.height(220)
);

// generate dashboard with three linked histograms
// display(vg.vconcat(makePlot("delay"), makePlot("time"), makePlot("distance")));
```

```js
// import vgplot and configure Mosaic to use DuckDB-WASM
const vg = await import("npm:@uwdata/vgplot");
{
const wasm = await vg.wasmConnector();
await vg.coordinator().databaseConnector(wasm);
}
```

0 comments on commit 1d701e8

Please sign in to comment.