зеркало из https://github.com/Azure/ms-rest-js.git
550 строки
22 KiB
TypeScript
550 строки
22 KiB
TypeScript
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// Licensed under the MIT License. See License.txt in the project root for license information.
|
|
|
|
import { ServiceClientCredentials } from "./credentials/serviceClientCredentials";
|
|
import { DefaultHttpClient } from "./defaultHttpClient";
|
|
import { HttpClient } from "./httpClient";
|
|
import { HttpOperationResponse, RestResponse } from "./httpOperationResponse";
|
|
import { HttpPipelineLogger } from "./httpPipelineLogger";
|
|
import { OperationArguments } from "./operationArguments";
|
|
import { getPathStringFromParameter, getPathStringFromParameterPath, OperationParameter, ParameterPath } from "./operationParameter";
|
|
import { isStreamOperation, OperationSpec } from "./operationSpec";
|
|
import { deserializationPolicy } from "./policies/deserializationPolicy";
|
|
import { exponentialRetryPolicy } from "./policies/exponentialRetryPolicy";
|
|
import { generateClientRequestIdPolicy } from "./policies/generateClientRequestIdPolicy";
|
|
import { msRestUserAgentPolicy } from "./policies/msRestUserAgentPolicy";
|
|
import { redirectPolicy } from "./policies/redirectPolicy";
|
|
import { RequestPolicy, RequestPolicyFactory, RequestPolicyOptions } from "./policies/requestPolicy";
|
|
import { rpRegistrationPolicy } from "./policies/rpRegistrationPolicy";
|
|
import { signingPolicy } from "./policies/signingPolicy";
|
|
import { systemErrorRetryPolicy } from "./policies/systemErrorRetryPolicy";
|
|
import { QueryCollectionFormat } from "./queryCollectionFormat";
|
|
import { CompositeMapper, DictionaryMapper, Mapper, MapperType, Serializer } from "./serializer";
|
|
import { URLBuilder } from "./url";
|
|
import { Constants } from "./util/constants";
|
|
import * as utils from "./util/utils";
|
|
import { stringifyXML } from "./util/xml";
|
|
import { RequestOptionsBase, RequestPrepareOptions, WebResource } from "./webResource";
|
|
import { OperationResponse } from "./operationResponse";
|
|
import { ServiceCallback } from "./util/utils";
|
|
|
|
/**
|
|
* Options to be provided while creating the client.
|
|
*/
|
|
export interface ServiceClientOptions {
|
|
/**
|
|
* An array of factories which get called to create the RequestPolicy pipeline
|
|
* used to send a HTTP request on the wire.
|
|
*/
|
|
requestPolicyFactories?: RequestPolicyFactory[];
|
|
/**
|
|
* The HttpClient that will be used to send HTTP requests.
|
|
*/
|
|
httpClient?: HttpClient;
|
|
/**
|
|
* The HttpPipelineLogger that can be used to debug RequestPolicies within the HTTP pipeline.
|
|
*/
|
|
httpPipelineLogger?: HttpPipelineLogger;
|
|
/**
|
|
* If set to true, turn off the default retry policy.
|
|
*/
|
|
noRetryPolicy?: boolean;
|
|
/**
|
|
* Gets or sets the retry timeout in seconds for AutomaticRPRegistration. Default value is 30.
|
|
*/
|
|
rpRegistrationRetryTimeout?: number;
|
|
/**
|
|
* Whether or not to generate a client request ID header for each HTTP request.
|
|
*/
|
|
generateClientRequestIdHeader?: boolean;
|
|
/**
|
|
* Whether to include credentials in CORS requests in the browser.
|
|
* See https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest/withCredentials for more information.
|
|
*/
|
|
withCredentials?: boolean;
|
|
/**
|
|
* If specified, a GenerateRequestIdPolicy will be added to the HTTP pipeline that will add a
|
|
* header to all outgoing requests with this header name and a random UUID as the request ID.
|
|
*/
|
|
clientRequestIdHeaderName?: string;
|
|
}
|
|
|
|
/**
|
|
* @class
|
|
* Initializes a new instance of the ServiceClient.
|
|
*/
|
|
export class ServiceClient {
|
|
/**
|
|
* If specified, this is the base URI that requests will be made against for this ServiceClient.
|
|
* If it is not specified, then all OperationSpecs must contain a baseUrl property.
|
|
*/
|
|
protected baseUri?: string;
|
|
|
|
/**
|
|
* The default request content type for the service.
|
|
* Used if no requestContentType is present on an OperationSpec.
|
|
*/
|
|
protected requestContentType?: string;
|
|
|
|
/**
|
|
* The string to be appended to the User-Agent header while sending the request.
|
|
* This will be applicable only for node.js environment as the fetch library in browser does not allow setting custom UA.
|
|
*/
|
|
userAgentInfo: { value: string[] };
|
|
|
|
/**
|
|
* The HTTP client that will be used to send requests.
|
|
*/
|
|
private readonly _httpClient: HttpClient;
|
|
private readonly _requestPolicyOptions: RequestPolicyOptions;
|
|
|
|
private readonly _requestPolicyFactories: RequestPolicyFactory[];
|
|
private readonly _withCredentials: boolean;
|
|
|
|
/**
|
|
* The ServiceClient constructor
|
|
* @constructor
|
|
* @param {ServiceClientCredentials} [credentials] The credentials object used for authentication.
|
|
* @param {ServiceClientOptions} [options] The service client options that govern the behavior of the client.
|
|
*/
|
|
constructor(credentials?: ServiceClientCredentials, options?: ServiceClientOptions) {
|
|
if (!options) {
|
|
options = {};
|
|
}
|
|
|
|
this.userAgentInfo = { value: [] };
|
|
|
|
if (credentials && !credentials.signRequest) {
|
|
throw new Error("credentials argument needs to implement signRequest method");
|
|
}
|
|
|
|
try {
|
|
const moduleName = "ms-rest-js";
|
|
const moduleVersion = Constants.msRestVersion;
|
|
this.addUserAgentInfo(`${moduleName}/${moduleVersion}`);
|
|
} catch (err) {
|
|
// do nothing
|
|
}
|
|
|
|
this._withCredentials = options.withCredentials || false;
|
|
this._httpClient = options.httpClient || new DefaultHttpClient();
|
|
this._requestPolicyOptions = new RequestPolicyOptions(options.httpPipelineLogger);
|
|
|
|
this._requestPolicyFactories = options.requestPolicyFactories || createDefaultRequestPolicyFactories(credentials, options, this.userAgentInfo.value);
|
|
}
|
|
|
|
/**
|
|
* Adds custom information to user agent header
|
|
* @param {string} additionalUserAgentInfo information to be added to user agent header, as string.
|
|
*/
|
|
addUserAgentInfo(additionalUserAgentInfo: string): void {
|
|
if (this.userAgentInfo.value.indexOf(additionalUserAgentInfo) === -1) {
|
|
this.userAgentInfo.value.push(additionalUserAgentInfo);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Send the provided httpRequest.
|
|
*/
|
|
sendRequest(options: RequestPrepareOptions | WebResource): Promise<HttpOperationResponse> {
|
|
if (options === null || options === undefined || typeof options !== "object") {
|
|
throw new Error("options cannot be null or undefined and it must be of type object.");
|
|
}
|
|
|
|
let httpRequest: WebResource;
|
|
try {
|
|
if (options instanceof WebResource) {
|
|
options.validateRequestProperties();
|
|
httpRequest = options;
|
|
} else {
|
|
httpRequest = new WebResource();
|
|
httpRequest = httpRequest.prepare(options);
|
|
}
|
|
} catch (error) {
|
|
return Promise.reject(error);
|
|
}
|
|
|
|
let httpPipeline: RequestPolicy = this._httpClient;
|
|
if (this._requestPolicyFactories && this._requestPolicyFactories.length > 0) {
|
|
for (let i = this._requestPolicyFactories.length - 1; i >= 0; --i) {
|
|
httpPipeline = this._requestPolicyFactories[i].create(httpPipeline, this._requestPolicyOptions);
|
|
}
|
|
}
|
|
return httpPipeline.sendRequest(httpRequest);
|
|
}
|
|
|
|
/**
|
|
* Send an HTTP request that is populated using the provided OperationSpec.
|
|
* @param {OperationArguments} operationArguments The arguments that the HTTP request's templated values will be populated from.
|
|
* @param {OperationSpec} operationSpec The OperationSpec to use to populate the httpRequest.
|
|
* @param {ServiceCallback} callback The callback to call when the response is received.
|
|
*/
|
|
sendOperationRequest(operationArguments: OperationArguments, operationSpec: OperationSpec, callback?: ServiceCallback<any>): Promise<RestResponse> {
|
|
if (typeof operationArguments.options === "function") {
|
|
callback = operationArguments.options;
|
|
operationArguments.options = undefined;
|
|
}
|
|
|
|
const httpRequest = new WebResource();
|
|
|
|
let result: Promise<RestResponse>;
|
|
try {
|
|
const baseUri: string | undefined = operationSpec.baseUrl || this.baseUri;
|
|
if (!baseUri) {
|
|
throw new Error("If operationSpec.baseUrl is not specified, then the ServiceClient must have a baseUri string property that contains the base URL to use.");
|
|
}
|
|
|
|
httpRequest.method = operationSpec.httpMethod;
|
|
httpRequest.operationSpec = operationSpec;
|
|
|
|
const requestUrl: URLBuilder = URLBuilder.parse(baseUri);
|
|
if (operationSpec.path) {
|
|
requestUrl.appendPath(operationSpec.path);
|
|
}
|
|
if (operationSpec.urlParameters && operationSpec.urlParameters.length > 0) {
|
|
for (const urlParameter of operationSpec.urlParameters) {
|
|
let urlParameterValue: string = getOperationArgumentValueFromParameter(this, operationArguments, urlParameter, operationSpec.serializer);
|
|
urlParameterValue = operationSpec.serializer.serialize(urlParameter.mapper, urlParameterValue, getPathStringFromParameter(urlParameter));
|
|
if (!urlParameter.skipEncoding) {
|
|
urlParameterValue = encodeURIComponent(urlParameterValue);
|
|
}
|
|
requestUrl.replaceAll(`{${urlParameter.mapper.serializedName || getPathStringFromParameter(urlParameter)}}`, urlParameterValue);
|
|
}
|
|
}
|
|
if (operationSpec.queryParameters && operationSpec.queryParameters.length > 0) {
|
|
for (const queryParameter of operationSpec.queryParameters) {
|
|
let queryParameterValue: any = getOperationArgumentValueFromParameter(this, operationArguments, queryParameter, operationSpec.serializer);
|
|
if (queryParameterValue != undefined) {
|
|
queryParameterValue = operationSpec.serializer.serialize(queryParameter.mapper, queryParameterValue, getPathStringFromParameter(queryParameter));
|
|
if (queryParameter.collectionFormat != undefined) {
|
|
if (queryParameter.collectionFormat === QueryCollectionFormat.Multi) {
|
|
if (queryParameterValue.length === 0) {
|
|
queryParameterValue = "";
|
|
} else {
|
|
for (const index in queryParameterValue) {
|
|
const item = queryParameterValue[index];
|
|
queryParameterValue[index] = item == undefined ? "" : item.toString();
|
|
}
|
|
}
|
|
} else {
|
|
queryParameterValue = queryParameterValue.join(queryParameter.collectionFormat);
|
|
}
|
|
}
|
|
if (!queryParameter.skipEncoding) {
|
|
if (Array.isArray(queryParameterValue)) {
|
|
for (const index in queryParameterValue) {
|
|
queryParameterValue[index] = encodeURIComponent(queryParameterValue[index]);
|
|
}
|
|
}
|
|
else {
|
|
queryParameterValue = encodeURIComponent(queryParameterValue);
|
|
}
|
|
}
|
|
requestUrl.setQueryParameter(queryParameter.mapper.serializedName || getPathStringFromParameter(queryParameter), queryParameterValue);
|
|
}
|
|
}
|
|
}
|
|
httpRequest.url = requestUrl.toString();
|
|
|
|
const contentType = operationSpec.contentType || this.requestContentType;
|
|
if (contentType) {
|
|
httpRequest.headers.set("Content-Type", contentType);
|
|
}
|
|
|
|
if (operationSpec.headerParameters) {
|
|
for (const headerParameter of operationSpec.headerParameters) {
|
|
let headerValue: any = getOperationArgumentValueFromParameter(this, operationArguments, headerParameter, operationSpec.serializer);
|
|
if (headerValue != undefined) {
|
|
headerValue = operationSpec.serializer.serialize(headerParameter.mapper, headerValue, getPathStringFromParameter(headerParameter));
|
|
const headerCollectionPrefix = (headerParameter.mapper as DictionaryMapper).headerCollectionPrefix;
|
|
if (headerCollectionPrefix) {
|
|
for (const key of Object.keys(headerValue)) {
|
|
httpRequest.headers.set(headerCollectionPrefix + key, headerValue[key]);
|
|
}
|
|
} else {
|
|
httpRequest.headers.set(headerParameter.mapper.serializedName || getPathStringFromParameter(headerParameter), headerValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const options: RequestOptionsBase | undefined = operationArguments.options;
|
|
if (options) {
|
|
if (options.customHeaders) {
|
|
for (const customHeaderName in options.customHeaders) {
|
|
httpRequest.headers.set(customHeaderName, options.customHeaders[customHeaderName]);
|
|
}
|
|
}
|
|
|
|
if (options.abortSignal) {
|
|
httpRequest.abortSignal = options.abortSignal;
|
|
}
|
|
|
|
if (options.timeout) {
|
|
httpRequest.timeout = options.timeout;
|
|
}
|
|
|
|
if (options.onUploadProgress) {
|
|
httpRequest.onUploadProgress = options.onUploadProgress;
|
|
}
|
|
|
|
if (options.onDownloadProgress) {
|
|
httpRequest.onDownloadProgress = options.onDownloadProgress;
|
|
}
|
|
}
|
|
|
|
httpRequest.withCredentials = this._withCredentials;
|
|
|
|
serializeRequestBody(this, httpRequest, operationArguments, operationSpec);
|
|
|
|
if (httpRequest.streamResponseBody == undefined) {
|
|
httpRequest.streamResponseBody = isStreamOperation(operationSpec);
|
|
}
|
|
|
|
result = this.sendRequest(httpRequest)
|
|
.then(res => flattenResponse(res, operationSpec.responses[res.status]));
|
|
} catch (error) {
|
|
result = Promise.reject(error);
|
|
}
|
|
|
|
const cb = callback;
|
|
if (cb) {
|
|
result
|
|
// tslint:disable-next-line:no-null-keyword
|
|
.then(res => cb(null, res._response.parsedBody, res._response.request, res._response))
|
|
.catch(err => cb(err));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
|
|
export function serializeRequestBody(serviceClient: ServiceClient, httpRequest: WebResource, operationArguments: OperationArguments, operationSpec: OperationSpec): void {
|
|
if (operationSpec.requestBody && operationSpec.requestBody.mapper) {
|
|
httpRequest.body = getOperationArgumentValueFromParameter(serviceClient, operationArguments, operationSpec.requestBody, operationSpec.serializer);
|
|
|
|
const bodyMapper = operationSpec.requestBody.mapper;
|
|
const { required, xmlName, xmlElementName, serializedName } = bodyMapper;
|
|
const typeName = bodyMapper.type.name;
|
|
try {
|
|
if (httpRequest.body != undefined || required) {
|
|
const requestBodyParameterPathString: string = getPathStringFromParameter(operationSpec.requestBody);
|
|
httpRequest.body = operationSpec.serializer.serialize(bodyMapper, httpRequest.body, requestBodyParameterPathString);
|
|
const isStream = typeName === MapperType.Stream;
|
|
if (operationSpec.isXML) {
|
|
if (typeName === MapperType.Sequence) {
|
|
httpRequest.body = stringifyXML(utils.prepareXMLRootList(httpRequest.body, xmlElementName || xmlName || serializedName!), { rootName: xmlName || serializedName });
|
|
}
|
|
else if (!isStream) {
|
|
httpRequest.body = stringifyXML(httpRequest.body, { rootName: xmlName || serializedName });
|
|
}
|
|
} else if (!isStream) {
|
|
httpRequest.body = JSON.stringify(httpRequest.body);
|
|
}
|
|
}
|
|
} catch (error) {
|
|
throw new Error(`Error "${error.message}" occurred in serializing the payload - ${JSON.stringify(serializedName, undefined, " ")}.`);
|
|
}
|
|
} else if (operationSpec.formDataParameters && operationSpec.formDataParameters.length > 0) {
|
|
httpRequest.formData = {};
|
|
for (const formDataParameter of operationSpec.formDataParameters) {
|
|
const formDataParameterValue: any = getOperationArgumentValueFromParameter(serviceClient, operationArguments, formDataParameter, operationSpec.serializer);
|
|
if (formDataParameterValue != undefined) {
|
|
const formDataParameterPropertyName: string = formDataParameter.mapper.serializedName || getPathStringFromParameter(formDataParameter);
|
|
httpRequest.formData[formDataParameterPropertyName] = operationSpec.serializer.serialize(formDataParameter.mapper, formDataParameterValue, getPathStringFromParameter(formDataParameter));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function isRequestPolicyFactory(instance: any): instance is RequestPolicyFactory {
|
|
return typeof instance.create === "function";
|
|
}
|
|
|
|
function createDefaultRequestPolicyFactories(credentials: ServiceClientCredentials | RequestPolicyFactory | undefined, options: ServiceClientOptions, userAgentInfo: string[]): RequestPolicyFactory[] {
|
|
const factories: RequestPolicyFactory[] = [];
|
|
|
|
if (options.generateClientRequestIdHeader) {
|
|
factories.push(generateClientRequestIdPolicy(options.clientRequestIdHeaderName));
|
|
}
|
|
|
|
if (credentials) {
|
|
if (isRequestPolicyFactory(credentials)) {
|
|
factories.push(credentials);
|
|
} else {
|
|
factories.push(signingPolicy(credentials));
|
|
}
|
|
}
|
|
|
|
if (utils.isNode) {
|
|
factories.push(msRestUserAgentPolicy(userAgentInfo));
|
|
}
|
|
|
|
factories.push(redirectPolicy());
|
|
factories.push(rpRegistrationPolicy(options.rpRegistrationRetryTimeout));
|
|
|
|
if (!options.noRetryPolicy) {
|
|
factories.push(exponentialRetryPolicy());
|
|
factories.push(systemErrorRetryPolicy());
|
|
}
|
|
|
|
factories.push(deserializationPolicy());
|
|
|
|
return factories;
|
|
}
|
|
|
|
export type PropertyParent = { [propertyName: string]: any };
|
|
|
|
/**
|
|
* Get the property parent for the property at the provided path when starting with the provided
|
|
* parent object.
|
|
*/
|
|
export function getPropertyParent(parent: PropertyParent, propertyPath: string[]): PropertyParent {
|
|
if (parent && propertyPath) {
|
|
const propertyPathLength: number = propertyPath.length;
|
|
for (let i = 0; i < propertyPathLength - 1; ++i) {
|
|
const propertyName: string = propertyPath[i];
|
|
if (!parent[propertyName]) {
|
|
parent[propertyName] = {};
|
|
}
|
|
parent = parent[propertyName];
|
|
}
|
|
}
|
|
return parent;
|
|
}
|
|
|
|
function getOperationArgumentValueFromParameter(serviceClient: ServiceClient, operationArguments: OperationArguments, parameter: OperationParameter, serializer: Serializer): any {
|
|
return getOperationArgumentValueFromParameterPath(serviceClient, operationArguments, parameter.parameterPath, parameter.mapper, serializer);
|
|
}
|
|
|
|
export function getOperationArgumentValueFromParameterPath(serviceClient: ServiceClient, operationArguments: OperationArguments, parameterPath: ParameterPath, parameterMapper: Mapper, serializer: Serializer): any {
|
|
let value: any;
|
|
if (typeof parameterPath === "string") {
|
|
parameterPath = [parameterPath];
|
|
}
|
|
if (Array.isArray(parameterPath)) {
|
|
if (parameterPath.length > 0) {
|
|
if (parameterMapper.isConstant) {
|
|
value = parameterMapper.defaultValue;
|
|
} else {
|
|
let propertySearchResult: PropertySearchResult = getPropertyFromParameterPath(operationArguments, parameterPath);
|
|
if (!propertySearchResult.propertyFound) {
|
|
propertySearchResult = getPropertyFromParameterPath(serviceClient, parameterPath);
|
|
}
|
|
|
|
let useDefaultValue = false;
|
|
if (!propertySearchResult.propertyFound) {
|
|
useDefaultValue = parameterMapper.required || (parameterPath[0] === "options" && parameterPath.length === 2);
|
|
}
|
|
value = useDefaultValue ? parameterMapper.defaultValue : propertySearchResult.propertyValue;
|
|
}
|
|
|
|
// Serialize just for validation purposes.
|
|
const parameterPathString: string = getPathStringFromParameterPath(parameterPath, parameterMapper);
|
|
serializer.serialize(parameterMapper, value, parameterPathString);
|
|
}
|
|
} else {
|
|
for (const propertyName in parameterPath) {
|
|
const propertyMapper: Mapper = (parameterMapper as CompositeMapper).type.modelProperties![propertyName];
|
|
const propertyPath: ParameterPath = parameterPath[propertyName];
|
|
const propertyValue: any = getOperationArgumentValueFromParameterPath(serviceClient, operationArguments, propertyPath, propertyMapper, serializer);
|
|
// Serialize just for validation purposes.
|
|
const propertyPathString: string = getPathStringFromParameterPath(propertyPath, propertyMapper);
|
|
serializer.serialize(propertyMapper, propertyValue, propertyPathString);
|
|
if (propertyValue !== undefined) {
|
|
if (!value) {
|
|
value = {};
|
|
}
|
|
value[propertyName] = propertyValue;
|
|
}
|
|
}
|
|
}
|
|
return value;
|
|
}
|
|
|
|
interface PropertySearchResult {
|
|
propertyValue?: any;
|
|
propertyFound: boolean;
|
|
}
|
|
|
|
function getPropertyFromParameterPath(parent: { [parameterName: string]: any }, parameterPath: string[]): PropertySearchResult {
|
|
const result: PropertySearchResult = { propertyFound: false };
|
|
let i = 0;
|
|
for (; i < parameterPath.length; ++i) {
|
|
const parameterPathPart: string = parameterPath[i];
|
|
// Make sure to check inherited properties too, so don't use hasOwnProperty().
|
|
if (parent != undefined && parameterPathPart in parent) {
|
|
parent = parent[parameterPathPart];
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (i === parameterPath.length) {
|
|
result.propertyValue = parent;
|
|
result.propertyFound = true;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
export function flattenResponse(_response: HttpOperationResponse, responseSpec: OperationResponse | undefined): RestResponse {
|
|
const parsedHeaders = _response.parsedHeaders;
|
|
const bodyMapper = responseSpec && responseSpec.bodyMapper;
|
|
|
|
const addOperationResponse = (obj: {}) =>
|
|
Object.defineProperty(obj, "_response", {
|
|
value: _response
|
|
});
|
|
|
|
if (bodyMapper) {
|
|
const typeName = bodyMapper.type.name;
|
|
if (typeName === "Stream") {
|
|
return addOperationResponse({
|
|
...parsedHeaders,
|
|
blobBody: _response.blobBody,
|
|
readableStreamBody: _response.readableStreamBody
|
|
});
|
|
}
|
|
|
|
const modelProperties = typeName === "Composite" && (bodyMapper as CompositeMapper).type.modelProperties || {};
|
|
const isPageableResponse = Object.keys(modelProperties).some(k => modelProperties[k].serializedName === "");
|
|
if (typeName === "Sequence" || isPageableResponse) {
|
|
const arrayResponse = [...(_response.parsedBody || [])] as RestResponse & any[];
|
|
|
|
for (const key of Object.keys(modelProperties)) {
|
|
if (modelProperties[key].serializedName) {
|
|
arrayResponse[key] = _response.parsedBody[key];
|
|
}
|
|
}
|
|
|
|
if (parsedHeaders) {
|
|
for (const key of Object.keys(parsedHeaders)) {
|
|
arrayResponse[key] = parsedHeaders[key];
|
|
}
|
|
}
|
|
addOperationResponse(arrayResponse);
|
|
return arrayResponse;
|
|
}
|
|
|
|
if (typeName === "Composite" || typeName === "Dictionary") {
|
|
return addOperationResponse({
|
|
...parsedHeaders,
|
|
..._response.parsedBody
|
|
});
|
|
}
|
|
}
|
|
|
|
if (bodyMapper || _response.request.method === "HEAD") {
|
|
// primitive body types and HEAD booleans
|
|
return addOperationResponse({
|
|
...parsedHeaders,
|
|
body: _response.parsedBody
|
|
});
|
|
}
|
|
|
|
return addOperationResponse({
|
|
...parsedHeaders,
|
|
..._response.parsedBody
|
|
});
|
|
}
|