mirror of
				https://github.com/microsoft/playwright.git
				synced 2025-06-26 21:40:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			901 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			901 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * Copyright (c) Microsoft Corporation.
 | |
|  *
 | |
|  * Licensed 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 CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| // @ts-check
 | |
| 
 | |
| const path = require('path');
 | |
| // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | |
| const Documentation = require('./documentation');
 | |
| const XmlDoc = require('./dotnetXmlDocumentation');
 | |
| const PROJECT_DIR = path.join(__dirname, '..', '..');
 | |
| const fs = require('fs');
 | |
| const { parseApi } = require('./api_parser');
 | |
| const { Type } = require('./documentation');
 | |
| const { EOL } = require('os');
 | |
| 
 | |
| const maxDocumentationColumnWidth = 80;
 | |
| Error.stackTraceLimit = 100;
 | |
| 
 | |
| /** @type {Map<string, Documentation.Type>} */
 | |
| const modelTypes = new Map(); // this will hold types that we discover, because of .NET specifics, like results
 | |
| /** @type {Map<string, string>} */
 | |
| const documentedResults = new Map(); // will hold documentation for new types
 | |
| /** @type {Map<string, string[]>} */
 | |
| const enumTypes = new Map();
 | |
| /** @type {Map<string, Documentation.Type>} */
 | |
| const optionTypes = new Map();
 | |
| const customTypeNames = new Map([
 | |
|   ['domcontentloaded', 'DOMContentLoaded'],
 | |
|   ['networkidle', 'NetworkIdle'],
 | |
| ]);
 | |
| 
 | |
| const outputDir = process.argv[2] || path.join(__dirname, 'generate_types', 'csharp');
 | |
| const apiDir = path.join(outputDir, 'API', 'Generated');
 | |
| const optionsDir = path.join(outputDir, 'API', 'Generated', 'Options');
 | |
| const enumsDir = path.join(outputDir, 'API', 'Generated', 'Enums');
 | |
| const typesDir = path.join(outputDir, 'API', 'Generated', 'Types');
 | |
| 
 | |
| for (const dir of [apiDir, optionsDir, enumsDir, typesDir])
 | |
|   fs.mkdirSync(dir, { recursive: true });
 | |
| 
 | |
| const documentation = parseApi(path.join(PROJECT_DIR, 'docs', 'src', 'api'));
 | |
| documentation.filterForLanguage('csharp');
 | |
| 
 | |
| documentation.setLinkRenderer(item => {
 | |
|   const asyncSuffix = item.member && item.member.async ? 'Async' : '';
 | |
|   if (item.clazz)
 | |
|     return `<see cref="I${toTitleCase(item.clazz.name)}"/>`;
 | |
|   else if (item.member)
 | |
|     return `<see cref="I${toTitleCase(item.member.clazz.name)}.${toMemberName(item.member)}${asyncSuffix}"/>`;
 | |
|   else if (item.option)
 | |
|     return `<paramref name="${item.option.name}"/>`;
 | |
|   else if (item.param)
 | |
|     return `<paramref name="${item.param.name}"/>`;
 | |
|   else
 | |
|     throw new Error('Unknown link format.');
 | |
| });
 | |
| 
 | |
| // get the template for a class
 | |
| const template = fs.readFileSync(path.join(__dirname, 'templates', 'interface.cs'), 'utf-8');
 | |
| 
 | |
| // map the name to a C# friendly one (we prepend an I to denote an interface)
 | |
| const classNameMap = new Map(documentation.classesArray.map(x => [x.name, `I${toTitleCase(x.name)}`]));
 | |
| 
 | |
| // map some types that we know of
 | |
| classNameMap.set('Error', 'Exception');
 | |
| classNameMap.set('TimeoutError', 'TimeoutException');
 | |
| classNameMap.set('EvaluationArgument', 'object');
 | |
| classNameMap.set('boolean', 'bool');
 | |
| classNameMap.set('any', 'object');
 | |
| classNameMap.set('Buffer', 'byte[]');
 | |
| classNameMap.set('path', 'string');
 | |
| classNameMap.set('Date', 'DateTime');
 | |
| classNameMap.set('URL', 'string');
 | |
| classNameMap.set('RegExp', 'Regex');
 | |
| classNameMap.set('Readable', 'Stream');
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param {string} kind
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.MarkdownNode[]|null} spec
 | |
|  * @param {string[]} body
 | |
|  * @param {string} folder
 | |
|  * @param {string|null} extendsName
 | |
|  */
 | |
