build(aio): ensure the correct decorator properties are merged (#24289)
Previously only the `description` and `usageNotes` were being copied over from the call-member of the decorator interface. Important properties such as `shortDescription` were missed. These are now added and the code has been refactored to make it simpler and clearer to update which properties get copied as the requirements change. PR Close #24289
This commit is contained in:
parent
acf270d724
commit
68d37ef0c1
|
@ -119,6 +119,17 @@ module.exports = new Package('angular-api', [basePackage, typeScriptPackage])
|
||||||
addNotYetDocumentedProperty.docTypes = API_DOC_TYPES;
|
addNotYetDocumentedProperty.docTypes = API_DOC_TYPES;
|
||||||
})
|
})
|
||||||
|
|
||||||
|
.config(function(mergeDecoratorDocs) {
|
||||||
|
mergeDecoratorDocs.propertiesToMerge = [
|
||||||
|
'shortDescription',
|
||||||
|
'description',
|
||||||
|
'security',
|
||||||
|
'deprecated',
|
||||||
|
'see',
|
||||||
|
'usageNotes',
|
||||||
|
];
|
||||||
|
})
|
||||||
|
|
||||||
.config(function(checkContentRules, EXPORT_DOC_TYPES) {
|
.config(function(checkContentRules, EXPORT_DOC_TYPES) {
|
||||||
// Min length rules
|
// Min length rules
|
||||||
const createMinLengthRule = require('./content-rules/minLength');
|
const createMinLengthRule = require('./content-rules/minLength');
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
const {mergeProperties} = require('../../helpers/utils');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decorators in the Angular code base are made up from three code items:
|
* Decorators in the Angular code base are made up from three code items:
|
||||||
*
|
*
|
||||||
|
@ -47,6 +49,7 @@ module.exports = function mergeDecoratorDocs(log) {
|
||||||
return {
|
return {
|
||||||
$runAfter: ['processing-docs'],
|
$runAfter: ['processing-docs'],
|
||||||
$runBefore: ['docs-processed'],
|
$runBefore: ['docs-processed'],
|
||||||
|
propertiesToMerge: [],
|
||||||
makeDecoratorCalls: [
|
makeDecoratorCalls: [
|
||||||
{type: '', description: 'toplevel', functionName: 'makeDecorator'},
|
{type: '', description: 'toplevel', functionName: 'makeDecorator'},
|
||||||
{type: 'Prop', description: 'property', functionName: 'makePropDecorator'},
|
{type: 'Prop', description: 'property', functionName: 'makePropDecorator'},
|
||||||
|
@ -55,6 +58,7 @@ module.exports = function mergeDecoratorDocs(log) {
|
||||||
$process: function(docs) {
|
$process: function(docs) {
|
||||||
|
|
||||||
var makeDecoratorCalls = this.makeDecoratorCalls;
|
var makeDecoratorCalls = this.makeDecoratorCalls;
|
||||||
|
var propertiesToMerge = this.propertiesToMerge;
|
||||||
var docsToMerge = Object.create(null);
|
var docsToMerge = Object.create(null);
|
||||||
|
|
||||||
docs.forEach(function(doc) {
|
docs.forEach(function(doc) {
|
||||||
|
@ -88,9 +92,9 @@ module.exports = function mergeDecoratorDocs(log) {
|
||||||
log.debug(
|
log.debug(
|
||||||
'mergeDecoratorDocs: merging', doc.name, 'into', decoratorDoc.name,
|
'mergeDecoratorDocs: merging', doc.name, 'into', decoratorDoc.name,
|
||||||
callMember.description.substring(0, 50));
|
callMember.description.substring(0, 50));
|
||||||
|
|
||||||
// Merge the documentation found in this call signature into the original decorator
|
// Merge the documentation found in this call signature into the original decorator
|
||||||
decoratorDoc.description = callMember.description;
|
mergeProperties(decoratorDoc, callMember, propertiesToMerge);
|
||||||
decoratorDoc.usageNotes = callMember.usageNotes;
|
|
||||||
|
|
||||||
// remove doc from its module doc's exports
|
// remove doc from its module doc's exports
|
||||||
doc.moduleDoc.exports =
|
doc.moduleDoc.exports =
|
||||||
|
|
|
@ -9,17 +9,21 @@ describe('mergeDecoratorDocs processor', () => {
|
||||||
const injector = dgeni.configureInjector();
|
const injector = dgeni.configureInjector();
|
||||||
processor = injector.get('mergeDecoratorDocs');
|
processor = injector.get('mergeDecoratorDocs');
|
||||||
|
|
||||||
|
// Note that we do not include usageNotes in the tests.
|
||||||
|
processor.propertiesToMerge = ['description', 'shortDescription'];
|
||||||
|
|
||||||
moduleDoc = {};
|
moduleDoc = {};
|
||||||
|
|
||||||
decoratorDoc = {
|
decoratorDoc = {
|
||||||
name: 'Component',
|
name: 'Component',
|
||||||
docType: 'const',
|
docType: 'const',
|
||||||
description: 'A description of the metadata for the Component decorator',
|
shortDescription: 'decorator - short description',
|
||||||
|
description: 'decorator - description',
|
||||||
symbol: {
|
symbol: {
|
||||||
valueDeclaration: { initializer: { expression: { text: 'makeDecorator' }, arguments: [{ text: 'X' }] } }
|
valueDeclaration: { initializer: { expression: { text: 'makeDecorator' }, arguments: [{ text: 'X' }] } }
|
||||||
},
|
},
|
||||||
members: [
|
members: [
|
||||||
{ name: 'templateUrl', description: 'A description of the templateUrl property' }
|
{ name: 'templateUrl', description: 'templateUrl - description' }
|
||||||
],
|
],
|
||||||
moduleDoc
|
moduleDoc
|
||||||
};
|
};
|
||||||
|
@ -27,15 +31,15 @@ describe('mergeDecoratorDocs processor', () => {
|
||||||
metadataDoc = {
|
metadataDoc = {
|
||||||
name: 'ComponentDecorator',
|
name: 'ComponentDecorator',
|
||||||
docType: 'interface',
|
docType: 'interface',
|
||||||
description: 'A description of the interface for the call signature for the Component decorator',
|
description: 'call interface - description',
|
||||||
members: [
|
members: [
|
||||||
{
|
{
|
||||||
isCallMember: true,
|
isCallMember: true,
|
||||||
description: 'The actual description of the call signature',
|
description: 'call interface - call member - description',
|
||||||
usageNotes: 'Use it like this...'
|
usageNotes: 'call interface - call member - usageNotes',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
description: 'Some other member'
|
description: 'call interface - non call member - description'
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
moduleDoc
|
moduleDoc
|
||||||
|
@ -65,10 +69,13 @@ describe('mergeDecoratorDocs processor', () => {
|
||||||
expect(decoratorDoc.decoratorType).toEqual('X');
|
expect(decoratorDoc.decoratorType).toEqual('X');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should copy across properties from the call signature doc', () => {
|
it('should copy across specified properties from the call signature doc', () => {
|
||||||
processor.$process([decoratorDoc, metadataDoc, otherDoc]);
|
processor.$process([decoratorDoc, metadataDoc, otherDoc]);
|
||||||
expect(decoratorDoc.description).toEqual('The actual description of the call signature');
|
expect(decoratorDoc.description).toEqual('call interface - call member - description');
|
||||||
expect(decoratorDoc.usageNotes).toEqual('Use it like this...');
|
// Since usageNotes is not in `propertiesToMerge` it will not get copied over in these tests.
|
||||||
|
expect(decoratorDoc.usageNotes).toBeUndefined();
|
||||||
|
// Since `shortDescription` does not exist on the call-member this will not get overridden.
|
||||||
|
expect(decoratorDoc.shortDescription).toEqual('decorator - short description');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should remove the metadataDoc from the module exports', () => {
|
it('should remove the metadataDoc from the module exports', () => {
|
||||||
|
|
|
@ -96,5 +96,19 @@ module.exports = {
|
||||||
attrMap[key] === false ? '' :
|
attrMap[key] === false ? '' :
|
||||||
attrMap[key] === true ? ` ${key}` :
|
attrMap[key] === true ? ` ${key}` :
|
||||||
` ${key}="${attrMap[key].replace(/"/g, '"')}"`).join('');
|
` ${key}="${attrMap[key].replace(/"/g, '"')}"`).join('');
|
||||||
}
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Merge the specified properties from the source to the target document
|
||||||
|
* @param {Document} target The document to receive the properties from the source
|
||||||
|
* @param {Document} source The document from which to get the properties to merge
|
||||||
|
* @param {string[]} properties A collection of the names of the properties to merge
|
||||||
|
*/
|
||||||
|
mergeProperties(target, source, properties) {
|
||||||
|
properties.forEach(property => {
|
||||||
|
if (source.hasOwnProperty(property)) {
|
||||||
|
target[property] = source[property];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
const { mapObject, parseAttributes, renderAttributes } = require('./utils');
|
const { mergeProperties, mapObject, parseAttributes, renderAttributes } = require('./utils');
|
||||||
|
|
||||||
describe('utils', () => {
|
describe('utils', () => {
|
||||||
describe('mapObject', () => {
|
describe('mapObject', () => {
|
||||||
|
@ -96,4 +96,34 @@ describe('utils', () => {
|
||||||
expect(renderAttributes({ })).toEqual('');
|
expect(renderAttributes({ })).toEqual('');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('mergeProperties', () => {
|
||||||
|
it('should write specified properties from the source to the target', () => {
|
||||||
|
const source = { a: 1, b: 2, c: 3 };
|
||||||
|
const target = { };
|
||||||
|
mergeProperties(target, source, ['a', 'b']);
|
||||||
|
expect(target).toEqual({ a: 1, b: 2 });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not overwrite target properties that are not specified', () => {
|
||||||
|
const source = { a: 1, b: 2, c: 3 };
|
||||||
|
const target = { b: 10 };
|
||||||
|
mergeProperties(target, source, ['a']);
|
||||||
|
expect(target).toEqual({ a: 1, b: 10 });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not overwrite target properties that are specified but do not exist in the source', () => {
|
||||||
|
const source = { a: 1 };
|
||||||
|
const target = { b: 10 };
|
||||||
|
mergeProperties(target, source, ['a', 'b']);
|
||||||
|
expect(target).toEqual({ a: 1, b: 10 });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should overwrite target properties even if they are `undefined` in the source', () => {
|
||||||
|
const source = { a: undefined };
|
||||||
|
const target = { a: 10 };
|
||||||
|
mergeProperties(target, source, ['a']);
|
||||||
|
expect(target).toEqual({ a: undefined });
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue