Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore: add basic project constraints #5152

Merged
merged 3 commits into from
Mar 25, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -146,6 +146,7 @@
"@web/test-runner-playwright": "^0.11.0",
"@web/test-runner-visual-regression": "^0.9.0",
"@webcomponents/webcomponentsjs": "^2.8.0",
"@yarnpkg/types": "^4.0.1",
"alex": "^11.0.1",
"cem-plugin-module-file-extensions": "^0.0.5",
"chromatic": "^11.20.0",
Expand Down
193 changes: 193 additions & 0 deletions yarn.config.cjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
/* eslint-disable @typescript-eslint/no-var-requires */
/*!
* Copyright 2025 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/

/** @type {import('@yarnpkg/types')} */
const { defineConfig } = require('@yarnpkg/types');
const fg = require('fast-glob');

/**
* The workspace object used in the constraints function
* @typedef {import('@yarnpkg/types').Yarn.Constraints.Workspace} Workspace
*/

module.exports = defineConfig({
async constraints({ Yarn }) {
/**
* Fetch a list of all the component workspaces using a glob pattern
* @type {string[]} components
*/
const components = fg.sync('packages/*', {
Copy link
Contributor

@Rajdeepc Rajdeepc Mar 6, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fg.sync() is called synchronously here, which might block execution since there are a lot of packages.
can we use an async function and await fg()

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's no difference here between using await or sync because the list of components must be present to validate against in the functions below. This is a preference rather than an important requirement to update during code review. There is the same amount of "blocking execution" as you say, between awaiting an async function as there is using the synchronous version of a function. You only gain benefits from asynchronous if you are running tasks against it as the information becomes available, which we are not.

cwd: __dirname,
onlyDirectories: true,
});

/**
* This function checks the workspace for any local package references
* and ensure that the devDependencies are up-to-date with the latest version
* currently in the project
* @param {Workspace} workspace
* @returns {void}
*/
function validateLocalPackages(workspace) {
// Return early if the workspace does not have any peerDependencies
if (!workspace.manifest.peerDependencies) {
return;
}

// Start by filtering out the local packages from the external dependencies
const localPackages = Object.keys(
workspace.manifest.peerDependencies
)
.filter((pkg) => Yarn.workspace({ ident: pkg }))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yarn.workspace({ ident: pkg }) is called twice on a single package here. can we reuse the result and call it once?

const localPackages = Object.keys(workspace.manifest.peerDependencies)
    .map((pkg) => Yarn.workspace({ ident: pkg }))
    .filter(Boolean);

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See my note on using the standard constraints syntax and tooling. This is not a node script. https://yarnpkg.com/features/constraints

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yarn.workspace({ ident: pkg }) returns a filtered result based on the object keys specific to the workspace passed into the function. You can't call it "once" as you say and get the same result.

.map((pkg) => Yarn.workspace({ ident: pkg }));

// Iterate over the local packages and ensure that the devDependencies are up-to-date
for (const localWorkspace of localPackages) {
const localVersion = localWorkspace.manifest.version;
workspace.set(
`devDependencies.${localWorkspace.manifest.name}`,
localVersion ?? 'workspace:^'
);
}
}

/**
* A reusable function to add keywords to ensure workspaces
* include a minimal set of keywords for discoverability
* with additionalKeywords as an optional parameter to add more
* specific keywords that are relevant to the workspace
* @param {string[]} additionalKeywords
* @returns {string[]}
*/
function keywords(additionalKeywords = []) {
return [
'design-system',
'spectrum',
'adobe',
'adobe-spectrum',
'web components',
'web-components',
'lit-element',
'lit-html',
Comment on lines +74 to +81
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we add the values to a new array like baseKeyWords and use this? Saves re-allocation of new array everytime keywords is called!

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Minimally beneficial and I'm not sure the abstraction is worth it. If the script was slow to run, I'd say the optimization makes sense but I'm not sure it's worth more abstraction in an already foreign syntax to most contributors.

...additionalKeywords,
];
}

/**
* This function rolls up all the component package.json
* requirements for all workspaces into a single function
* to simplify into a readable set of operations
* @param {Workspace} workspace
* @param {string} folderName
* @returns {void}
*/
function validateComponentPackageJson(workspace, folderName) {
// Only update the homepage if it does not already exist
if (!workspace.manifest.homepage) {
workspace.set(
'homepage',
`https://opensource.adobe.com/spectrum-web-components/components/${folderName}`
);
}

workspace.set('publishConfig.access', 'public');
workspace.set('type', 'module');
workspace.set('keywords', keywords(['component', 'css']));

// A subset of components have a different entry point than the default
if (
['clear-button', 'close-button', 'modal', 'opacity-checkerboard'].includes(folderName)
) {
workspace.set('main', `./src/${folderName}.css.js`);
workspace.set('module', `./src/${folderName}.css.js`);
} else {
workspace.set('main', './src/index.js');
workspace.set('module', './src/index.js');
}
}

/**
* This function rolls up all the package.json requirements
* for all workspaces into a single function to simplify
* the workspace for loop into a readable set of operations
* @param {Workspace} workspace
* @returns {void}
*/
function validatePackageJson(workspace) {
const isRoot = workspace.cwd === '.';
const isComponent = components.includes(workspace.cwd);

/**
* -------------- GLOBAL --------------
* Global configuration for all workspaces
*/
if (
!workspace.manifest.license ||
workspace.manifest.license === ''
) {
workspace.set('license', 'Apache-2.0');
}

workspace.set('author', 'Adobe');

workspace.set('repository.type', 'git');
workspace.set(
'repository.url',
'https://github.com/adobe/spectrum-web-components.git'
);

// We don't need to set the directory for the root workspace
if (!isRoot) {
workspace.set('repository.directory', workspace.cwd);
}

workspace.set(
'bugs.url',
'https://github.com/adobe/spectrum-web-components/issues'
);

/**
* -------------- COMPONENTS --------------
* Process the components workspaces with component-specific configuration
*/
if (isComponent) {
const folderName = workspace.cwd?.split('/')?.[1];
Copy link
Contributor

@Rajdeepc Rajdeepc Mar 6, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A more robust approach like this path.basename(workspace.cwd) would be helpful?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's important to distinguish the contraints syntax from a node script. This is not a node script. This is a yarn constraints file that uses node syntax. Please do read up on constraints in the documentation: https://yarnpkg.com/features/constraints

validateComponentPackageJson(workspace, folderName);
validateLocalPackages(workspace);
} else {
/**
* -------------- OTHER --------------
* All other workspaces should have at least the following configuration
*/
if (!workspace.manifest.keywords) {
workspace.set('keywords', keywords());
}

if (!workspace.manifest.homepage) {
workspace.set(
'homepage',
'https://opensource.adobe.com/spectrum-web-components/'
);
}
}
}

/**
* This loop iterates over all the workspaces in the project
* and updates the package.json file with the necessary
*/
for (const workspace of Yarn.workspaces()) {
validatePackageJson(workspace);
}
},
});
10 changes: 10 additions & 0 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -244,6 +244,7 @@ __metadata:
"@web/test-runner-playwright": "npm:^0.11.0"
"@web/test-runner-visual-regression": "npm:^0.9.0"
"@webcomponents/webcomponentsjs": "npm:^2.8.0"
"@yarnpkg/types": "npm:^4.0.1"
alex: "npm:^11.0.1"
cem-plugin-module-file-extensions: "npm:^0.0.5"
chromatic: "npm:^11.20.0"
Expand Down Expand Up @@ -11987,6 +11988,15 @@ __metadata:
languageName: node
linkType: hard

"@yarnpkg/types@npm:^4.0.1":
version: 4.0.1
resolution: "@yarnpkg/types@npm:4.0.1"
dependencies:
tslib: "npm:^2.4.0"
checksum: 10c0/90226789475680ba599833571dd76c0718dd5b4c5022481263ef309d6a628f6246671cd6ca86e49c966ddefa7aca6ccef82240dc1476d2cea702ea5bee2a6b72
languageName: node
linkType: hard

"JSONStream@npm:^1.3.5":
version: 1.3.5
resolution: "JSONStream@npm:1.3.5"
Expand Down
Loading