Skip to content
This repository has been archived by the owner on Nov 9, 2023. It is now read-only.

Initial prototype implementation #2

Open
wants to merge 31 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
e9a5c52
Setup Lerna
justinfagnani Apr 24, 2021
80a4e01
Add wc-registry package
justinfagnani May 1, 2021
2360eff
Add Firestore emulator
justinfagnani May 1, 2021
c6ff722
Add graphql
justinfagnani May 2, 2021
899049d
Implement basic packageInfo query + npm fetch
justinfagnani May 2, 2021
9cc3a9d
Initial draft implementation
justinfagnani Jul 4, 2021
30bab17
Format and license
justinfagnani Jul 7, 2021
f89052e
Specify config for Codegen Script
bennypowers Jul 7, 2021
5a983bf
Add Quick Start to README
bennypowers Jul 7, 2021
80fa85e
Merge pull request #1 from bennypowers/docs/build
justinfagnani Jul 7, 2021
e241922
Start on content site
justinfagnani Apr 7, 2022
b09799c
Update graphql deps
justinfagnani Apr 7, 2022
d4ae6a6
Update firebase deps
justinfagnani Apr 7, 2022
46ce491
Update more deps
justinfagnani Apr 7, 2022
df25ba4
Add wc-org-shared to hold the schema
justinfagnani Apr 8, 2022
b1bd018
Fix resolvers: add a PackageInfo resolver
justinfagnani Apr 8, 2022
f4decea
Add package type to schema
justinfagnani Apr 8, 2022
4a2a1fd
WIP
justinfagnani Apr 12, 2022
e2be3bb
Add dist-tag query support
justinfagnani Jul 19, 2022
1c68e8e
Update firebase-tools
justinfagnani Jul 19, 2022
76c1bc9
Use wireit for script running
justinfagnani Jul 20, 2022
ca12eda
Run npm update
justinfagnani Jul 20, 2022
72cd2f5
Update typescript
justinfagnani Jul 20, 2022
95f36bb
Update more deps
justinfagnani Jul 20, 2022
eddd5e5
Add wc-org-server package
justinfagnani Jul 20, 2022
9af1a86
Add stub of element page
justinfagnani Jul 20, 2022
c967887
Start on /catalog/element page
justinfagnani Jul 21, 2022
632afbf
Render more element data on the element page
justinfagnani Jul 23, 2022
a499b6a
Rename packages
justinfagnani Jul 23, 2022
9a51888
Clarify referenceString args
justinfagnani Aug 31, 2022
00aef2d
Add initial validator
justinfagnani Sep 2, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
node_modules/
.wireit/

*.tgz
*.tsbuildinfo
.DS_Store
4 changes: 4 additions & 0 deletions .prettierrc.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{
"singleQuote": true,
"bracketSpacing": false
}
173 changes: 173 additions & 0 deletions DESIGN.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,173 @@
# webcomponents.org v2 Design

## Stack

* Server
* Google Cloud Run
* Node 17
* Firestore
* GraphQL via graphql-helix and graphql-tools
* Client
* Web components with Lit
* Apollo GraphQL Client
* Playground Elements
* Static site generation
* eleventy
* Schema
* GraphQL SDL
* graphql-codegen to generate TypeScript interfaces

## Catalog

### Indexing Packages

The catalog will initially be oriented around open-source packages published to npm. This is where the vast majority of elements are now published.

Cataloging closed-source elements is not a goal, as that would require indexing proprietary information about where to retrieve elements, their license and cost, etc.

Package sources other than npm, like GitHub packages or package-less distributions available via HTTP, may be included in the future. npm sources will be namespaced to prevserve the ability to add new sources.

On npm, we will require that packages have a `"customElements"` field in their `package.json`, and that it contains a valid custom elements manifest. This is so that we don't have to download package tarballs in order to detect if a package has a custom elements manifest, and so that we don't have to analyze source code to find custom element definitions.

#### Collections

Design systems and other groups of components benefit from first-class support in the catalog. Users may want to search for collections of components rather than individual components.

Collections don't neccesaarily correspond to packages. Many design systems publish each component as a separate npm package.

We would like a way for element authors to define their own collections controll exactly what elements are included. For this it would be ideal to have a centralized definition of a collection. Even though a collection is not published to npm, a collection definition could be published to a well-known URL, such as a file stored in GitHub. A custom element manifest can then include references to collection definitions so that while indexing element we can discover and index collections.

### Schema

The catalog server has two schemas in play: A GraphQL schema and an implcit Firestore schema (implicit, since Firestore is schemaless). We will try to keep these as close as possible. Some places where schemas differ:
* IDs: Firestore documents have immutable IDs that must be unique within their subcollection. This makes immutable package data like package name, or the version string for a specific version, well-suited to be document IDs. They will name to GraphQL fields like name or version.
* Collections: Firestore has both array-valued fields and subcollections. In GraphQL these are both described by a array-valued field, so Firestore subcollections will be mapped to GraphQL array fields.
* Dates vs Timestamps: Firestore has a Timestamp data type that will be converted to JS dates for GraphQL
* Maps: Firestore supports map-valued fields. GraphQL only supports list values, so we must convert maps to lists of key/value pairs.
* Denormalized fields in Firestore documents may or may not be present in the GraphQL schema.

Firestore stores data in documents and collections. Collections and documents have immutable IDs, and can be referenced by a path of alternating collection IDs and document IDs. This forms a hierarchy somewhat like a filesystem.

GraphQL schemas describe a graph of types, though query results are always a tree. GraphQL fields with a list type can represent Firestore collections. Since Firestore is schemaless we'll use the GraphQL schema types to describe Firestore documents.

#### Package, PackageVersion, and CustomElement

One root Firestore collection is `packages`, containing `Package` documents. A `Package` represents a single package name and all published versions of the package, under the subcollection `versions`. `PackageVersion` describes a single publisend version of a package. `PackageVersion` then contains a `customElements` collection of `CustomElement` documents.

We end up with a hierarchy like:

`Package` -> `versions` -> `PackageVersion` -> `customElements` -> `CustomElement`

Only one of the `PackageVersion` documents represents the latest version of a package. Queries will generally be performed against the lastest versions. Other versions will be available for documentation.

#### Collections

Since ollections don't neccesaarily correspond to packages, we will have a separate root Firestore collection to represent element collections.