| function writeFile(kind, name, spec, body, folder, extendsName = null) {
 | |
|   const out = [];
 | |
|   // console.log(`Generating ${name}`);
 | |
| 
 | |
|   if (spec) {
 | |
|     out.push(...XmlDoc.renderXmlDoc(spec, maxDocumentationColumnWidth));
 | |
|   } else {
 | |
|     const ownDocumentation = documentedResults.get(name);
 | |
|     if (ownDocumentation) {
 | |
|       out.push('/// <summary>');
 | |
|       out.push(`/// ${ownDocumentation}`);
 | |
|       out.push('/// </summary>');
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (extendsName === 'IEventEmitter')
 | |
|     extendsName = null;
 | |
| 
 | |
|   if (body[0] === '')
 | |
|     body = body.slice(1);
 | |
| 
 | |
|   out.push(`${kind} ${name}${extendsName ? ` : ${extendsName}` : ''}`);
 | |
|   out.push('{');
 | |
|   out.push(...body);
 | |
|   out.push('}');
 | |
| 
 | |
|   const content = template.replace('[CONTENT]', out.join(EOL));
 | |
|   fs.writeFileSync(path.join(folder, name + '.cs'), content);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {Documentation.Class} clazz
 | |
|  */
 | |
| function renderClass(clazz) {
 | |
|   const name = classNameMap.get(clazz.name);
 | |
|   if (name === 'TimeoutException')
 | |
|     return;
 | |
| 
 | |
|   const body = [];
 | |
|   for (const member of clazz.membersArray) {
 | |
|     // Classes inherit it from IAsyncDisposable
 | |
|     if (member.name === 'dispose')
 | |
|       continue;
 | |
|     if (member.alias.startsWith('RunAnd'))
 | |
|       renderMember(member, clazz, { trimRunAndPrefix: true }, body);
 | |
|     renderMember(member, clazz, {}, body);
 | |
|   }
 | |
| 
 | |
|   /** @type {Documentation.MarkdownNode[]} */
 | |
|   const spec = [];
 | |
|   if (clazz.deprecated)
 | |
|     spec.push({ type: 'text', text: '**DEPRECATED** ' + clazz.deprecated });
 | |
|   if (clazz.discouraged)
 | |
|     spec.push({ type: 'text', text: clazz.discouraged });
 | |
|   if (clazz.spec)
 | |
|     spec.push(...clazz.spec);
 | |
| 
 | |
|   writeFile(
 | |
|       'public partial interface',
 | |
|       name,
 | |
|       spec,
 | |
|       body,
 | |
|       apiDir,
 | |
|       clazz.extends ? `I${toTitleCase(clazz.extends)}` : null);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.Type} type
 | |
|  */
 | |
| function renderModelType(name, type) {
 | |
|   const body = [];
 | |
|   // TODO: consider how this could be merged with the `translateType` check
 | |
|   if (type.union
 | |
|     && type.union[0].name === 'null'
 | |
|     && type.union.length === 2)
 | |
|     type = type.union[1];
 | |
| 
 | |
| 
 | |
|   if (type.name === 'Array') {
 | |
|     throw new Error('Array at this stage is unexpected.');
 | |
|   } else if (type.properties) {
 | |
|     for (const member of type.properties) {
 | |
|       const fakeType = new Type(name, null);
 | |
|       renderMember(member, fakeType, {}, body);
 | |
|     }
 | |
|   } else {
 | |
|     console.log(type);
 | |
|     throw new Error(`Not sure what to do in this case.`);
 | |
|   }
 | |
|   writeFile('public partial class', name, null, body, typesDir);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @param {string[]} literals
 | |
|  */
 | |
| function renderEnum(name, literals) {
 | |
|   const body = [];
 | |
|   for (let literal of literals) {
 | |
|     // strip out the quotes
 | |
|     literal = literal.replace(/[\"]/g, ``);
 | |
|     const escapedName = literal.replace(/[-]/g, ' ')
 | |
|         .split(' ')
 | |
|         .map(word => customTypeNames.get(word) || word[0].toUpperCase() + word.substring(1)).join('');
 | |
| 
 | |
|     body.push(`[EnumMember(Value = "${literal}")]`);
 | |
|     body.push(`${escapedName},`);
 | |
|   }
 | |
|   writeFile('public enum', name, null, body, enumsDir);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.Type} type
 | |
|  */
 | |
| function renderOptionType(name, type) {
 | |
|   const body = [];
 | |
| 
 | |
|   renderConstructors(name, type, body);
 | |
| 
 | |
|   for (const member of type.properties)
 | |
|     renderMember(member, member.type, {}, body);
 | |
|   writeFile('public class', name, null, body, optionsDir);
 | |
| }
 | |
| 
 | |
| for (const element of documentation.classesArray)
 | |
|   renderClass(element);
 | |
| 
 | |
| 
 | |
| for (const [name, type] of optionTypes)
 | |
|   renderOptionType(name, type);
 | |
| 
 | |
| for (const [name, type] of modelTypes)
 | |
|   renderModelType(name, type);
 | |
| 
 | |
| for (const [name, literals] of enumTypes)
 | |
|   renderEnum(name, literals);
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  */
 | |
| function toArgumentName(name) {
 | |
|   return name === 'event' ? `@${name}` : name;
 | |
| }
 | |
| 
 | |
| /**
 | |
| * @param {Documentation.Member} member
 | |
| */
 | |
| function toMemberName(member, makeAsync = false) {
 | |
|   const assumedName = toTitleCase(member.alias || member.name);
 | |
|   if (member.kind === 'interface')
 | |
|     return `I${assumedName}`;
 | |
|   if (makeAsync && member.async)
 | |
|     return assumedName + 'Async';
 | |
|   if (!makeAsync && assumedName.endsWith('Async'))
 | |
|     return assumedName.substring(0, assumedName.length - 'Async'.length);
 | |
|   return assumedName;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @returns {string}
 | |
|  */
 | |
| function toTitleCase(name) {
 | |
|   return name.charAt(0).toUpperCase() + name.substring(1);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.Type} type
 | |
|  * @param {string[]} out
 | |
|  */
 | |
| function renderConstructors(name, type, out) {
 | |
|   out.push(`public ${name}(){}`);
 | |
|   out.push('');
 | |
|   out.push(`public ${name}(${name} clone) {`);
 | |
|   out.push(`if(clone == null) return;`);
 | |
| 
 | |
|   type.properties.forEach(p => {
 | |
|     const propType = translateType(p.type, type, t => generateNameDefault(p, name, t, type));
 | |
|     const propName = toMemberName(p);
 | |
|     const overloads = getPropertyOverloads(propType, p, propName, p.type);
 | |
|     for (const { name } of overloads)
 | |
|       out.push(`${name} = clone.${name};`);
 | |
|   });
 | |
|   out.push(`}`);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {Documentation.Member} member
 | |
|  * @param {string[]} out
 | |
|  */
 | |
| function renderMemberDoc(member, out) {
 | |
|   /** @type {Documentation.MarkdownNode[]} */
 | |
|   const nodes = [];
 | |
|   if (member.deprecated)
 | |
|     nodes.push({ type: 'text', text: '**DEPRECATED** ' + member.deprecated });
 | |
|   if (member.discouraged)
 | |
|     nodes.push({ type: 'text', text: member.discouraged });
 | |
|   if (member.spec)
 | |
|     nodes.push(...member.spec);
 | |
|   out.push(...XmlDoc.renderXmlDoc(nodes, maxDocumentationColumnWidth));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {Documentation.Member} member
 | |
|  * @param {Documentation.Class|Documentation.Type} parent
 | |
|  * @param {{nojson?: boolean, trimRunAndPrefix?: boolean}} options
 | |
|  * @param {string[]} out
 | |
|  */
 | |
| function renderMember(member, parent, options, out) {
 | |
|   const name = toMemberName(member);
 | |
|   if (member.kind === 'method') {
 | |
|     renderMethod(member, parent, name, { trimRunAndPrefix: options.trimRunAndPrefix }, out);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   let type = translateType(member.type, parent, t => generateNameDefault(member, name, t, parent));
 | |
|   if (member.kind === 'event') {
 | |
|     if (!member.type)
 | |
|       throw new Error(`No Event Type for ${name} in ${parent.name}`);
 | |
|     out.push('');
 | |
|     renderMemberDoc(member, out);
 | |
|     if (member.deprecated)
 | |
|       out.push(`[System.Obsolete]`);
 | |
|     out.push(`event EventHandler<${type}> ${name};`);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (member.kind === 'property') {
 | |
|     if (parent && member && member.name === 'children') {  // this is a special hack for Accessibility
 | |
|       console.warn(`children property found in ${parent.name}, assuming array.`);
 | |
|       type = `IEnumerable<${parent.name}>`;
 | |
|     }
 | |
|     const overloads = getPropertyOverloads(type, member, name, parent);
 | |
|     for (const overload of overloads) {
 | |
|       const { name, jsonName } = overload;
 | |
|       let { type } = overload;
 | |
|       out.push('');
 | |
|       renderMemberDoc(member, out);
 | |
|       if (!member.clazz)
 | |
|         out.push(`${member.required ? '[Required]\n' : ''}[JsonPropertyName("${jsonName}")]`);
 | |
|       if (member.deprecated)
 | |
|         out.push(`[System.Obsolete]`);
 | |
|       if (!type.endsWith('?') && !member.required)
 | |
|         type = `${type}?`;
 | |
|       const requiredSuffix = type.endsWith('?') ? '' : ' = default!;';
 | |
|       if (member.clazz)
 | |
|         out.push(`public ${type} ${name} { get; }`);
 | |
|       else
 | |
|         out.push(`public ${type} ${name} { get; set; }${requiredSuffix}`);
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
|   throw new Error(`Problem rendering a member: ${type} - ${name} (${member.kind})`);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param {string} type
 | |
|  * @param {Documentation.Member} member
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.Class|Documentation.Type} parent
 | |
|  * @returns [{ type: string; name: string; jsonName: string; }]
 | |
|  */
 | |
| function getPropertyOverloads(type, member, name, parent) {
 | |
|   const overloads = [];
 | |
|   if (type) {
 | |
|     let jsonName = member.name;
 | |
|     if (member.type.expression === '[string]|[float]')
 | |
|       jsonName = `${member.name}String`;
 | |
|     overloads.push({ type, name, jsonName });
 | |
|   }
 | |
|   return overloads;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param {Documentation.Member} member
 | |
|  * @param {string} name
 | |
|  * @param {Documentation.Type} t
 | |
|  * @param {*} parent
 | |
|  */
 | |
| function generateNameDefault(member, name, t, parent) {
 | |
|   if (!t.properties
 | |
|     && !t.templates
 | |
|     && !t.union
 | |
|     && t.expression === '[Object]')
 | |
|     return 'object';
 | |
| 
 | |
|   // we'd get this call for enums, primarily
 | |
|   const enumName = generateEnumNameIfApplicable(t);
 | |
|   if (!enumName && member) {
 | |
|     if (member.kind === 'method' || member.kind === 'property') {
 | |
|       const names = [
 | |
|         parent.alias || parent.name,
 | |
|         toTitleCase(member.alias || member.name),
 | |
|         toTitleCase(name),
 | |
|       ];
 | |
|       if (names[2] === names[1])
 | |
|         names.pop(); // get rid of duplicates, cheaply
 | |
|       let attemptedName = names.pop();
 | |
|       const typesDiffer = function(/** @type {Documentation.Type} */ left, /** @type {Documentation.Type} */ right) {
 | |
|         if (left.expression && right.expression)
 | |
|           return left.expression !== right.expression;
 | |
|         const toExpression = (/** @type {Documentation.Member} */ t) => t.name + t.type?.expression;
 | |
|         const leftOverRightProperties = new Set(left.properties?.map(toExpression) ?? []);
 | |
|         for (const prop of right.properties ?? []) {
 | |
|           const expression = toExpression(prop);
 | |
|           if (!leftOverRightProperties.has(expression))
 | |
|             return true;
 | |
|           leftOverRightProperties.delete(expression);
 | |
|         }
 | |
|         return leftOverRightProperties.size > 0;
 | |
|       };
 | |
|       while (true) {
 | |
|         // crude attempt at removing plurality
 | |
|         if (attemptedName.endsWith('s')
 | |
|           && !['properties', 'httpcredentials'].includes(attemptedName.toLowerCase()))
 | |
|           attemptedName = attemptedName.substring(0, attemptedName.length - 1);
 | |
| 
 | |
|         // For some of these we don't want to generate generic types.
 | |
|         // For some others we simply did not have the code that was deduping the names.
 | |
|         if (attemptedName === 'BoundingBox')
 | |
|           attemptedName = `${parent.name}BoundingBoxResult`;
 | |
|         if (attemptedName === 'BrowserContextCookie')
 | |
|           attemptedName = 'BrowserContextCookiesResult';
 | |
|         if (attemptedName === 'File' || (parent.name === 'FormData' && ['SetValue', 'AppendValue'].includes(attemptedName)))
 | |
|           attemptedName = `FilePayload`;
 | |
|         if (attemptedName === 'Size')
 | |
|           attemptedName = 'RequestSizesResult';
 | |
|         if (attemptedName === 'ViewportSize' && parent.name === 'Page')
 | |
|           attemptedName = 'PageViewportSizeResult';
 | |
|         if (attemptedName === 'SecurityDetail')
 | |
|           attemptedName = 'ResponseSecurityDetailsResult';
 | |
|         if (attemptedName === 'ServerAddr')
 | |
|           attemptedName = 'ResponseServerAddrResult';
 | |
|         if (attemptedName === 'Timing')
 | |
|           attemptedName = 'RequestTimingResult';
 | |
|         if (attemptedName === 'HeadersArray')
 | |
|           attemptedName = 'Header';
 | |
| 
 | |
|         const probableType = modelTypes.get(attemptedName);
 | |
|         if ((probableType && typesDiffer(t, probableType))
 | |
|           || (['Value'].includes(attemptedName))) {
 | |
|           if (!names.length)
 | |
|             throw new Error(`Ran out of possible names: ${attemptedName}`);
 | |
|           attemptedName = `${names.pop()}${attemptedName}`;
 | |
|           continue;
 | |
|         } else {
 | |
|           registerModelType(attemptedName, t);
 | |
|         }
 | |
|         break;
 | |
|       }
 | |
|       return attemptedName;
 | |
|     }
 | |
| 
 | |
|     if (member.kind === 'event')
 | |
|       return `${name}Payload`;
 | |
| 
 | |
|   }
 | |
| 
 | |
|   return enumName || t.name;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param {Documentation.Type} type
 | |
|  * @returns
 | |
|  */
 | |
| function generateEnumNameIfApplicable(type) {
 | |
|   if (!type.union)
 | |
|     return null;
 | |
| 
 | |
|   const potentialValues = type.union.filter(u => u.name.startsWith('"'));
 | |
|   if ((potentialValues.length !== type.union.length)
 | |
|     && !(type.union[0].name === 'null' && potentialValues.length === type.union.length - 1))
 | |
|     return null; // this isn't an enum, so we don't care, we let the caller generate the name
 | |
| 
 | |
|   return type.name;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Rendering a method is so _special_, with so many weird edge cases, that it
 | |
|  * makes sense to put it separate from the other logic.
 | |
|  * @param {Documentation.Member} member
 | |
|  * @param {Documentation.Class | Documentation.Type} parent
 | |
|  * @param {string} name
 | |
|  * @param {{
 | |
|  *   nodocs?: boolean,
 | |
|  *   abstract?: boolean,
 | |
|  *   public?: boolean,
 | |
|  *   trimRunAndPrefix?: boolean,
 | |
|  * }} options
 | |
|  * @param {string[]} out
 | |
|  */
 | |
| function renderMethod(member, parent, name, options, out) {
 | |
|   out.push('');
 | |
| 
 | |
|   if (options.trimRunAndPrefix)
 | |
|     name = name.substring('RunAnd'.length);
 | |
| 
 | |
|   /** @type {Map<string, string[]>} */
 | |
|   const paramDocs = new Map();
 | |
|   const addParamsDoc = (paramName, docs) => {
 | |
|     if (paramName.startsWith('@'))
 | |
|       paramName = paramName.substring(1);
 | |
|     if (paramDocs.get(paramName) && paramDocs.get(paramName) !== docs)
 | |
|       throw new Error(`Parameter ${paramName} already exists in the docs.`);
 | |
|     paramDocs.set(paramName, docs);
 | |
|   };
 | |
| 
 | |
|   let type = translateType(member.type, parent, t => generateNameDefault(member, name, t, parent), false, true);
 | |
| 
 | |
|   // TODO: this is something that will probably go into the docs
 | |
|   // translate simple getters into read-only properties, and simple
 | |
|   // set-only methods to settable properties
 | |
|   if (member.args.size === 0
 | |
|     && type !== 'void'
 | |
|     && !name.startsWith('Get')
 | |
|     && name !== 'CreateFormData'
 | |
|     && !name.startsWith('PostDataJSON')
 | |
|     && !name.startsWith('As')
 | |
|     && name !== 'ConnectToServer') {
 | |
|     if (!member.async) {
 | |
|       if (member.spec && !options.nodocs)
 | |
|         out.push(...XmlDoc.renderXmlDoc(member.spec, maxDocumentationColumnWidth));
 | |
|       if (member.deprecated)
 | |
|         out.push(`[System.Obsolete]`);
 | |
|       out.push(`${type} ${name} { get; }`);
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // HACK: special case for generics handling!
 | |
|   if (type === 'T')
 | |
|     name = `${name}<T>`;
 | |
| 
 | |
| 
 | |
|   // adjust the return type for async methods
 | |
|   if (member.async) {
 | |
|     if (type === 'void')
 | |
|       type = `Task`;
 | |
|     else
 | |
|       type = `Task<${type}>`;
 | |
|   }
 | |
| 
 | |
|   // render args
 | |
|   /** @type {string[]} */
 | |
|   const args = [];
 | |
|   /** @type {string[]} */
 | |
|   const explodedArgs = [];
 | |
|   /** @type {Map<string, string>} */
 | |
|   const argTypeMap = new Map([]);
 | |
|   /**
 | |
|    *
 | |
|    * @param {string} innerArgType
 | |
|    * @param {string} innerArgName
 | |
|    * @param {Documentation.Member} argument
 | |
|    * @param {boolean} isExploded
 | |
|    */
 | |
|   function pushArg(innerArgType, innerArgName, argument, isExploded = false) {
 | |
|     if (innerArgType === 'null')
 | |
|       return;
 | |
|     const requiredPrefix = (argument.required || isExploded) ? '' : '?';
 | |
|     const requiredSuffix = (argument.required || isExploded) ? '' : ' = default';
 | |
|     const push = `${innerArgType}${requiredPrefix} ${innerArgName}${requiredSuffix}`;
 | |
|     if (isExploded)
 | |
|       explodedArgs.push(push);
 | |
|     else
 | |
|       args.push(push);
 | |
|     argTypeMap.set(push, innerArgName);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * @param {Documentation.Member} arg
 | |
|    */
 | |
|   function processArg(arg) {
 | |
|     if (options.trimRunAndPrefix && arg.name === 'action')
 | |
|       return;
 | |
| 
 | |
|     if (arg.name === 'options') {
 | |
|       const optionsType = rewriteSuggestedOptionsName(member.clazz.name + name.replace('<T>', '') + 'Options');
 | |
|       if (!optionTypes.has(optionsType) || arg.type.properties.length > optionTypes.get(optionsType).properties.length)
 | |
|         optionTypes.set(optionsType, arg.type);
 | |
|       args.push(`${optionsType}? options = default`);
 | |
|       argTypeMap.set(`${optionsType}? options = default`, 'options');
 | |
|       addParamsDoc('options', ['Call options']);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (arg.type.expression === '[string]|[path]') {
 | |
|       const argName = toArgumentName(arg.name);
 | |
|       pushArg('string?', `${argName} = default`, arg);
 | |
|       pushArg('string?', `${argName}Path = default`, arg);
 | |
|       if (arg.spec) {
 | |
|         addParamsDoc(argName, XmlDoc.renderTextOnly(arg.spec, maxDocumentationColumnWidth));
 | |
|         addParamsDoc(`${argName}Path`, [`Instead of specifying <paramref name="${argName}"/>, gives the file name to load from.`]);
 | |
|       }
 | |
|       return;
 | |
|     } else if (arg.type.expression === '[boolean]|[Array]<[string]>') {
 | |
|       // HACK: this hurts my brain too
 | |
|       // we split this into two args, one boolean, with the logical name
 | |
|       const argName = toArgumentName(arg.name);
 | |
|       const leftArgType = translateType(arg.type.union[0], parent, t => { throw new Error('Not supported'); });
 | |
|       const rightArgType = translateType(arg.type.union[1], parent, t => { throw new Error('Not supported'); });
 | |
| 
 | |
|       pushArg(leftArgType, argName, arg);
 | |
|       pushArg(rightArgType, `${argName}Values`, arg);
 | |
| 
 | |
|       addParamsDoc(argName, XmlDoc.renderTextOnly(arg.spec, maxDocumentationColumnWidth));
 | |
|       addParamsDoc(`${argName}Values`, [`The values to take into account when <paramref name="${argName}"/> is <code>true</code>.`]);
 | |
| 
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     const argName = toArgumentName(arg.alias || arg.name);
 | |
|     const argType = translateType(arg.type, parent, t => generateNameDefault(member, argName, t, parent));
 | |
| 
 | |
|     if (argType === null && arg.type.union) {
 | |
|       // we might have to split this into multiple arguments
 | |
|       const translatedArguments = arg.type.union.map(t => translateType(t, parent, x => generateNameDefault(member, argName, x, parent)));
 | |
|       if (translatedArguments.includes(null))
 | |
|         throw new Error('Unexpected null in translated argument types. Aborting.');
 | |
| 
 | |
|       const argDocumentation = XmlDoc.renderTextOnly(arg.spec, maxDocumentationColumnWidth);
 | |
|       for (const newArg of translatedArguments) {
 | |
|         pushArg(newArg, argName, arg, true); // push the exploded arg
 | |
|         addParamsDoc(argName, argDocumentation);
 | |
|       }
 | |
|       args.push(arg.required ? 'EXPLODED_ARG' : 'OPTIONAL_EXPLODED_ARG');
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     addParamsDoc(argName, XmlDoc.renderTextOnly(arg.spec, maxDocumentationColumnWidth));
 | |
| 
 | |
|     if (argName === 'timeout' && argType === 'decimal') {
 | |
|       args.push(`int timeout = 0`); // a special argument, we ignore our convention
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     pushArg(argType, argName, arg);
 | |
|   }
 | |
| 
 | |
|   let modifiers = '';
 | |
|   if (options.abstract)
 | |
|     modifiers = 'protected abstract ';
 | |
|   if (options.public)
 | |
|     modifiers = 'public ';
 | |
| 
 | |
|   member.argsArray
 | |
|       .sort((a, b) => b.alias === 'options' ? -1 : 0) // move options to the back to the arguments list
 | |
|       .forEach(processArg);
 | |
| 
 | |
|   if (!explodedArgs.length) {
 | |
|     if (!options.nodocs) {
 | |
|       renderMemberDoc(member, out);
 | |
|       paramDocs.forEach((value, i) => printArgDoc(i, value, out));
 | |
|     }
 | |
|     if (member.deprecated)
 | |
|       out.push(`[System.Obsolete]`);
 | |
|     out.push(`${modifiers}${type} ${toAsync(name, member.async)}(${args.join(', ')});`);
 | |
|   } else {
 | |
|     let containsOptionalExplodedArgs = false;
 | |
|     explodedArgs.forEach((explodedArg, argIndex) => {
 | |
|       if (!options.nodocs)
 | |
|         renderMemberDoc(member, out);
 | |
|       const overloadedArgs = [];
 | |
|       for (let i = 0; i < args.length; i++) {
 | |
|         const arg = args[i];
 | |
|         if (arg === 'EXPLODED_ARG' || arg === 'OPTIONAL_EXPLODED_ARG') {
 | |
|           containsOptionalExplodedArgs = arg === 'OPTIONAL_EXPLODED_ARG';
 | |
|           const argType = argTypeMap.get(explodedArg);
 | |
|           if (!options.nodocs)
 | |
|             printArgDoc(argType, paramDocs.get(argType), out);
 | |
|           overloadedArgs.push(explodedArg);
 | |
|         } else {
 | |
|           const argType = argTypeMap.get(arg);
 | |
|           if (!options.nodocs)
 | |
|             printArgDoc(argType, paramDocs.get(argType), out);
 | |
|           overloadedArgs.push(arg);
 | |
|         }
 | |
|       }
 | |
|       out.push(`${modifiers}${type} ${toAsync(name, member.async)}(${overloadedArgs.join(', ')});`);
 | |
|       if (argIndex < explodedArgs.length - 1)
 | |
|         out.push(''); // output a special blank line
 | |
|     });
 | |
| 
 | |
|     // If the exploded union arguments are optional, we also output a special
 | |
|     // signature, to help prevent compilation errors with ambiguous overloads.
 | |
|     // That particular overload only contains the required arguments, or rather
 | |
|     // contains all the arguments *except* the exploded ones.
 | |
|     if (containsOptionalExplodedArgs) {
 | |
|       const filteredArgs = args.filter(x => x !== 'OPTIONAL_EXPLODED_ARG');
 | |
|       if (!options.nodocs)
 | |
|         renderMemberDoc(member, out);
 | |
|       filteredArgs.forEach(arg => {
 | |
|         if (arg === 'EXPLODED_ARG')
 | |
|           throw new Error(`Unsupported required union arg combined an optional union inside ${member.name}`);
 | |
|         const argType = argTypeMap.get(arg);
 | |
|         if (!options.nodocs)
 | |
|           printArgDoc(argType, paramDocs.get(argType), out);
 | |
|       });
 | |
|       out.push(`${type} ${name}(${filteredArgs.join(', ')});`);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  *  @param {Documentation.Type} type
 | |
|  *  @param {Documentation.Class|Documentation.Type} parent
 | |
|  *  @param {function(Documentation.Type): string} generateNameCallback
 | |
|  *  @param {boolean=} optional
 | |
|  *  @returns {string}
 | |
|  */
 | |
| function translateType(type, parent, generateNameCallback = t => t.name, optional = false, isReturnType = false) {
 | |
|   // a few special cases we can fix automatically
 | |
|   if (type.expression === '[null]|[Error]')
 | |
|     return 'void';
 | |
| 
 | |
|   if (type.name === 'Promise' && type.templates?.[0].name === 'any')
 | |
|     return 'Task';
 | |
| 
 | |
|   if (type.union) {
 | |
|     if (type.union[0].name === 'null' && type.union.length === 2)
 | |
|       return translateType(type.union[1], parent, generateNameCallback, true, isReturnType);
 | |
| 
 | |
|     if (type.expression === '[string]|[Buffer]')
 | |
|       return `byte[]`; // TODO: make sure we implement extension methods for this!
 | |
|     if (type.expression === '[string]|[float]' || type.expression === '[string]|[float]|[boolean]') {
 | |
|       console.warn(`${type.name} should be a 'string', but was a ${type.expression}`);
 | |
|       return `string`;
 | |
|     }
 | |
|     if (type.union.length === 2 && type.union[1].name === 'Array' && type.union[1].templates[0].name === type.union[0].name)
 | |
|       return `IEnumerable<${type.union[0].name}>`; // an example of this is [string]|[Array]<[string]>
 | |
|     if (type.expression === '[float]|"raf"')
 | |
|       return `Polling`; // hardcoded because there's no other way to denote this
 | |
| 
 | |
|     // Regular primitive enums are named in the markdown.
 | |
|     if (type.name) {
 | |
|       enumTypes.set(type.name, type.union.map(t => t.name));
 | |
|       return optional ? type.name + '?' : type.name;
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   if (type.name === 'Array') {
 | |
|     if (type.templates.length !== 1)
 | |
|       throw new Error(`Array (${type.name} from ${parent.name}) has more than 1 dimension. Panic.`);
 | |
| 
 | |
|     const innerType = translateType(type.templates[0], parent, generateNameCallback, false, isReturnType);
 | |
|     return isReturnType ? `IReadOnlyList<${innerType}>` : `IEnumerable<${innerType}>`;
 | |
|   }
 | |
| 
 | |
|   if (type.name === 'Object') {
 | |
|     // take care of some common cases
 | |
|     // TODO: this can be genericized
 | |
|     if (type.templates && type.templates.length === 2) {
 | |
|       // get the inner types of both templates, and if they're strings, it's a keyvaluepair string, string,
 | |
|       const keyType = translateType(type.templates[0], parent, generateNameCallback, false, isReturnType);
 | |
|       const valueType = translateType(type.templates[1], parent, generateNameCallback, false, isReturnType);
 | |
|       if (['Request', 'Response', 'APIResponse'].includes(parent.name))
 | |
|         return `Dictionary<${keyType}, ${valueType}>`;
 | |
|       return `IEnumerable<KeyValuePair<${keyType}, ${valueType}>>`;
 | |
|     }
 | |
| 
 | |
|     if ((type.name === 'Object')
 | |
|       && !type.properties
 | |
|       && !type.union)
 | |
|       return 'object';
 | |
| 
 | |
|     // this is an additional type that we need to generate
 | |
|     const objectName = generateNameCallback(type);
 | |
|     if (objectName === 'Object')
 | |
|       throw new Error('Object unexpected');
 | |
|     else if (type.name === 'Object')
 | |
|       registerModelType(objectName, type);
 | |
| 
 | |
|     return `${objectName}${optional ? '?' : ''}`;
 | |
|   }
 | |
| 
 | |
|   if (type.name === 'Map') {
 | |
|     if (type.templates && type.templates.length === 2) {
 | |
|       // we map to a dictionary
 | |
|       const keyType = translateType(type.templates[0], parent, generateNameCallback, false, isReturnType);
 | |
|       const valueType = translateType(type.templates[1], parent, generateNameCallback, false, isReturnType);
 | |
|       return `Dictionary<${keyType}, ${valueType}>`;
 | |
|     } else {
 | |
|       throw 'Map has invalid number of templates.';
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (type.name === 'function') {
 | |
|     if (type.expression === '[function]' || !type.args)
 | |
|       return 'Action'; // super simple mapping
 | |
| 
 | |
|     let argsList = '';
 | |
|     if (type.args) {
 | |
|       const translatedCallbackArguments = type.args.map(t => translateType(t, parent, generateNameCallback, false, isReturnType));
 | |
|       if (translatedCallbackArguments.includes(null))
 | |
|         throw new Error('There was an argument we could not parse. Aborting.');
 | |
| 
 | |
|       argsList = translatedCallbackArguments.join(', ');
 | |
|     }
 | |
| 
 | |
|     if (!type.returnType) {
 | |
|       // this is an Action
 | |
|       return `Action<${argsList}>`;
 | |
|     } else {
 | |
|       const returnType = translateType(type.returnType, parent, generateNameCallback, false, isReturnType);
 | |
|       if (returnType === null)
 | |
|         throw new Error('Unexpected null as return type.');
 | |
| 
 | |
|       if (!argsList)
 | |
|         return `Func<${returnType}>`;
 | |
|       return `Func<${argsList}, ${returnType}>`;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (type.templates) {
 | |
|     // this should mean we have a generic type and we can translate that
 | |
|     /** @type {string[]} */
 | |
|     const types = type.templates.map(template => translateType(template, parent));
 | |
|     return `${type.name}<${types.join(', ')}>`;
 | |
|   }
 | |
| 
 | |
|   if (type.name === 'Serializable')
 | |
|     return isReturnType ? 'T' : 'object';
 | |
| 
 | |
|   // there's a chance this is a name we've already seen before, so check
 | |
|   // this is also where we map known types, like boolean -> bool, etc.
 | |
|   const name = classNameMap.get(type.name) || type.name;
 | |
|   return `${name}${optional ? '?' : ''}`;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} typeName
 | |
|  * @param {Documentation.Type} type
 | |
|  */
 | |
| function registerModelType(typeName, type) {
 | |
|   if (['object', 'string', 'int', 'long'].includes(typeName))
 | |
|     return;
 | |
|   if (typeName.endsWith('Option'))
 | |
|     return;
 | |
| 
 | |
|   const potentialType = modelTypes.get(typeName);
 | |
|   if (potentialType) {
 | |
|     // console.log(`Type ${typeName} already exists, so skipping...`);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   modelTypes.set(typeName, type);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @param {string[]} value
 | |
|  * @param {string[]} out
 | |
|  */
 | |
| function printArgDoc(name, value, out) {
 | |
|   if (value.length === 1) {
 | |
|     out.push(`/// <param name="${name}">${value}</param>`);
 | |
|   } else {
 | |
|     out.push(`/// <param name="${name}">`);
 | |
|     out.push(...value.map(l => `/// ${l}`));
 | |
|     out.push(`/// </param>`);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} name
 | |
|  * @param {boolean} convert
 | |
|  */
 | |
| function toAsync(name, convert) {
 | |
|   if (!convert)
 | |
|     return name;
 | |
|   if (name.includes('<'))
 | |
|     return name.replace('<', 'Async<');
 | |
|   return name + 'Async';
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param {string} suggestedName
 | |
|  * @returns {string}
 | |
|  */
 | |
| function rewriteSuggestedOptionsName(suggestedName) {
 | |
|   if ([
 | |
|     'APIRequestContextDeleteOptions',
 | |
|     'APIRequestContextFetchOptions',
 | |
|     'APIRequestContextGetOptions',
 | |
|     'APIRequestContextHeadOptions',
 | |
|     'APIRequestContextPatchOptions',
 | |
|     'APIRequestContextPostOptions',
 | |
|     'APIRequestContextPutOptions',
 | |
|   ].includes(suggestedName))
 | |
|     return 'APIRequestContextOptions';
 | |
|   return suggestedName;
 | |
| }
 | |
| 
 | 
