Skip to content

Commit

Permalink
Merge 2b378e0 into 5ffb681
Browse files Browse the repository at this point in the history
  • Loading branch information
davidfirst authored Aug 2, 2019
2 parents 5ffb681 + 2b378e0 commit c7a9b48
Show file tree
Hide file tree
Showing 11 changed files with 178 additions and 37 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/).

## [unreleased]

- [#1865](/~https://github.com/teambit/bit/issues/1865) allow override `package.json` props via `overrides`
- [#1837](/~https://github.com/teambit/bit/issues/1837) enable executing commands on remote components outside of bit-workspace
- [#1774](/~https://github.com/teambit/bit/issues/1774) improve access errors and warn when sudo is used

Expand Down
73 changes: 70 additions & 3 deletions e2e/functionalities/workspace-config.e2e.3.js
Original file line number Diff line number Diff line change
Expand Up @@ -1297,6 +1297,73 @@ describe('workspace config', function () {
expect(showBar.compiler).to.be.null;
});
});
describe('override package.json values', () => {
before(() => {
helper.setNewLocalAndRemoteScopes();
helper.createComponentBarFoo();
helper.addComponentBarFoo();
const overrides = {
'bar/*': {
bin: 'my-bin-file.js'
}
};
helper.addOverridesToBitJson(overrides);
});
it('bit show should show the overrides', () => {
const show = helper.showComponentParsed('bar/foo');
expect(show.overrides)
.to.have.property('bin')
.equal('my-bin-file.js');
});
describe('tag, export and import the component', () => {
before(() => {
helper.tagAllComponents();
helper.exportAllComponents();
helper.reInitLocalScope();
helper.addRemoteScope();
helper.importComponent('bar/foo');
});
it('should write the values into package.json file', () => {
const packageJson = helper.readPackageJson(path.join(helper.localScopePath, 'components/bar/foo'));
expect(packageJson)
.to.have.property('bin')
.that.equals('my-bin-file.js');
});
it('should not show the component as modified', () => {
const status = helper.status();
expect(status).to.have.string(statusWorkspaceIsCleanMsg);
});
describe('changing the value in the package.json directly (not inside overrides)', () => {
before(() => {
const compDir = path.join(helper.localScopePath, 'components/bar/foo');
const packageJson = helper.readPackageJson(compDir);
packageJson.bin = 'my-new-file.js';
helper.writePackageJson(packageJson, compDir);
});
it('should not show the component as modified', () => {
const status = helper.status();
expect(status).to.not.have.string('modified components');
});
});
describe('changing the value in the package.json inside overrides', () => {
before(() => {
const compDir = path.join(helper.localScopePath, 'components/bar/foo');
const packageJson = helper.readPackageJson(compDir);
packageJson.bit.overrides.bin = 'my-new-file.js';
helper.writePackageJson(packageJson, compDir);
});
it('should show the component as modified', () => {
const status = helper.status();
expect(status).to.have.string('modified components');
});
it('bit diff should show the field diff', () => {
const diff = helper.diff('bar/foo');
expect(diff).to.have.string('my-bin-file.js');
expect(diff).to.have.string('my-new-file.js');
});
});
});
});
});
describe('basic validations', () => {
before(() => {
Expand Down Expand Up @@ -1324,18 +1391,18 @@ describe('workspace config', function () {
expect(output).to.have.string('expected overrides.bar to be object, got number');
});
});
describe('when an unrecognized field is added into overrides of a component', () => {
describe('when a forbidden field is added into overrides of a component', () => {
before(() => {
const overrides = {
bar: {
some_field: {}
name: 'foo'
}
};
helper.addOverridesToBitJson(overrides);
});
it('any bit command should throw an error', () => {
const output = helper.runWithTryCatch('bit list');
expect(output).to.have.string('found an unrecognized field "some_field" inside "overrides.bar" property');
expect(output).to.have.string('found a forbidden field "name" inside "overrides.bar" property');
});
});
describe('when a dependency field is not an object', () => {
Expand Down
17 changes: 11 additions & 6 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -109,6 +109,7 @@
"ora": "^1.1.0",
"p-event": "^4.1.0",
"p-map-series": "^1.0.0",
"package-json-validator": "^0.6.3",
"pad-right": "^0.2.2",
"parse-gitignore": "^1.0.1",
"porter-stemmer": "^0.9.1",
Expand Down
12 changes: 12 additions & 0 deletions src/consumer/component-ops/component-writer.js
Original file line number Diff line number Diff line change
Expand Up @@ -177,6 +177,7 @@ export default class ComponentWriter {
componentConfig.tester = this.component.tester ? this.component.tester.toBitJsonObject('.') : {};
packageJson.addOrUpdateProperty('bit', componentConfig.toPlainObject());
this._mergeChangedPackageJsonProps(packageJson);
this._mergePackageJsonPropsFromOverrides(packageJson);
await this._populateEnvFilesIfNeeded();
this.component.dataToPersist.addFile(packageJson.toVinylFile());
if (distPackageJson) this.component.dataToPersist.addFile(distPackageJson.toVinylFile());
Expand Down Expand Up @@ -252,6 +253,17 @@ export default class ComponentWriter {
}
}

/**
* these changes were entered manually by a user via `overrides` key
*/
_mergePackageJsonPropsFromOverrides(packageJson: PackageJsonFile) {
const valuesToMerge = this.component.overrides.componentOverridesPackageJsonData;
packageJson.mergePackageJsonObject(valuesToMerge);
}

/**
* these are changes done by a compiler
*/
_mergeChangedPackageJsonProps(packageJson: PackageJsonFile) {
if (!this.component.packageJsonChangedProps) return;
const valuesToMerge = this._replaceDistPathTemplateWithCalculatedDistPath(packageJson);
Expand Down
1 change: 1 addition & 0 deletions src/consumer/component-ops/components-object-diff.js
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ export function componentToPrintableForDiff(component: Component): Object {
obj.overridesDependencies = parsePackages(overrides.dependencies);
obj.overridesDevDependencies = parsePackages(overrides.devDependencies);
obj.overridesPeerDependencies = parsePackages(overrides.peerDependencies);
obj.overridesPackageJsonProps = JSON.stringify(component.overrides.componentOverridesPackageJsonData);
return obj;
}

Expand Down
4 changes: 3 additions & 1 deletion src/consumer/config/component-config.js
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,9 @@ export default class ComponentConfig extends AbstractConfig {
*/
static mergeWithWorkspaceConfig(componentConfig: Object, consumerConfig: ?WorkspaceConfig): ComponentConfig {
const plainConsumerConfig = consumerConfig ? consumerConfig.toPlainObject() : {};
return ComponentConfig.fromPlainObject(R.merge(plainConsumerConfig, componentConfig));
const consumerConfigWithoutConsumerSpecifics = filterObject(plainConsumerConfig, (val, key) => key !== 'overrides');
const mergedObject = R.merge(consumerConfigWithoutConsumerSpecifics, componentConfig);
return ComponentConfig.fromPlainObject(mergedObject);
}

/**
Expand Down
15 changes: 7 additions & 8 deletions src/consumer/config/component-overrides.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import {
OVERRIDE_COMPONENT_PREFIX
} from '../../constants';
import type { ConsumerOverridesOfComponent } from './consumer-overrides';
import { dependenciesFields } from './consumer-overrides';
import { dependenciesFields, overridesSystemFields, nonPackageJsonFields } from './consumer-overrides';

export type ComponentOverridesData = {
dependencies?: Object,
Expand Down Expand Up @@ -59,14 +59,13 @@ export default class ComponentOverrides {
// $FlowFixMe
return new ComponentOverrides(R.clone(overridesFromModel), {});
}

static componentOverridesDataFields() {
return dependenciesFields;
}
get componentOverridesData() {
const fields = ComponentOverrides.componentOverridesDataFields();
const isDependencyField = (val, field) => fields.includes(field);
return R.pickBy(isDependencyField, this.overrides);
const isNotSystemField = (val, field) => !overridesSystemFields.includes(field);
return R.pickBy(isNotSystemField, this.overrides);
}
get componentOverridesPackageJsonData() {
const isPackageJsonField = (val, field) => !nonPackageJsonFields.includes(field);
return R.pickBy(isPackageJsonField, this.overrides);
}
getComponentDependenciesWithVersion(): Object {
const allDeps = Object.assign(
Expand Down
20 changes: 11 additions & 9 deletions src/consumer/config/consumer-overrides.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,9 @@ export type ConsumerOverridesOfComponent = {
export type ConsumerOverridesConfig = { [string]: ConsumerOverridesOfComponent };

export const dependenciesFields = ['dependencies', 'devDependencies', 'peerDependencies'];
const consumerOverridesPermittedFields = [...dependenciesFields, 'env'];
export const overridesForbiddenFields = ['name', 'main', 'version'];
export const overridesSystemFields = ['propagate', 'env'];
export const nonPackageJsonFields = [...dependenciesFields, ...overridesSystemFields];

export default class ConsumerOverrides {
overrides: ConsumerOverridesConfig;
Expand Down Expand Up @@ -47,10 +49,9 @@ export default class ConsumerOverrides {
if (!current.propagate) {
stopPropagation = true;
}
consumerOverridesPermittedFields.forEach((field) => {
if (!current[field]) return;
if (!acc[field]) acc[field] = {};
Object.keys(current).forEach((field) => {
if (field === 'env') {
if (!acc[field]) acc[field] = {};
['compiler', 'tester'].forEach((envField) => {
// $FlowFixMe we made sure before that current.env is set
if (acc.env[envField] || !current.env[envField]) return;
Expand All @@ -59,8 +60,9 @@ export default class ConsumerOverrides {
} else if (dependenciesFields.includes(field)) {
// $FlowFixMe
acc[field] = Object.assign(current[field], acc[field]);
} else {
throw new Error(`consumer-overrides, ${field} does not have a merge strategy`);
} else if (!overridesSystemFields.includes(field)) {
// $FlowFixMe propagate is a system field
acc[field] = current[field];
}
});
return acc;
Expand Down Expand Up @@ -154,9 +156,9 @@ export default class ConsumerOverrides {
function validateComponentOverride(id, override) {
validateUserInputType(message, override, `overrides.${id}`, 'object');
Object.keys(override).forEach((field) => {
if (!consumerOverridesPermittedFields.includes(field)) {
throw new GeneralError(`${message} found an unrecognized field "${field}" inside "overrides.${id}" property.
only the following fields are allowed: ${consumerOverridesPermittedFields.join(', ')}.`);
if (overridesForbiddenFields.includes(field)) {
throw new GeneralError(`${message} found a forbidden field "${field}" inside "overrides.${id}" property.
the following fields are not allowed: ${overridesForbiddenFields.join(', ')}.`);
}
if (dependenciesFields.includes(field)) {
validateDependencyField(field, override, id);
Expand Down
22 changes: 22 additions & 0 deletions src/scope/models/version.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -307,5 +307,27 @@ describe('Version', () => {
version.packageJsonChangedProps = [1, 2, 3, 4];
expect(validateFunc).to.throw('expected packageJsonChangedProps to be object, got array');
});
it('should throw when packageJsonChangedProps has a non-compliant npm value', () => {
version.packageJsonChangedProps = { bin: 1234 };
expect(validateFunc).to.throw('the generated package.json field "bin" is not compliant with npm requirements');
});
it('should not throw when packageJsonChangedProps has a compliant npm value', () => {
version.packageJsonChangedProps = { bin: 'my-file.js' };
expect(validateFunc).to.not.throw();
});
it('should throw when overrides has a package.json field that is non-compliant npm value', () => {
version.overrides = { bin: 1234 };
expect(validateFunc).to.throw(
'"overrides.bin" is a package.json field but is not compliant with npm requirements'
);
});
it('should not throw when overrides has a package.json field that is compliant npm value', () => {
version.overrides = { bin: 'my-file.js' };
expect(validateFunc).to.not.throw();
});
it('should show the original error from package-json-validator when overrides has a package.json field that is non-compliant npm value', () => {
version.overrides = { scripts: false };
expect(validateFunc).to.throw('Type for field scripts, was expected to be object, not boolean');
});
});
});
49 changes: 39 additions & 10 deletions src/scope/version-validator.js
Original file line number Diff line number Diff line change
@@ -1,14 +1,19 @@
// @flow
import R from 'ramda';
import { PJV } from 'package-json-validator';
import packageNameValidate from 'validate-npm-package-name';
import validateType from '../utils/validate-type';
import { BitId, BitIds } from '../bit-id';
import VersionInvalid from './exceptions/version-invalid';
import { isValidPath } from '../utils';
import type Version from './models/version';
import { Dependencies } from '../consumer/component/dependencies';
import ComponentOverrides from '../consumer/config/component-overrides';
import PackageJsonFile from '../consumer/component/package-json-file';
import {
overridesForbiddenFields,
dependenciesFields,
nonPackageJsonFields
} from '../consumer/config/consumer-overrides';

/**
* make sure a Version instance is correct. throw an exceptions if it is not.
Expand Down Expand Up @@ -174,18 +179,36 @@ export default function validateVersionInstance(version: Version): void {
if (version.bindingPrefix) {
validateType(message, version.bindingPrefix, 'bindingPrefix', 'string');
}
const overridesAllowedKeys = ComponentOverrides.componentOverridesDataFields();
const validateOverrides = (dependencies: Object, fieldName) => {
const npmSpecs = PJV.getSpecMap('npm');
const validatePackageJsonField = (fieldName: string, fieldValue: any): ?string => {
if (!npmSpecs[fieldName]) {
// it's not a standard package.json field, can't validate
return null;
}
const validateResult = PJV.validateType(fieldName, npmSpecs[fieldName], fieldValue);
if (!validateResult.length) return null;
return validateResult.join(', ');
};
const validateOverrides = (fieldValue: Object, fieldName: string) => {
const field = `overrides.${fieldName}`;
validateType(message, dependencies, field, 'object');
Object.keys(dependencies).forEach((key) => {
validateType(message, key, `property name of ${field}`, 'string');
validateType(message, dependencies[key], `version of "${field}.${key}"`, 'string');
});
if (dependenciesFields.includes(fieldName)) {
validateType(message, fieldValue, field, 'object');
Object.keys(fieldValue).forEach((key) => {
validateType(message, key, `property name of ${field}`, 'string');
validateType(message, fieldValue[key], `version of "${field}.${key}"`, 'string');
});
} else if (!nonPackageJsonFields.includes(fieldName)) {
const result = validatePackageJsonField(fieldName, fieldValue);
if (result) {
throw new VersionInvalid(
`${message}, "${field}" is a package.json field but is not compliant with npm requirements. ${result}`
);
}
}
};
Object.keys(version.overrides).forEach((field) => {
if (!overridesAllowedKeys.includes(field)) {
throw new VersionInvalid(`${message}, the "overrides" has unidentified key "${field}"`);
if (overridesForbiddenFields.includes(field)) {
throw new VersionInvalid(`${message}, the "overrides" has a forbidden key "${field}"`);
}
// $FlowFixMe
validateOverrides(version.overrides[field], field);
Expand All @@ -197,5 +220,11 @@ export default function validateVersionInstance(version: Version): void {
if (forbiddenPackageJsonProps.includes(prop)) {
throw new VersionInvalid(`${message}, the packageJsonChangedProps should not override the prop ${prop}`);
}
const result = validatePackageJsonField(prop, version.packageJsonChangedProps[prop]);
if (result) {
throw new VersionInvalid(
`${message}, the generated package.json field "${prop}" is not compliant with npm requirements. ${result}`
);
}
});
}

0 comments on commit c7a9b48

Please sign in to comment.