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:
Pete Bacon Darwin 2018-06-04 11:43:15 +01:00 committed by Victor Berchet
parent acf270d724
commit 68d37ef0c1
5 changed files with 79 additions and 13 deletions

View File

@ -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');

View File

@ -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 =

View File

@ -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', () => {

View File

@ -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];
}
});
},
}; };

View File

@ -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 });
});
});
}); });