A simple representation of a collection would be a list of elements that belong to it. For this we need a way to reference an element. Package and custom element name works if element names are unique per package, which they should be. See [Identifying Elements](#identifying-elements).

So one optiion for a collection schema is:

`Collection`:
* name
* description
* homepage
* elements
* package
* element

### Queries

#### Querying elements

Firestore does not allow queries across collections, but it does suppport "collection group" queries. So we can do a query on the `customElements` collection group, regardless of what document is the direct parent of the subcollcetion, like:

```ts
db.collectionGroup('customElements').get()
```

The problem with this query is that it will return elements from every version of a package, not just the latest. The latest version of a package is denoted with the dist-tag `"latest"` and that's part of the `Package` metadata. In order to query elements from the latest published version, we need to denormalize the dist tag to `PackageVersion` and `CustomElement`. Then we can perform a query like:

```ts
// TODO: change this to support a distTags array
db.collectionGroup('customElements').where('distTag', '==', 'latest').get();
```

Denormalizataion requrires that every time `distTags` is updated for a package that we also update every `PackageVersion` and `CustomElement` affected by that change. For instance, we may have to remove the `"latest"` dist tag from the previous version and add it to the new latest version.

This denormalization requirement extends to any field in `Package` or `PackageVersion` that we would like to use in a query on `CustomElement`. This in effect turns the `customElements` collection into a search index.

Possible denormalized query fields on `CustomElement`:
* dist tags
* package name
* author
* keywords
* publish date
* libraries used
* npm downloads
* collections

We can also denormalize fields that we want to read along with `CustomElement`, though this is less neccessary as once we have query results we can read a `CustomElement`'s containing `PackageVersion` easily with one extra read. This is better than reading a `PackageVersion` for potential matches only to throw them out when they fail a criteria.

#### Get all elements

Useful for catalog browsing and showing a set of components from a default query
on a page or inside an embedded catalog widget. This query may be ordered by some default ranking (popularity, quality, freshness, etc)

#### Get elements by query

A filtered version of the above query, queryable on any of the fields in `CustomElement`

#### Get collections

#### Get authors

### Identifying Elements

In several places, like URLs and cross-references from collections, we will need a way to refer to an element.

Packages are not actually organized by element, they are organized by modules. Yet it's common and easy for users to refer to an element from a package, regarless of which module the element is exported from.

To be convenient, we can form element identifiers from a package-name/element-name combination. If for some reason an element name is used more than once in a package (other than re-exporting it) we can add a disambiguation token, or allow the element to be referred from by module.

For instance, we might have a URL to a catalog page for `@shoelace-style/shoelace/sl-button` like:

`webcomponents.org/catalog/element/@shoelace-style/shoelace/sl-button/`

If there are two `sl-button`s in the `@shoelace-style/shoelace` package, we can refer to the second (determined by some heuristic) like:

`webcomponents.org/catalog/element/@shoelace-style/shoelace/sl-button/2`.

This case should be rare since there will usually only be one "custom element export" (the `customElements.define()` or `@customElement()` call) for an element in a package.

We can also refer to elements via their full package/module path:

`webcomponents.org/catalog/package/@shoelace-style/shoelace/dist/components/button/button.js#sl-button`.

Package versions will be encoded into the package name:

`webcomponents.org/catalog/package/@shoelace-style/[email protected]/...`.

### Requirements for Inclusion in the catalog

We would like to ensure that elements in the catalog are actual custom elements, and easy to include in projects that use a variety of build systems. We can do analysis of projects as we index them to see if they meet certain requirements to facilitiate this.

* `"customElements"` `package.json` field pointing to a valid custom element manifest
* `"type": "module"` in `package.json`
* Element module paths in custom element manifest exist and include the declared export.

Additional, optional requirements based on static analysis:
* No non-browser-supported imports in custom element definition modules (so that elements don't require specific bundlers to use, etc).
* No unguarded use of non-Browser APIs like `process.env`, etc.
* All non-dev dependencies meet the same requirements.

Quality signals:
* Does not bundle common libraries (like `lit`)

## Abuse and moderation

The catalog has little to no direct user-generated content - there are no plans at the moment for features like comments. But the catalog does display user-generated content from the npm packages and custom element definitions themselves. This is an avenue for abuse. We should not rely on npm to remove abusive content. We need a way to remove packages from the index and accept abuse reports. Assuming that abuse via npm is very infrequent, we can likely handle reports via email.

## Privacy and PII

We would like to store as little personal information as possible and mostly have a read-only site. If we ever add personalization features, like starring elements, creating lists, user-submitted element rankings, etc., we'll need to allow accounts and log-in, likely via a third-party login provider.
28 changes: 28 additions & 0 deletions LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
BSD 3-Clause License

Copyright (c) 2021 Google LLC. All rights reserved.

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.

2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.

3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 changes: 24 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,28 @@

A new implementation of the webcompoents.org site

This repo will contain several packages:
This repo contains several packages:

* A data-only backend that indexes npm packages and provides a GraphQL API into the database
* A frontend server that serves the user-facing webcompoents.org site
* An HTML client served by the frontend server
- `@webcomponents/registry`: A data-only backend that indexes npm packages and provides a GraphQL API into the database
- `@webcomponents/registry-api`: GraphQL schemas and TypeScript interfaces for the registry API.
- `@webcomponents/site-server`: A frontend server that serves the user-facing webcompoents.org site
- `@webcomponents/site-content`: An HTML client served by the frontend server
- `@webcomponents/custom-element-manifest-tools`: Tools for working with Custom Element Manifests

## Quick Start

1. Install dependencies:
```bash
npm ci
```
2. Start the registry server:
```bash
cd packages/registry
npm run emulators:start & npm run dev
```
This starts a GraphQL service. The web server only serves a simple welcome page and an interactive GraphiQL editor at `/graphql`.
3. Start the site server:
```bash
cd packages/site-server
npm run dev
```
Loading