From 13388044d3f20993276674ef1827c613594b0e80 Mon Sep 17 00:00:00 2001 From: Stefan Aebischer Date: Fri, 6 Dec 2019 13:30:59 -0500 Subject: [PATCH 1/5] Improved error handling for performAuthorizationRequest Resolves: #94 --- built/authorization_request_handler.d.ts | 3 +- built/authorization_request_handler.js | 2 +- built/node_support/node_request_handler.d.ts | 2 +- built/node_support/node_request_handler.js | 19 ++++++---- built/redirect_based_handler.d.ts | 2 +- built/redirect_based_handler.js | 35 +++++++++--------- src/app/index.ts | 6 +++- src/authorization_request_handler.ts | 3 +- src/node_app/index.ts | 5 ++- src/node_support/node_request_handler.ts | 22 +++++++++--- src/redirect_based_handler.ts | 37 +++++++++++--------- 11 files changed, 86 insertions(+), 50 deletions(-) diff --git a/built/authorization_request_handler.d.ts b/built/authorization_request_handler.d.ts index 9d42238..16e04e8 100644 --- a/built/authorization_request_handler.d.ts +++ b/built/authorization_request_handler.d.ts @@ -52,8 +52,9 @@ export declare abstract class AuthorizationRequestHandler { setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler; /** * Makes an authorization request. + * Returns a `Promise`, when the request was sent succesfully. */ - abstract performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): void; + abstract performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; /** * Checks if an authorization flow can be completed, and completes it. * The handler returns a `Promise` if ready, or a `Promise` diff --git a/built/authorization_request_handler.js b/built/authorization_request_handler.js index 01ddaac..cfbb4fd 100644 --- a/built/authorization_request_handler.js +++ b/built/authorization_request_handler.js @@ -111,4 +111,4 @@ var AuthorizationRequestHandler = /** @class */ (function () { return AuthorizationRequestHandler; }()); exports.AuthorizationRequestHandler = AuthorizationRequestHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authorization_request_handler.js","sourceRoot":"","sources":["../src/authorization_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response';\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto} from './crypto_utils';\nimport {log} from './logger';\nimport {QueryStringUtils} from './query_string_utils';\nimport {StringMap} from './types';\n\n\n/**\n * This type represents a lambda that can take an AuthorizationRequest,\n * and an AuthorizationResponse as arguments.\n */\nexport type AuthorizationListener =\n    (request: AuthorizationRequest,\n     response: AuthorizationResponse|null,\n     error: AuthorizationError|null) => void;\n\n/**\n * Represents a structural type holding both authorization request and response.\n */\nexport interface AuthorizationRequestResponse {\n  request: AuthorizationRequest;\n  response: AuthorizationResponse|null;\n  error: AuthorizationError|null;\n}\n\n/**\n * Authorization Service notifier.\n * This manages the communication of the AuthorizationResponse to the 3p client.\n */\nexport class AuthorizationNotifier {\n  private listener: AuthorizationListener|null = null;\n\n  setAuthorizationListener(listener: AuthorizationListener) {\n    this.listener = listener;\n  }\n\n  /**\n   * The authorization complete callback.\n   */\n  onAuthorizationComplete(\n      request: AuthorizationRequest,\n      response: AuthorizationResponse|null,\n      error: AuthorizationError|null): void {\n    if (this.listener) {\n      // complete authorization request\n      this.listener(request, response, error);\n    }\n  }\n}\n\n// TODO(rahulrav@): add more built in parameters.\n/* built in parameters. */\nexport const BUILT_IN_PARAMETERS = ['redirect_uri', 'client_id', 'response_type', 'state', 'scope'];\n\n/**\n * Defines the interface which is capable of handling an authorization request\n * using various methods (iframe / popup / different process etc.).\n */\nexport abstract class AuthorizationRequestHandler {\n  constructor(public utils: QueryStringUtils, protected crypto: Crypto) {}\n\n  // notifier send the response back to the client.\n  protected notifier: AuthorizationNotifier|null = null;\n\n  /**\n   * A utility method to be able to build the authorization request URL.\n   */\n  protected buildRequestUrl(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // build the query string\n    // coerce to any type for convenience\n    let requestMap: StringMap = {\n      'redirect_uri': request.redirectUri,\n      'client_id': request.clientId,\n      'response_type': request.responseType,\n      'state': request.state,\n      'scope': request.scope\n    };\n\n    // copy over extras\n    if (request.extras) {\n      for (let extra in request.extras) {\n        if (request.extras.hasOwnProperty(extra)) {\n          // check before inserting to requestMap\n          if (BUILT_IN_PARAMETERS.indexOf(extra) < 0) {\n            requestMap[extra] = request.extras[extra];\n          }\n        }\n      }\n    }\n\n    let query = this.utils.stringify(requestMap);\n    let baseUrl = configuration.authorizationEndpoint;\n    let url = `${baseUrl}?${query}`;\n    return url;\n  }\n\n  /**\n   * Completes the authorization request if necessary & when possible.\n   */\n  completeAuthorizationRequestIfPossible(): Promise<void> {\n    // call complete authorization if possible to see there might\n    // be a response that needs to be delivered.\n    log(`Checking to see if there is an authorization response to be delivered.`);\n    if (!this.notifier) {\n      log(`Notifier is not present on AuthorizationRequest handler.\n          No delivery of result will be possible`)\n    }\n    return this.completeAuthorizationRequest().then(result => {\n      if (!result) {\n        log(`No result is available yet.`);\n      }\n      if (result && this.notifier) {\n        this.notifier.onAuthorizationComplete(result.request, result.response, result.error);\n      }\n    });\n  }\n\n  /**\n   * Sets the default Authorization Service notifier.\n   */\n  setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler {\n    this.notifier = notifier;\n    return this;\n  };\n\n  /**\n   * Makes an authorization request.\n   */\n  abstract performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest): void;\n\n  /**\n   * Checks if an authorization flow can be completed, and completes it.\n   * The handler returns a `Promise<AuthorizationRequestResponse>` if ready, or a `Promise<null>`\n   * if not ready.\n   */\n  protected abstract completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null>;\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authorization_request_handler.js","sourceRoot":"","sources":["../src/authorization_request_handler.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;GAYG;;AAMH,mCAA6B;AAuB7B;;;GAGG;AACH;IAAA;QACU,aAAQ,GAA+B,IAAI,CAAC;IAkBtD,CAAC;IAhBC,wDAAwB,GAAxB,UAAyB,QAA+B;QACtD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;IAC3B,CAAC;IAED;;OAEG;IACH,uDAAuB,GAAvB,UACI,OAA6B,EAC7B,QAAoC,EACpC,KAA8B;QAChC,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,iCAAiC;YACjC,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;SACzC;IACH,CAAC;IACH,4BAAC;AAAD,CAAC,AAnBD,IAmBC;AAnBY,sDAAqB;AAqBlC,iDAAiD;AACjD,0BAA0B;AACb,QAAA,mBAAmB,GAAG,CAAC,cAAc,EAAE,WAAW,EAAE,eAAe,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;AAEpG;;;GAGG;AACH;IACE,qCAAmB,KAAuB,EAAY,MAAc;QAAjD,UAAK,GAAL,KAAK,CAAkB;QAAY,WAAM,GAAN,MAAM,CAAQ;QAEpE,iDAAiD;QACvC,aAAQ,GAA+B,IAAI,CAAC;IAHiB,CAAC;IAKxE;;OAEG;IACO,qDAAe,GAAzB,UACI,aAAgD,EAChD,OAA6B;QAC/B,yBAAyB;QACzB,qCAAqC;QACrC,IAAI,UAAU,GAAc;YAC1B,cAAc,EAAE,OAAO,CAAC,WAAW;YACnC,WAAW,EAAE,OAAO,CAAC,QAAQ;YAC7B,eAAe,EAAE,OAAO,CAAC,YAAY;YACrC,OAAO,EAAE,OAAO,CAAC,KAAK;YACtB,OAAO,EAAE,OAAO,CAAC,KAAK;SACvB,CAAC;QAEF,mBAAmB;QACnB,IAAI,OAAO,CAAC,MAAM,EAAE;YAClB,KAAK,IAAI,KAAK,IAAI,OAAO,CAAC,MAAM,EAAE;gBAChC,IAAI,OAAO,CAAC,MAAM,CAAC,cAAc,CAAC,KAAK,CAAC,EAAE;oBACxC,uCAAuC;oBACvC,IAAI,2BAAmB,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE;wBAC1C,UAAU,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;qBAC3C;iBACF;aACF;SACF;QAED,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QAC7C,IAAI,OAAO,GAAG,aAAa,CAAC,qBAAqB,CAAC;QAClD,IAAI,GAAG,GAAM,OAAO,SAAI,KAAO,CAAC;QAChC,OAAO,GAAG,CAAC;IACb,CAAC;IAED;;OAEG;IACH,4EAAsC,GAAtC;QAAA,iBAgBC;QAfC,6DAA6D;QAC7D,4CAA4C;QAC5C,YAAG,CAAC,wEAAwE,CAAC,CAAC;QAC9E,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAClB,YAAG,CAAC,4GACuC,CAAC,CAAA;SAC7C;QACD,OAAO,IAAI,CAAC,4BAA4B,EAAE,CAAC,IAAI,CAAC,UAAA,MAAM;YACpD,IAAI,CAAC,MAAM,EAAE;gBACX,YAAG,CAAC,6BAA6B,CAAC,CAAC;aACpC;YACD,IAAI,MAAM,IAAI,KAAI,CAAC,QAAQ,EAAE;gBAC3B,KAAI,CAAC,QAAQ,CAAC,uBAAuB,CAAC,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;aACtF;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,8DAAwB,GAAxB,UAAyB,QAA+B;QACtD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,OAAO,IAAI,CAAC;IACd,CAAC;IAAA,CAAC;IAgBJ,kCAAC;AAAD,CAAC,AAnFD,IAmFC;AAnFqB,kEAA2B","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response';\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto} from './crypto_utils';\nimport {log} from './logger';\nimport {QueryStringUtils} from './query_string_utils';\nimport {StringMap} from './types';\n\n\n/**\n * This type represents a lambda that can take an AuthorizationRequest,\n * and an AuthorizationResponse as arguments.\n */\nexport type AuthorizationListener =\n    (request: AuthorizationRequest,\n     response: AuthorizationResponse|null,\n     error: AuthorizationError|null) => void;\n\n/**\n * Represents a structural type holding both authorization request and response.\n */\nexport interface AuthorizationRequestResponse {\n  request: AuthorizationRequest;\n  response: AuthorizationResponse|null;\n  error: AuthorizationError|null;\n}\n\n/**\n * Authorization Service notifier.\n * This manages the communication of the AuthorizationResponse to the 3p client.\n */\nexport class AuthorizationNotifier {\n  private listener: AuthorizationListener|null = null;\n\n  setAuthorizationListener(listener: AuthorizationListener) {\n    this.listener = listener;\n  }\n\n  /**\n   * The authorization complete callback.\n   */\n  onAuthorizationComplete(\n      request: AuthorizationRequest,\n      response: AuthorizationResponse|null,\n      error: AuthorizationError|null): void {\n    if (this.listener) {\n      // complete authorization request\n      this.listener(request, response, error);\n    }\n  }\n}\n\n// TODO(rahulrav@): add more built in parameters.\n/* built in parameters. */\nexport const BUILT_IN_PARAMETERS = ['redirect_uri', 'client_id', 'response_type', 'state', 'scope'];\n\n/**\n * Defines the interface which is capable of handling an authorization request\n * using various methods (iframe / popup / different process etc.).\n */\nexport abstract class AuthorizationRequestHandler {\n  constructor(public utils: QueryStringUtils, protected crypto: Crypto) {}\n\n  // notifier send the response back to the client.\n  protected notifier: AuthorizationNotifier|null = null;\n\n  /**\n   * A utility method to be able to build the authorization request URL.\n   */\n  protected buildRequestUrl(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // build the query string\n    // coerce to any type for convenience\n    let requestMap: StringMap = {\n      'redirect_uri': request.redirectUri,\n      'client_id': request.clientId,\n      'response_type': request.responseType,\n      'state': request.state,\n      'scope': request.scope\n    };\n\n    // copy over extras\n    if (request.extras) {\n      for (let extra in request.extras) {\n        if (request.extras.hasOwnProperty(extra)) {\n          // check before inserting to requestMap\n          if (BUILT_IN_PARAMETERS.indexOf(extra) < 0) {\n            requestMap[extra] = request.extras[extra];\n          }\n        }\n      }\n    }\n\n    let query = this.utils.stringify(requestMap);\n    let baseUrl = configuration.authorizationEndpoint;\n    let url = `${baseUrl}?${query}`;\n    return url;\n  }\n\n  /**\n   * Completes the authorization request if necessary & when possible.\n   */\n  completeAuthorizationRequestIfPossible(): Promise<void> {\n    // call complete authorization if possible to see there might\n    // be a response that needs to be delivered.\n    log(`Checking to see if there is an authorization response to be delivered.`);\n    if (!this.notifier) {\n      log(`Notifier is not present on AuthorizationRequest handler.\n          No delivery of result will be possible`)\n    }\n    return this.completeAuthorizationRequest().then(result => {\n      if (!result) {\n        log(`No result is available yet.`);\n      }\n      if (result && this.notifier) {\n        this.notifier.onAuthorizationComplete(result.request, result.response, result.error);\n      }\n    });\n  }\n\n  /**\n   * Sets the default Authorization Service notifier.\n   */\n  setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler {\n    this.notifier = notifier;\n    return this;\n  };\n\n  /**\n   * Makes an authorization request.\n   * Returns a `Promise<null>`, when the request was sent succesfully.\n   */\n  abstract performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest): Promise<null>;\n\n  /**\n   * Checks if an authorization flow can be completed, and completes it.\n   * The handler returns a `Promise<AuthorizationRequestResponse>` if ready, or a `Promise<null>`\n   * if not ready.\n   */\n  protected abstract completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null>;\n}\n"]} \ No newline at end of file diff --git a/built/node_support/node_request_handler.d.ts b/built/node_support/node_request_handler.d.ts index 3728765..443763b 100644 --- a/built/node_support/node_request_handler.d.ts +++ b/built/node_support/node_request_handler.d.ts @@ -7,6 +7,6 @@ export declare class NodeBasedHandler extends AuthorizationRequestHandler { httpServerPort: number; authorizationPromise: Promise | null; constructor(httpServerPort?: number, utils?: QueryStringUtils, crypto?: Crypto); - performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): void; + performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; protected completeAuthorizationRequest(): Promise; } diff --git a/built/node_support/node_request_handler.js b/built/node_support/node_request_handler.js index 3a2b1c4..38855bd 100644 --- a/built/node_support/node_request_handler.js +++ b/built/node_support/node_request_handler.js @@ -29,6 +29,7 @@ Object.defineProperty(exports, "__esModule", { value: true }); var EventEmitter = require("events"); var Http = require("http"); var Url = require("url"); +var errors_1 = require("../errors"); var authorization_request_handler_1 = require("../authorization_request_handler"); var authorization_response_1 = require("../authorization_response"); var logger_1 = require("../logger"); @@ -41,6 +42,7 @@ var ServerEventsEmitter = /** @class */ (function (_super) { function ServerEventsEmitter() { return _super !== null && _super.apply(this, arguments) || this; } + ServerEventsEmitter.ON_START = 'start'; ServerEventsEmitter.ON_UNABLE_TO_START = 'unable_to_start'; ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE = 'authorization_response'; return ServerEventsEmitter; @@ -98,10 +100,8 @@ var NodeBasedHandler = /** @class */ (function (_super) { emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse); response.end('Close your browser to continue'); }; - this.authorizationPromise = new Promise(function (resolve, reject) { - emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function () { - reject("Unable to create HTTP server at port " + _this.httpServerPort); - }); + this.authorizationPromise = new Promise(function (resolve) { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function () { return resolve(null); }); emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, function (result) { server.close(); // resolve pending promise @@ -118,10 +118,17 @@ var NodeBasedHandler = /** @class */ (function (_super) { var url = _this.buildRequestUrl(configuration, request); logger_1.log('Making a request to ', request, url); opener(url); + emitter.emit(ServerEventsEmitter.ON_START); }) .catch(function (error) { logger_1.log('Something bad happened ', error); - emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START); + emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); + }); + return new Promise(function (resolve, reject) { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function (error) { + reject(new errors_1.AppAuthError("Unable to create HTTP server at port " + _this.httpServerPort, { origError: error })); + }); + emitter.once(ServerEventsEmitter.ON_START, function () { return resolve(null); }); }); }; NodeBasedHandler.prototype.completeAuthorizationRequest = function () { @@ -133,4 +140,4 @@ var NodeBasedHandler = /** @class */ (function (_super) { return NodeBasedHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.NodeBasedHandler = NodeBasedHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => {\n        reject(`Unable to create HTTP server at port ${this.httpServerPort}`);\n      });\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort);\n          const url = this.buildRequestUrl(configuration, request);\n          log('Making a request to ', request, url);\n          opener(url);\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START);\n        });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;GAYG;;;;;;;;;;;;;;;AAEH,qCAAuC;AACvC,2BAA6B;AAC7B,yBAA2B;AAC3B,oCAAuC;AAEvC,kFAA2G;AAC3G,oEAAoF;AAGpF,oCAA8B;AAC9B,4DAA8E;AAC9E,+CAA0C;AAG1C,iFAAiF;AACjF,+BAAkC;AAElC;IAAkC,uCAAY;IAA9C;;IAIA,CAAC;IAHQ,4BAAQ,GAAG,OAAO,CAAC;IACnB,sCAAkB,GAAG,iBAAiB,CAAC;IACvC,6CAAyB,GAAG,wBAAwB,CAAC;IAC9D,0BAAC;CAAA,AAJD,CAAkC,YAAY,GAI7C;AAED;IAAsC,oCAA2B;IAI/D;IACI,uBAAuB;IAChB,cAAqB,EAC5B,KAAqD,EACrD,MAAiC;QAF1B,+BAAA,EAAA,qBAAqB;QAC5B,sBAAA,EAAA,YAA8B,0CAAqB,EAAE;QACrD,uBAAA,EAAA,aAAqB,yBAAU,EAAE;QAJrC,YAKE,kBAAM,KAAK,EAAE,MAAM,CAAC,SACrB;QAJU,oBAAc,GAAd,cAAc,CAAO;QALhC,kDAAkD;QAClD,0BAAoB,GAAoD,IAAI,CAAC;;IAQ7E,CAAC;IAED,sDAA2B,GAA3B,UACI,aAAgD,EAChD,OAA6B;QAFjC,iBAkFC;QA/EC,uEAAuE;QACvE,2DAA2D;QAC3D,IAAM,OAAO,GAAG,IAAI,mBAAmB,EAAE,CAAC;QAE1C,IAAM,cAAc,GAAG,UAAC,WAAiC,EAAE,QAA6B;YACtF,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE;gBACpB,OAAO;aACR;YAED,IAAM,GAAG,GAAG,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;YACvC,IAAM,YAAY,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;YAE9D,IAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,SAAS,CAAC;YACrD,IAAM,IAAI,GAAG,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACtC,IAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAExC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE;gBAC7B,gDAAgD;gBAChD,OAAO;aACR;YAED,YAAG,CAAC,iCAAiC,EAAE,YAAY,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC;YACzE,IAAI,qBAAqB,GAA+B,IAAI,CAAC;YAC7D,IAAI,kBAAkB,GAA4B,IAAI,CAAC;YACvD,IAAI,KAAK,EAAE;gBACT,YAAG,CAAC,OAAO,CAAC,CAAC;gBACb,gCAAgC;gBAChC,IAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,WAAW,CAAC,IAAI,SAAS,CAAC;gBAC5D,IAAM,gBAAgB,GAAG,YAAY,CAAC,GAAG,CAAC,mBAAmB,CAAC,IAAI,SAAS,CAAC;gBAC5E,kBAAkB,GAAG,IAAI,2CAAkB,CACvC,EAAC,KAAK,EAAE,KAAK,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,SAAS,EAAE,QAAQ,EAAE,KAAK,EAAE,KAAK,EAAC,CAAC,CAAC;aAC7F;iBAAM;gBACL,qBAAqB,GAAG,IAAI,8CAAqB,CAAC,EAAC,IAAI,EAAE,IAAK,EAAE,KAAK,EAAE,KAAM,EAAC,CAAC,CAAC;aACjF;YACD,IAAM,gBAAgB,GAAG;gBACvB,OAAO,SAAA;gBACP,QAAQ,EAAE,qBAAqB;gBAC/B,KAAK,EAAE,kBAAkB;aACM,CAAC;YAClC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,yBAAyB,EAAE,gBAAgB,CAAC,CAAC;YAC9E,QAAQ,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;QACjD,CAAC,CAAC;QAEF,IAAI,CAAC,oBAAoB,GAAG,IAAI,OAAO,CAAoC,UAAC,OAAO;YACjF,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,cAAM,OAAA,OAAO,CAAC,IAAI,CAAC,EAAb,CAAa,CAAC,CAAC;YAC1E,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,yBAAyB,EAAE,UAAC,MAAW;gBACtE,MAAM,CAAC,KAAK,EAAE,CAAC;gBACf,0BAA0B;gBAC1B,OAAO,CAAC,MAAsC,CAAC,CAAC;gBAChD,8BAA8B;gBAC9B,KAAI,CAAC,sCAAsC,EAAE,CAAC;YAChD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,MAAmB,CAAC;QACxB,OAAO,CAAC,iBAAiB,EAAE;aACtB,IAAI,CAAC;YACJ,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,CAAC,MAAM,CAAC,KAAI,CAAC,cAAc,CAAC,CAAC;YACnC,IAAM,GAAG,GAAG,KAAI,CAAC,eAAe,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;YACzD,YAAG,CAAC,sBAAsB,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC;YAC1C,MAAM,CAAC,GAAG,CAAC,CAAC;YACZ,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC;QAC7C,CAAC,CAAC;aACD,KAAK,CAAC,UAAC,KAAK;YACX,YAAG,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;YACtC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,KAAK,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEP,OAAO,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;YACvC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,UAAC,KAAK;gBACzD,MAAM,CAAC,IAAI,qBAAY,CACrB,0CAAwC,KAAI,CAAC,cAAgB,EAC7D,EAAC,SAAS,EAAE,KAAK,EAAC,CACnB,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,EAAE,cAAM,OAAA,OAAO,CAAC,IAAI,CAAC,EAAb,CAAa,CAAC,CAAC;QAClE,CAAC,CAAC,CAAC;IACL,CAAC;IAES,uDAA4B,GAAtC;QACE,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE;YAC9B,OAAO,OAAO,CAAC,MAAM,CACjB,wEAAwE,CAAC,CAAC;SAC/E;QAED,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IACH,uBAAC;AAAD,CAAC,AAxGD,CAAsC,2DAA2B,GAwGhE;AAxGY,4CAAgB","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort);\n          const url = this.buildRequestUrl(configuration, request);\n          log('Making a request to ', request, url);\n          opener(url);\n          emitter.emit(ServerEventsEmitter.ON_START);\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file diff --git a/built/redirect_based_handler.d.ts b/built/redirect_based_handler.d.ts index 5fcdabc..6b5b698 100644 --- a/built/redirect_based_handler.d.ts +++ b/built/redirect_based_handler.d.ts @@ -13,7 +13,7 @@ export declare class RedirectRequestHandler extends AuthorizationRequestHandler storageBackend: StorageBackend; locationLike: LocationLike; constructor(storageBackend?: StorageBackend, utils?: BasicQueryStringUtils, locationLike?: LocationLike, crypto?: Crypto); - performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): void; + performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; /** * Attempts to introspect the contents of storage backend and completes the * request. diff --git a/built/redirect_based_handler.js b/built/redirect_based_handler.js index cd12d4f..92c23be 100644 --- a/built/redirect_based_handler.js +++ b/built/redirect_based_handler.js @@ -65,21 +65,24 @@ var RedirectRequestHandler = /** @class */ (function (_super) { } RedirectRequestHandler.prototype.performAuthorizationRequest = function (configuration, request) { var _this = this; - var handle = this.crypto.generateRandom(10); - // before you make request, persist all request related data in local storage. - var persisted = Promise.all([ - this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle), - // Calling toJson() adds in the code & challenge when possible - request.toJson().then(function (result) { - return _this.storageBackend.setItem(authorizationRequestKey(handle), JSON.stringify(result)); - }), - this.storageBackend.setItem(authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())), - ]); - persisted.then(function () { - // make the redirect request - var url = _this.buildRequestUrl(configuration, request); - logger_1.log('Making a request to ', request, url); - _this.locationLike.assign(url); + return new Promise(function (resolve, reject) { + var handle = _this.crypto.generateRandom(10); + // before you make request, persist all request related data in local storage. + var persisted = Promise.all([ + _this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle), + // Calling toJson() adds in the code & challenge when possible + request.toJson().then(function (result) { return _this.storageBackend.setItem(authorizationRequestKey(handle), JSON.stringify(result)); }), + _this.storageBackend.setItem(authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())), + ]); + persisted + .then(function () { + // make the redirect request + var url = _this.buildRequestUrl(configuration, request); + logger_1.log('Making a request to ', request, url); + _this.locationLike.assign(url); + resolve(null); + }) + .catch(function (error) { return reject(error); }); }); }; /** @@ -155,4 +158,4 @@ var RedirectRequestHandler = /** @class */ (function (_super) { return RedirectRequestHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.RedirectRequestHandler = RedirectRequestHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redirect_based_handler.js","sourceRoot":"","sources":["../src/redirect_based_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from './authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response'\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto, DefaultCrypto} from './crypto_utils';\nimport {log} from './logger';\nimport {BasicQueryStringUtils} from './query_string_utils';\nimport {LocalStorageBackend, StorageBackend} from './storage';\nimport {LocationLike} from './types';\n\n\n/** key for authorization request. */\nconst authorizationRequestKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_request`;\n    }\n\n/** key for authorization service configuration */\nconst authorizationServiceConfigurationKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_service_configuration`;\n    }\n\n/** key in local storage which represents the current authorization request. */\nconst AUTHORIZATION_REQUEST_HANDLE_KEY = 'appauth_current_authorization_request';\n\n/**\n * Represents an AuthorizationRequestHandler which uses a standard\n * redirect based code flow.\n */\nexport class RedirectRequestHandler extends AuthorizationRequestHandler {\n  constructor(\n      // use the provided storage backend\n      // or initialize local storage with the default storage backend which\n      // uses window.localStorage\n      public storageBackend: StorageBackend = new LocalStorageBackend(),\n      utils = new BasicQueryStringUtils(),\n      public locationLike: LocationLike = window.location,\n      crypto: Crypto = new DefaultCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    const handle = this.crypto.generateRandom(10);\n\n    // before you make request, persist all request related data in local storage.\n    const persisted = Promise.all([\n      this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle),\n      // Calling toJson() adds in the code & challenge when possible\n      request.toJson().then(\n          result =>\n              this.storageBackend.setItem(authorizationRequestKey(handle), JSON.stringify(result))),\n      this.storageBackend.setItem(\n          authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())),\n    ]);\n\n    persisted.then(() => {\n      // make the redirect request\n      let url = this.buildRequestUrl(configuration, request);\n      log('Making a request to ', request, url);\n      this.locationLike.assign(url);\n    });\n  }\n\n  /**\n   * Attempts to introspect the contents of storage backend and completes the\n   * request.\n   */\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    // TODO(rahulrav@): handle authorization errors.\n    return this.storageBackend.getItem(AUTHORIZATION_REQUEST_HANDLE_KEY).then(handle => {\n      if (handle) {\n        // we have a pending request.\n        // fetch authorization request, and check state\n        return this.storageBackend\n            .getItem(authorizationRequestKey(handle))\n            // requires a corresponding instance of result\n            // TODO(rahulrav@): check for inconsitent state here\n            .then(result => JSON.parse(result!))\n            .then(json => new AuthorizationRequest(json))\n            .then(request => {\n              // check redirect_uri and state\n              let currentUri = `${this.locationLike.origin}${this.locationLike.pathname}`;\n              let queryParams = this.utils.parse(this.locationLike, true /* use hash */);\n              let state: string|undefined = queryParams['state'];\n              let code: string|undefined = queryParams['code'];\n              let error: string|undefined = queryParams['error'];\n              log('Potential authorization request ', currentUri, queryParams, state, code, error);\n              let shouldNotify = state === request.state;\n              let authorizationResponse: AuthorizationResponse|null = null;\n              let authorizationError: AuthorizationError|null = null;\n              if (shouldNotify) {\n                if (error) {\n                  // get additional optional info.\n                  let errorUri = queryParams['error_uri'];\n                  let errorDescription = queryParams['error_description'];\n                  authorizationError = new AuthorizationError({\n                    error: error,\n                    error_description: errorDescription,\n                    error_uri: errorUri,\n                    state: state\n                  });\n                } else {\n                  authorizationResponse = new AuthorizationResponse({code: code, state: state});\n                }\n                // cleanup state\n                return Promise\n                    .all([\n                      this.storageBackend.removeItem(AUTHORIZATION_REQUEST_HANDLE_KEY),\n                      this.storageBackend.removeItem(authorizationRequestKey(handle)),\n                      this.storageBackend.removeItem(authorizationServiceConfigurationKey(handle))\n                    ])\n                    .then(() => {\n                      log('Delivering authorization response');\n                      return {\n                        request: request,\n                        response: authorizationResponse,\n                        error: authorizationError\n                      } as AuthorizationRequestResponse;\n                    });\n              } else {\n                log('Mismatched request (state and request_uri) dont match.');\n                return Promise.resolve(null);\n              }\n            });\n      } else {\n        return null;\n      }\n    });\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redirect_based_handler.js","sourceRoot":"","sources":["../src/redirect_based_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from './authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response'\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto, DefaultCrypto} from './crypto_utils';\nimport {log} from './logger';\nimport {BasicQueryStringUtils} from './query_string_utils';\nimport {LocalStorageBackend, StorageBackend} from './storage';\nimport {LocationLike} from './types';\n\n\n/** key for authorization request. */\nconst authorizationRequestKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_request`;\n    }\n\n/** key for authorization service configuration */\nconst authorizationServiceConfigurationKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_service_configuration`;\n    }\n\n/** key in local storage which represents the current authorization request. */\nconst AUTHORIZATION_REQUEST_HANDLE_KEY = 'appauth_current_authorization_request';\n\n/**\n * Represents an AuthorizationRequestHandler which uses a standard\n * redirect based code flow.\n */\nexport class RedirectRequestHandler extends AuthorizationRequestHandler {\n  constructor(\n      // use the provided storage backend\n      // or initialize local storage with the default storage backend which\n      // uses window.localStorage\n      public storageBackend: StorageBackend = new LocalStorageBackend(),\n      utils = new BasicQueryStringUtils(),\n      public locationLike: LocationLike = window.location,\n      crypto: Crypto = new DefaultCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    return new Promise<null>((resolve, reject) => {\n      const handle = this.crypto.generateRandom(10);\n\n      // before you make request, persist all request related data in local storage.\n      const persisted = Promise.all([\n        this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle),\n        // Calling toJson() adds in the code & challenge when possible\n        request.toJson().then(\n            result => this.storageBackend.setItem(\n                authorizationRequestKey(handle), JSON.stringify(result))),\n        this.storageBackend.setItem(\n            authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())),\n      ]);\n\n      persisted\n          .then(() => {\n            // make the redirect request\n            let url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            this.locationLike.assign(url);\n            resolve(null);\n          })\n          .catch(error => reject(error));\n    });\n  }\n\n  /**\n   * Attempts to introspect the contents of storage backend and completes the\n   * request.\n   */\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    // TODO(rahulrav@): handle authorization errors.\n    return this.storageBackend.getItem(AUTHORIZATION_REQUEST_HANDLE_KEY).then(handle => {\n      if (handle) {\n        // we have a pending request.\n        // fetch authorization request, and check state\n        return this.storageBackend\n            .getItem(authorizationRequestKey(handle))\n            // requires a corresponding instance of result\n            // TODO(rahulrav@): check for inconsitent state here\n            .then(result => JSON.parse(result!))\n            .then(json => new AuthorizationRequest(json))\n            .then(request => {\n              // check redirect_uri and state\n              let currentUri = `${this.locationLike.origin}${this.locationLike.pathname}`;\n              let queryParams = this.utils.parse(this.locationLike, true /* use hash */);\n              let state: string|undefined = queryParams['state'];\n              let code: string|undefined = queryParams['code'];\n              let error: string|undefined = queryParams['error'];\n              log('Potential authorization request ', currentUri, queryParams, state, code, error);\n              let shouldNotify = state === request.state;\n              let authorizationResponse: AuthorizationResponse|null = null;\n              let authorizationError: AuthorizationError|null = null;\n              if (shouldNotify) {\n                if (error) {\n                  // get additional optional info.\n                  let errorUri = queryParams['error_uri'];\n                  let errorDescription = queryParams['error_description'];\n                  authorizationError = new AuthorizationError({\n                    error: error,\n                    error_description: errorDescription,\n                    error_uri: errorUri,\n                    state: state\n                  });\n                } else {\n                  authorizationResponse = new AuthorizationResponse({code: code, state: state});\n                }\n                // cleanup state\n                return Promise\n                    .all([\n                      this.storageBackend.removeItem(AUTHORIZATION_REQUEST_HANDLE_KEY),\n                      this.storageBackend.removeItem(authorizationRequestKey(handle)),\n                      this.storageBackend.removeItem(authorizationServiceConfigurationKey(handle))\n                    ])\n                    .then(() => {\n                      log('Delivering authorization response');\n                      return {\n                        request: request,\n                        response: authorizationResponse,\n                        error: authorizationError\n                      } as AuthorizationRequestResponse;\n                    });\n              } else {\n                log('Mismatched request (state and request_uri) dont match.');\n                return Promise.resolve(null);\n              }\n            });\n      } else {\n        return null;\n      }\n    });\n  }\n}\n"]} \ No newline at end of file diff --git a/src/app/index.ts b/src/app/index.ts index a4a8d41..ced9a49 100644 --- a/src/app/index.ts +++ b/src/app/index.ts @@ -113,7 +113,11 @@ export class App { }); if (this.configuration) { - this.authorizationHandler.performAuthorizationRequest(this.configuration, request); + this.authorizationHandler.performAuthorizationRequest(this.configuration, request) + .catch(error => { + log('Something bad happened', error); + this.showMessage(`Something bad happened ${error}`) + }); } else { this.showMessage( 'Fetch Authorization Service configuration, before you make the authorization request.'); diff --git a/src/authorization_request_handler.ts b/src/authorization_request_handler.ts index 1739718..955418d 100644 --- a/src/authorization_request_handler.ts +++ b/src/authorization_request_handler.ts @@ -143,10 +143,11 @@ export abstract class AuthorizationRequestHandler { /** * Makes an authorization request. + * Returns a `Promise`, when the request was sent succesfully. */ abstract performAuthorizationRequest( configuration: AuthorizationServiceConfiguration, - request: AuthorizationRequest): void; + request: AuthorizationRequest): Promise; /** * Checks if an authorization flow can be completed, and completes it. diff --git a/src/node_app/index.ts b/src/node_app/index.ts index d0f2fe7..bc84e74 100644 --- a/src/node_app/index.ts +++ b/src/node_app/index.ts @@ -86,7 +86,10 @@ export class App { }, new NodeCrypto()); log('Making authorization request ', configuration, request); - this.authorizationHandler.performAuthorizationRequest(configuration, request); + this.authorizationHandler.performAuthorizationRequest(configuration, request) + .catch(error => { + log('Something bad happened ', error); + }); } makeRefreshTokenRequest( diff --git a/src/node_support/node_request_handler.ts b/src/node_support/node_request_handler.ts index 83fc1e8..64908ae 100644 --- a/src/node_support/node_request_handler.ts +++ b/src/node_support/node_request_handler.ts @@ -15,6 +15,7 @@ import * as EventEmitter from 'events'; import * as Http from 'http'; import * as Url from 'url'; +import {AppAuthError} from '../errors'; import {AuthorizationRequest} from '../authorization_request'; import {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler'; import {AuthorizationError, AuthorizationResponse} from '../authorization_response'; @@ -29,6 +30,7 @@ import {NodeCrypto} from './crypto_utils'; import opener = require('opener'); class ServerEventsEmitter extends EventEmitter { + static ON_START = 'start'; static ON_UNABLE_TO_START = 'unable_to_start'; static ON_AUTHORIZATION_RESPONSE = 'authorization_response'; } @@ -91,10 +93,8 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { response.end('Close your browser to continue'); }; - this.authorizationPromise = new Promise((resolve, reject) => { - emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => { - reject(`Unable to create HTTP server at port ${this.httpServerPort}`); - }); + this.authorizationPromise = new Promise((resolve) => { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null)); emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => { server.close(); // resolve pending promise @@ -112,11 +112,23 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { const url = this.buildRequestUrl(configuration, request); log('Making a request to ', request, url); opener(url); + emitter.emit(ServerEventsEmitter.ON_START); }) .catch((error) => { log('Something bad happened ', error); - emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START); + emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); }); + + return new Promise((resolve, reject) => { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => { + reject(new AppAuthError( + `Unable to create HTTP server at port ${this.httpServerPort}`, + {origError: error} + )); + }); + + emitter.once(ServerEventsEmitter.ON_START, () => resolve(null)); + }); } protected completeAuthorizationRequest(): Promise { diff --git a/src/redirect_based_handler.ts b/src/redirect_based_handler.ts index 7a5f7ea..259c684 100644 --- a/src/redirect_based_handler.ts +++ b/src/redirect_based_handler.ts @@ -57,24 +57,29 @@ export class RedirectRequestHandler extends AuthorizationRequestHandler { performAuthorizationRequest( configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest) { - const handle = this.crypto.generateRandom(10); + return new Promise((resolve, reject) => { + const handle = this.crypto.generateRandom(10); - // before you make request, persist all request related data in local storage. - const persisted = Promise.all([ - this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle), - // Calling toJson() adds in the code & challenge when possible - request.toJson().then( - result => - this.storageBackend.setItem(authorizationRequestKey(handle), JSON.stringify(result))), - this.storageBackend.setItem( - authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())), - ]); + // before you make request, persist all request related data in local storage. + const persisted = Promise.all([ + this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle), + // Calling toJson() adds in the code & challenge when possible + request.toJson().then( + result => this.storageBackend.setItem( + authorizationRequestKey(handle), JSON.stringify(result))), + this.storageBackend.setItem( + authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())), + ]); - persisted.then(() => { - // make the redirect request - let url = this.buildRequestUrl(configuration, request); - log('Making a request to ', request, url); - this.locationLike.assign(url); + persisted + .then(() => { + // make the redirect request + let url = this.buildRequestUrl(configuration, request); + log('Making a request to ', request, url); + this.locationLike.assign(url); + resolve(null); + }) + .catch(error => reject(error)); }); } From 785b4fe1f8ff71b4c58464f37ab985e57cd16460 Mon Sep 17 00:00:00 2001 From: Stefan Aebischer Date: Fri, 6 Dec 2019 17:20:58 -0500 Subject: [PATCH 2/5] Handle server startup errors Resolves: #95 --- built/node_support/node_request_handler.js | 17 +++++++++++------ src/node_support/node_request_handler.ts | 16 +++++++++++----- 2 files changed, 22 insertions(+), 11 deletions(-) diff --git a/built/node_support/node_request_handler.js b/built/node_support/node_request_handler.js index 38855bd..1d02dd1 100644 --- a/built/node_support/node_request_handler.js +++ b/built/node_support/node_request_handler.js @@ -114,11 +114,16 @@ var NodeBasedHandler = /** @class */ (function (_super) { request.setupCodeVerifier() .then(function () { server = Http.createServer(requestHandler); - server.listen(_this.httpServerPort); - var url = _this.buildRequestUrl(configuration, request); - logger_1.log('Making a request to ', request, url); - opener(url); - emitter.emit(ServerEventsEmitter.ON_START); + server.listen(_this.httpServerPort, function () { + var url = _this.buildRequestUrl(configuration, request); + logger_1.log('Making a request to ', request, url); + opener(url); + emitter.emit(ServerEventsEmitter.ON_START); + }); + server.on('error', function (error) { + logger_1.log('Something bad happened ', error); + emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); + }); }) .catch(function (error) { logger_1.log('Something bad happened ', error); @@ -140,4 +145,4 @@ var NodeBasedHandler = /** @class */ (function (_super) { return NodeBasedHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.NodeBasedHandler = NodeBasedHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort);\n          const url = this.buildRequestUrl(configuration, request);\n          log('Making a request to ', request, url);\n          opener(url);\n          emitter.emit(ServerEventsEmitter.ON_START);\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file diff --git a/src/node_support/node_request_handler.ts b/src/node_support/node_request_handler.ts index 64908ae..b250a8d 100644 --- a/src/node_support/node_request_handler.ts +++ b/src/node_support/node_request_handler.ts @@ -108,11 +108,17 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { request.setupCodeVerifier() .then(() => { server = Http.createServer(requestHandler); - server.listen(this.httpServerPort); - const url = this.buildRequestUrl(configuration, request); - log('Making a request to ', request, url); - opener(url); - emitter.emit(ServerEventsEmitter.ON_START); + server.listen(this.httpServerPort, () => { + const url = this.buildRequestUrl(configuration, request); + log('Making a request to ', request, url); + opener(url); + emitter.emit(ServerEventsEmitter.ON_START); + }); + + server.on('error', (error: Error) => { + log('Something bad happened ', error); + emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); + }); }) .catch((error) => { log('Something bad happened ', error); From 9c3448eaadbcec81af8530a7e1b33bdd2aeaddfb Mon Sep 17 00:00:00 2001 From: Stefan Aebischer Date: Fri, 6 Dec 2019 18:07:10 -0500 Subject: [PATCH 3/5] Only listen on loopback interface Resolves: #93 --- built/node_support/node_request_handler.js | 4 ++-- src/node_support/node_request_handler.ts | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/built/node_support/node_request_handler.js b/built/node_support/node_request_handler.js index 1d02dd1..014fb91 100644 --- a/built/node_support/node_request_handler.js +++ b/built/node_support/node_request_handler.js @@ -114,7 +114,7 @@ var NodeBasedHandler = /** @class */ (function (_super) { request.setupCodeVerifier() .then(function () { server = Http.createServer(requestHandler); - server.listen(_this.httpServerPort, function () { + server.listen(_this.httpServerPort, '127.0.0.1', function () { var url = _this.buildRequestUrl(configuration, request); logger_1.log('Making a request to ', request, url); opener(url); @@ -145,4 +145,4 @@ var NodeBasedHandler = /** @class */ (function (_super) { return NodeBasedHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.NodeBasedHandler = NodeBasedHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, '127.0.0.1', () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file diff --git a/src/node_support/node_request_handler.ts b/src/node_support/node_request_handler.ts index b250a8d..4cc831e 100644 --- a/src/node_support/node_request_handler.ts +++ b/src/node_support/node_request_handler.ts @@ -108,7 +108,7 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { request.setupCodeVerifier() .then(() => { server = Http.createServer(requestHandler); - server.listen(this.httpServerPort, () => { + server.listen(this.httpServerPort, '127.0.0.1', () => { const url = this.buildRequestUrl(configuration, request); log('Making a request to ', request, url); opener(url); From eeade28ff2957c0537d4f9410907e0ee2159b2e2 Mon Sep 17 00:00:00 2001 From: Stefan Aebischer Date: Sat, 7 Dec 2019 09:32:12 -0500 Subject: [PATCH 4/5] Resolve with void instead of null --- built/authorization_request_handler.d.ts | 4 ++-- built/authorization_request_handler.js | 2 +- built/node_support/node_request_handler.d.ts | 2 +- built/node_support/node_request_handler.js | 4 ++-- built/redirect_based_handler.d.ts | 2 +- built/redirect_based_handler.js | 4 ++-- src/authorization_request_handler.ts | 4 ++-- src/node_support/node_request_handler.ts | 4 ++-- src/redirect_based_handler.ts | 4 ++-- 9 files changed, 15 insertions(+), 15 deletions(-) diff --git a/built/authorization_request_handler.d.ts b/built/authorization_request_handler.d.ts index 16e04e8..5518af2 100644 --- a/built/authorization_request_handler.d.ts +++ b/built/authorization_request_handler.d.ts @@ -52,9 +52,9 @@ export declare abstract class AuthorizationRequestHandler { setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler; /** * Makes an authorization request. - * Returns a `Promise`, when the request was sent succesfully. + * Returns a `Promise`, when the request was sent succesfully. */ - abstract performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; + abstract performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; /** * Checks if an authorization flow can be completed, and completes it. * The handler returns a `Promise` if ready, or a `Promise` diff --git a/built/authorization_request_handler.js b/built/authorization_request_handler.js index cfbb4fd..b830471 100644 --- a/built/authorization_request_handler.js +++ b/built/authorization_request_handler.js @@ -111,4 +111,4 @@ var AuthorizationRequestHandler = /** @class */ (function () { return AuthorizationRequestHandler; }()); exports.AuthorizationRequestHandler = AuthorizationRequestHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authorization_request_handler.js","sourceRoot":"","sources":["../src/authorization_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response';\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto} from './crypto_utils';\nimport {log} from './logger';\nimport {QueryStringUtils} from './query_string_utils';\nimport {StringMap} from './types';\n\n\n/**\n * This type represents a lambda that can take an AuthorizationRequest,\n * and an AuthorizationResponse as arguments.\n */\nexport type AuthorizationListener =\n    (request: AuthorizationRequest,\n     response: AuthorizationResponse|null,\n     error: AuthorizationError|null) => void;\n\n/**\n * Represents a structural type holding both authorization request and response.\n */\nexport interface AuthorizationRequestResponse {\n  request: AuthorizationRequest;\n  response: AuthorizationResponse|null;\n  error: AuthorizationError|null;\n}\n\n/**\n * Authorization Service notifier.\n * This manages the communication of the AuthorizationResponse to the 3p client.\n */\nexport class AuthorizationNotifier {\n  private listener: AuthorizationListener|null = null;\n\n  setAuthorizationListener(listener: AuthorizationListener) {\n    this.listener = listener;\n  }\n\n  /**\n   * The authorization complete callback.\n   */\n  onAuthorizationComplete(\n      request: AuthorizationRequest,\n      response: AuthorizationResponse|null,\n      error: AuthorizationError|null): void {\n    if (this.listener) {\n      // complete authorization request\n      this.listener(request, response, error);\n    }\n  }\n}\n\n// TODO(rahulrav@): add more built in parameters.\n/* built in parameters. */\nexport const BUILT_IN_PARAMETERS = ['redirect_uri', 'client_id', 'response_type', 'state', 'scope'];\n\n/**\n * Defines the interface which is capable of handling an authorization request\n * using various methods (iframe / popup / different process etc.).\n */\nexport abstract class AuthorizationRequestHandler {\n  constructor(public utils: QueryStringUtils, protected crypto: Crypto) {}\n\n  // notifier send the response back to the client.\n  protected notifier: AuthorizationNotifier|null = null;\n\n  /**\n   * A utility method to be able to build the authorization request URL.\n   */\n  protected buildRequestUrl(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // build the query string\n    // coerce to any type for convenience\n    let requestMap: StringMap = {\n      'redirect_uri': request.redirectUri,\n      'client_id': request.clientId,\n      'response_type': request.responseType,\n      'state': request.state,\n      'scope': request.scope\n    };\n\n    // copy over extras\n    if (request.extras) {\n      for (let extra in request.extras) {\n        if (request.extras.hasOwnProperty(extra)) {\n          // check before inserting to requestMap\n          if (BUILT_IN_PARAMETERS.indexOf(extra) < 0) {\n            requestMap[extra] = request.extras[extra];\n          }\n        }\n      }\n    }\n\n    let query = this.utils.stringify(requestMap);\n    let baseUrl = configuration.authorizationEndpoint;\n    let url = `${baseUrl}?${query}`;\n    return url;\n  }\n\n  /**\n   * Completes the authorization request if necessary & when possible.\n   */\n  completeAuthorizationRequestIfPossible(): Promise<void> {\n    // call complete authorization if possible to see there might\n    // be a response that needs to be delivered.\n    log(`Checking to see if there is an authorization response to be delivered.`);\n    if (!this.notifier) {\n      log(`Notifier is not present on AuthorizationRequest handler.\n          No delivery of result will be possible`)\n    }\n    return this.completeAuthorizationRequest().then(result => {\n      if (!result) {\n        log(`No result is available yet.`);\n      }\n      if (result && this.notifier) {\n        this.notifier.onAuthorizationComplete(result.request, result.response, result.error);\n      }\n    });\n  }\n\n  /**\n   * Sets the default Authorization Service notifier.\n   */\n  setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler {\n    this.notifier = notifier;\n    return this;\n  };\n\n  /**\n   * Makes an authorization request.\n   * Returns a `Promise<null>`, when the request was sent succesfully.\n   */\n  abstract performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest): Promise<null>;\n\n  /**\n   * Checks if an authorization flow can be completed, and completes it.\n   * The handler returns a `Promise<AuthorizationRequestResponse>` if ready, or a `Promise<null>`\n   * if not ready.\n   */\n  protected abstract completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null>;\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"authorization_request_handler.js","sourceRoot":"","sources":["../src/authorization_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response';\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto} from './crypto_utils';\nimport {log} from './logger';\nimport {QueryStringUtils} from './query_string_utils';\nimport {StringMap} from './types';\n\n\n/**\n * This type represents a lambda that can take an AuthorizationRequest,\n * and an AuthorizationResponse as arguments.\n */\nexport type AuthorizationListener =\n    (request: AuthorizationRequest,\n     response: AuthorizationResponse|null,\n     error: AuthorizationError|null) => void;\n\n/**\n * Represents a structural type holding both authorization request and response.\n */\nexport interface AuthorizationRequestResponse {\n  request: AuthorizationRequest;\n  response: AuthorizationResponse|null;\n  error: AuthorizationError|null;\n}\n\n/**\n * Authorization Service notifier.\n * This manages the communication of the AuthorizationResponse to the 3p client.\n */\nexport class AuthorizationNotifier {\n  private listener: AuthorizationListener|null = null;\n\n  setAuthorizationListener(listener: AuthorizationListener) {\n    this.listener = listener;\n  }\n\n  /**\n   * The authorization complete callback.\n   */\n  onAuthorizationComplete(\n      request: AuthorizationRequest,\n      response: AuthorizationResponse|null,\n      error: AuthorizationError|null): void {\n    if (this.listener) {\n      // complete authorization request\n      this.listener(request, response, error);\n    }\n  }\n}\n\n// TODO(rahulrav@): add more built in parameters.\n/* built in parameters. */\nexport const BUILT_IN_PARAMETERS = ['redirect_uri', 'client_id', 'response_type', 'state', 'scope'];\n\n/**\n * Defines the interface which is capable of handling an authorization request\n * using various methods (iframe / popup / different process etc.).\n */\nexport abstract class AuthorizationRequestHandler {\n  constructor(public utils: QueryStringUtils, protected crypto: Crypto) {}\n\n  // notifier send the response back to the client.\n  protected notifier: AuthorizationNotifier|null = null;\n\n  /**\n   * A utility method to be able to build the authorization request URL.\n   */\n  protected buildRequestUrl(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // build the query string\n    // coerce to any type for convenience\n    let requestMap: StringMap = {\n      'redirect_uri': request.redirectUri,\n      'client_id': request.clientId,\n      'response_type': request.responseType,\n      'state': request.state,\n      'scope': request.scope\n    };\n\n    // copy over extras\n    if (request.extras) {\n      for (let extra in request.extras) {\n        if (request.extras.hasOwnProperty(extra)) {\n          // check before inserting to requestMap\n          if (BUILT_IN_PARAMETERS.indexOf(extra) < 0) {\n            requestMap[extra] = request.extras[extra];\n          }\n        }\n      }\n    }\n\n    let query = this.utils.stringify(requestMap);\n    let baseUrl = configuration.authorizationEndpoint;\n    let url = `${baseUrl}?${query}`;\n    return url;\n  }\n\n  /**\n   * Completes the authorization request if necessary & when possible.\n   */\n  completeAuthorizationRequestIfPossible(): Promise<void> {\n    // call complete authorization if possible to see there might\n    // be a response that needs to be delivered.\n    log(`Checking to see if there is an authorization response to be delivered.`);\n    if (!this.notifier) {\n      log(`Notifier is not present on AuthorizationRequest handler.\n          No delivery of result will be possible`)\n    }\n    return this.completeAuthorizationRequest().then(result => {\n      if (!result) {\n        log(`No result is available yet.`);\n      }\n      if (result && this.notifier) {\n        this.notifier.onAuthorizationComplete(result.request, result.response, result.error);\n      }\n    });\n  }\n\n  /**\n   * Sets the default Authorization Service notifier.\n   */\n  setAuthorizationNotifier(notifier: AuthorizationNotifier): AuthorizationRequestHandler {\n    this.notifier = notifier;\n    return this;\n  };\n\n  /**\n   * Makes an authorization request.\n   * Returns a `Promise<void>`, when the request was sent succesfully.\n   */\n  abstract performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest): Promise<void>;\n\n  /**\n   * Checks if an authorization flow can be completed, and completes it.\n   * The handler returns a `Promise<AuthorizationRequestResponse>` if ready, or a `Promise<null>`\n   * if not ready.\n   */\n  protected abstract completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null>;\n}\n"]} \ No newline at end of file diff --git a/built/node_support/node_request_handler.d.ts b/built/node_support/node_request_handler.d.ts index 443763b..4f93b97 100644 --- a/built/node_support/node_request_handler.d.ts +++ b/built/node_support/node_request_handler.d.ts @@ -7,6 +7,6 @@ export declare class NodeBasedHandler extends AuthorizationRequestHandler { httpServerPort: number; authorizationPromise: Promise | null; constructor(httpServerPort?: number, utils?: QueryStringUtils, crypto?: Crypto); - performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; + performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; protected completeAuthorizationRequest(): Promise; } diff --git a/built/node_support/node_request_handler.js b/built/node_support/node_request_handler.js index 014fb91..afa48d7 100644 --- a/built/node_support/node_request_handler.js +++ b/built/node_support/node_request_handler.js @@ -133,7 +133,7 @@ var NodeBasedHandler = /** @class */ (function (_super) { emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function (error) { reject(new errors_1.AppAuthError("Unable to create HTTP server at port " + _this.httpServerPort, { origError: error })); }); - emitter.once(ServerEventsEmitter.ON_START, function () { return resolve(null); }); + emitter.once(ServerEventsEmitter.ON_START, function () { return resolve(); }); }); }; NodeBasedHandler.prototype.completeAuthorizationRequest = function () { @@ -145,4 +145,4 @@ var NodeBasedHandler = /** @class */ (function (_super) { return NodeBasedHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.NodeBasedHandler = NodeBasedHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, '127.0.0.1', () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve(null));\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, '127.0.0.1', () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<void>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve());\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file diff --git a/built/redirect_based_handler.d.ts b/built/redirect_based_handler.d.ts index 6b5b698..c683538 100644 --- a/built/redirect_based_handler.d.ts +++ b/built/redirect_based_handler.d.ts @@ -13,7 +13,7 @@ export declare class RedirectRequestHandler extends AuthorizationRequestHandler storageBackend: StorageBackend; locationLike: LocationLike; constructor(storageBackend?: StorageBackend, utils?: BasicQueryStringUtils, locationLike?: LocationLike, crypto?: Crypto); - performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; + performAuthorizationRequest(configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest): Promise; /** * Attempts to introspect the contents of storage backend and completes the * request. diff --git a/built/redirect_based_handler.js b/built/redirect_based_handler.js index 92c23be..7239d37 100644 --- a/built/redirect_based_handler.js +++ b/built/redirect_based_handler.js @@ -80,7 +80,7 @@ var RedirectRequestHandler = /** @class */ (function (_super) { var url = _this.buildRequestUrl(configuration, request); logger_1.log('Making a request to ', request, url); _this.locationLike.assign(url); - resolve(null); + resolve(); }) .catch(function (error) { return reject(error); }); }); @@ -158,4 +158,4 @@ var RedirectRequestHandler = /** @class */ (function (_super) { return RedirectRequestHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.RedirectRequestHandler = RedirectRequestHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redirect_based_handler.js","sourceRoot":"","sources":["../src/redirect_based_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from './authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response'\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto, DefaultCrypto} from './crypto_utils';\nimport {log} from './logger';\nimport {BasicQueryStringUtils} from './query_string_utils';\nimport {LocalStorageBackend, StorageBackend} from './storage';\nimport {LocationLike} from './types';\n\n\n/** key for authorization request. */\nconst authorizationRequestKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_request`;\n    }\n\n/** key for authorization service configuration */\nconst authorizationServiceConfigurationKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_service_configuration`;\n    }\n\n/** key in local storage which represents the current authorization request. */\nconst AUTHORIZATION_REQUEST_HANDLE_KEY = 'appauth_current_authorization_request';\n\n/**\n * Represents an AuthorizationRequestHandler which uses a standard\n * redirect based code flow.\n */\nexport class RedirectRequestHandler extends AuthorizationRequestHandler {\n  constructor(\n      // use the provided storage backend\n      // or initialize local storage with the default storage backend which\n      // uses window.localStorage\n      public storageBackend: StorageBackend = new LocalStorageBackend(),\n      utils = new BasicQueryStringUtils(),\n      public locationLike: LocationLike = window.location,\n      crypto: Crypto = new DefaultCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    return new Promise<null>((resolve, reject) => {\n      const handle = this.crypto.generateRandom(10);\n\n      // before you make request, persist all request related data in local storage.\n      const persisted = Promise.all([\n        this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle),\n        // Calling toJson() adds in the code & challenge when possible\n        request.toJson().then(\n            result => this.storageBackend.setItem(\n                authorizationRequestKey(handle), JSON.stringify(result))),\n        this.storageBackend.setItem(\n            authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())),\n      ]);\n\n      persisted\n          .then(() => {\n            // make the redirect request\n            let url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            this.locationLike.assign(url);\n            resolve(null);\n          })\n          .catch(error => reject(error));\n    });\n  }\n\n  /**\n   * Attempts to introspect the contents of storage backend and completes the\n   * request.\n   */\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    // TODO(rahulrav@): handle authorization errors.\n    return this.storageBackend.getItem(AUTHORIZATION_REQUEST_HANDLE_KEY).then(handle => {\n      if (handle) {\n        // we have a pending request.\n        // fetch authorization request, and check state\n        return this.storageBackend\n            .getItem(authorizationRequestKey(handle))\n            // requires a corresponding instance of result\n            // TODO(rahulrav@): check for inconsitent state here\n            .then(result => JSON.parse(result!))\n            .then(json => new AuthorizationRequest(json))\n            .then(request => {\n              // check redirect_uri and state\n              let currentUri = `${this.locationLike.origin}${this.locationLike.pathname}`;\n              let queryParams = this.utils.parse(this.locationLike, true /* use hash */);\n              let state: string|undefined = queryParams['state'];\n              let code: string|undefined = queryParams['code'];\n              let error: string|undefined = queryParams['error'];\n              log('Potential authorization request ', currentUri, queryParams, state, code, error);\n              let shouldNotify = state === request.state;\n              let authorizationResponse: AuthorizationResponse|null = null;\n              let authorizationError: AuthorizationError|null = null;\n              if (shouldNotify) {\n                if (error) {\n                  // get additional optional info.\n                  let errorUri = queryParams['error_uri'];\n                  let errorDescription = queryParams['error_description'];\n                  authorizationError = new AuthorizationError({\n                    error: error,\n                    error_description: errorDescription,\n                    error_uri: errorUri,\n                    state: state\n                  });\n                } else {\n                  authorizationResponse = new AuthorizationResponse({code: code, state: state});\n                }\n                // cleanup state\n                return Promise\n                    .all([\n                      this.storageBackend.removeItem(AUTHORIZATION_REQUEST_HANDLE_KEY),\n                      this.storageBackend.removeItem(authorizationRequestKey(handle)),\n                      this.storageBackend.removeItem(authorizationServiceConfigurationKey(handle))\n                    ])\n                    .then(() => {\n                      log('Delivering authorization response');\n                      return {\n                        request: request,\n                        response: authorizationResponse,\n                        error: authorizationError\n                      } as AuthorizationRequestResponse;\n                    });\n              } else {\n                log('Mismatched request (state and request_uri) dont match.');\n                return Promise.resolve(null);\n              }\n            });\n      } else {\n        return null;\n      }\n    });\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redirect_based_handler.js","sourceRoot":"","sources":["../src/redirect_based_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {AuthorizationRequest} from './authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from './authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from './authorization_response'\nimport {AuthorizationServiceConfiguration} from './authorization_service_configuration';\nimport {Crypto, DefaultCrypto} from './crypto_utils';\nimport {log} from './logger';\nimport {BasicQueryStringUtils} from './query_string_utils';\nimport {LocalStorageBackend, StorageBackend} from './storage';\nimport {LocationLike} from './types';\n\n\n/** key for authorization request. */\nconst authorizationRequestKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_request`;\n    }\n\n/** key for authorization service configuration */\nconst authorizationServiceConfigurationKey =\n    (handle: string) => {\n      return `${handle}_appauth_authorization_service_configuration`;\n    }\n\n/** key in local storage which represents the current authorization request. */\nconst AUTHORIZATION_REQUEST_HANDLE_KEY = 'appauth_current_authorization_request';\n\n/**\n * Represents an AuthorizationRequestHandler which uses a standard\n * redirect based code flow.\n */\nexport class RedirectRequestHandler extends AuthorizationRequestHandler {\n  constructor(\n      // use the provided storage backend\n      // or initialize local storage with the default storage backend which\n      // uses window.localStorage\n      public storageBackend: StorageBackend = new LocalStorageBackend(),\n      utils = new BasicQueryStringUtils(),\n      public locationLike: LocationLike = window.location,\n      crypto: Crypto = new DefaultCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    return new Promise<void>((resolve, reject) => {\n      const handle = this.crypto.generateRandom(10);\n\n      // before you make request, persist all request related data in local storage.\n      const persisted = Promise.all([\n        this.storageBackend.setItem(AUTHORIZATION_REQUEST_HANDLE_KEY, handle),\n        // Calling toJson() adds in the code & challenge when possible\n        request.toJson().then(\n            result => this.storageBackend.setItem(\n                authorizationRequestKey(handle), JSON.stringify(result))),\n        this.storageBackend.setItem(\n            authorizationServiceConfigurationKey(handle), JSON.stringify(configuration.toJson())),\n      ]);\n\n      persisted\n          .then(() => {\n            // make the redirect request\n            let url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            this.locationLike.assign(url);\n            resolve();\n          })\n          .catch(error => reject(error));\n    });\n  }\n\n  /**\n   * Attempts to introspect the contents of storage backend and completes the\n   * request.\n   */\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    // TODO(rahulrav@): handle authorization errors.\n    return this.storageBackend.getItem(AUTHORIZATION_REQUEST_HANDLE_KEY).then(handle => {\n      if (handle) {\n        // we have a pending request.\n        // fetch authorization request, and check state\n        return this.storageBackend\n            .getItem(authorizationRequestKey(handle))\n            // requires a corresponding instance of result\n            // TODO(rahulrav@): check for inconsitent state here\n            .then(result => JSON.parse(result!))\n            .then(json => new AuthorizationRequest(json))\n            .then(request => {\n              // check redirect_uri and state\n              let currentUri = `${this.locationLike.origin}${this.locationLike.pathname}`;\n              let queryParams = this.utils.parse(this.locationLike, true /* use hash */);\n              let state: string|undefined = queryParams['state'];\n              let code: string|undefined = queryParams['code'];\n              let error: string|undefined = queryParams['error'];\n              log('Potential authorization request ', currentUri, queryParams, state, code, error);\n              let shouldNotify = state === request.state;\n              let authorizationResponse: AuthorizationResponse|null = null;\n              let authorizationError: AuthorizationError|null = null;\n              if (shouldNotify) {\n                if (error) {\n                  // get additional optional info.\n                  let errorUri = queryParams['error_uri'];\n                  let errorDescription = queryParams['error_description'];\n                  authorizationError = new AuthorizationError({\n                    error: error,\n                    error_description: errorDescription,\n                    error_uri: errorUri,\n                    state: state\n                  });\n                } else {\n                  authorizationResponse = new AuthorizationResponse({code: code, state: state});\n                }\n                // cleanup state\n                return Promise\n                    .all([\n                      this.storageBackend.removeItem(AUTHORIZATION_REQUEST_HANDLE_KEY),\n                      this.storageBackend.removeItem(authorizationRequestKey(handle)),\n                      this.storageBackend.removeItem(authorizationServiceConfigurationKey(handle))\n                    ])\n                    .then(() => {\n                      log('Delivering authorization response');\n                      return {\n                        request: request,\n                        response: authorizationResponse,\n                        error: authorizationError\n                      } as AuthorizationRequestResponse;\n                    });\n              } else {\n                log('Mismatched request (state and request_uri) dont match.');\n                return Promise.resolve(null);\n              }\n            });\n      } else {\n        return null;\n      }\n    });\n  }\n}\n"]} \ No newline at end of file diff --git a/src/authorization_request_handler.ts b/src/authorization_request_handler.ts index 955418d..e1cdfbf 100644 --- a/src/authorization_request_handler.ts +++ b/src/authorization_request_handler.ts @@ -143,11 +143,11 @@ export abstract class AuthorizationRequestHandler { /** * Makes an authorization request. - * Returns a `Promise`, when the request was sent succesfully. + * Returns a `Promise`, when the request was sent succesfully. */ abstract performAuthorizationRequest( configuration: AuthorizationServiceConfiguration, - request: AuthorizationRequest): Promise; + request: AuthorizationRequest): Promise; /** * Checks if an authorization flow can be completed, and completes it. diff --git a/src/node_support/node_request_handler.ts b/src/node_support/node_request_handler.ts index 4cc831e..1cb841c 100644 --- a/src/node_support/node_request_handler.ts +++ b/src/node_support/node_request_handler.ts @@ -125,7 +125,7 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); }); - return new Promise((resolve, reject) => { + return new Promise((resolve, reject) => { emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => { reject(new AppAuthError( `Unable to create HTTP server at port ${this.httpServerPort}`, @@ -133,7 +133,7 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { )); }); - emitter.once(ServerEventsEmitter.ON_START, () => resolve(null)); + emitter.once(ServerEventsEmitter.ON_START, () => resolve()); }); } diff --git a/src/redirect_based_handler.ts b/src/redirect_based_handler.ts index 259c684..3c2db4e 100644 --- a/src/redirect_based_handler.ts +++ b/src/redirect_based_handler.ts @@ -57,7 +57,7 @@ export class RedirectRequestHandler extends AuthorizationRequestHandler { performAuthorizationRequest( configuration: AuthorizationServiceConfiguration, request: AuthorizationRequest) { - return new Promise((resolve, reject) => { + return new Promise((resolve, reject) => { const handle = this.crypto.generateRandom(10); // before you make request, persist all request related data in local storage. @@ -77,7 +77,7 @@ export class RedirectRequestHandler extends AuthorizationRequestHandler { let url = this.buildRequestUrl(configuration, request); log('Making a request to ', request, url); this.locationLike.assign(url); - resolve(null); + resolve(); }) .catch(error => reject(error)); }); From 67e115d242557baba53881b728619de3b60a844e Mon Sep 17 00:00:00 2001 From: Stefan Aebischer Date: Sat, 14 Dec 2019 18:12:26 -0500 Subject: [PATCH 5/5] Reject authorizationPromise on startup error Fixes: https://github.com/openid/AppAuth-JS/pull/133#discussion_r355074388 --- built/node_support/node_request_handler.js | 16 ++++++++++------ src/node_support/node_request_handler.ts | 15 ++++++++++----- 2 files changed, 20 insertions(+), 11 deletions(-) diff --git a/built/node_support/node_request_handler.js b/built/node_support/node_request_handler.js index afa48d7..d772b90 100644 --- a/built/node_support/node_request_handler.js +++ b/built/node_support/node_request_handler.js @@ -100,16 +100,22 @@ var NodeBasedHandler = /** @class */ (function (_super) { emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse); response.end('Close your browser to continue'); }; - this.authorizationPromise = new Promise(function (resolve) { - emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function () { return resolve(null); }); + this.authorizationPromise = new Promise(function (resolve, reject) { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, function (error) { return reject(error); }); emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, function (result) { server.close(); // resolve pending promise resolve(result); // complete authorization flow - _this.completeAuthorizationRequestIfPossible(); + _this.completeAuthorizationRequestIfPossible() + .catch(function (error) { + logger_1.log('Could not complete authorization request', error); + }); }); }); + this.authorizationPromise.catch(function (error) { + logger_1.log('Something bad happened ', error); + }); var server; request.setupCodeVerifier() .then(function () { @@ -121,12 +127,10 @@ var NodeBasedHandler = /** @class */ (function (_super) { emitter.emit(ServerEventsEmitter.ON_START); }); server.on('error', function (error) { - logger_1.log('Something bad happened ', error); emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); }); }) .catch(function (error) { - logger_1.log('Something bad happened ', error); emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); }); return new Promise(function (resolve, reject) { @@ -145,4 +149,4 @@ var NodeBasedHandler = /** @class */ (function (_super) { return NodeBasedHandler; }(authorization_request_handler_1.AuthorizationRequestHandler)); exports.NodeBasedHandler = NodeBasedHandler; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible();\n      });\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, '127.0.0.1', () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            log('Something bad happened ', error);\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          log('Something bad happened ', error);\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<void>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve());\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node_request_handler.js","sourceRoot":"","sources":["../../src/node_support/node_request_handler.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;GAYG;;;;;;;;;;;;;;;AAEH,qCAAuC;AACvC,2BAA6B;AAC7B,yBAA2B;AAC3B,oCAAuC;AAEvC,kFAA2G;AAC3G,oEAAoF;AAGpF,oCAA8B;AAC9B,4DAA8E;AAC9E,+CAA0C;AAG1C,iFAAiF;AACjF,+BAAkC;AAElC;IAAkC,uCAAY;IAA9C;;IAIA,CAAC;IAHQ,4BAAQ,GAAG,OAAO,CAAC;IACnB,sCAAkB,GAAG,iBAAiB,CAAC;IACvC,6CAAyB,GAAG,wBAAwB,CAAC;IAC9D,0BAAC;CAAA,AAJD,CAAkC,YAAY,GAI7C;AAED;IAAsC,oCAA2B;IAI/D;IACI,uBAAuB;IAChB,cAAqB,EAC5B,KAAqD,EACrD,MAAiC;QAF1B,+BAAA,EAAA,qBAAqB;QAC5B,sBAAA,EAAA,YAA8B,0CAAqB,EAAE;QACrD,uBAAA,EAAA,aAAqB,yBAAU,EAAE;QAJrC,YAKE,kBAAM,KAAK,EAAE,MAAM,CAAC,SACrB;QAJU,oBAAc,GAAd,cAAc,CAAO;QALhC,kDAAkD;QAClD,0BAAoB,GAAoD,IAAI,CAAC;;IAQ7E,CAAC;IAED,sDAA2B,GAA3B,UACI,aAAgD,EAChD,OAA6B;QAFjC,iBA6FC;QA1FC,uEAAuE;QACvE,2DAA2D;QAC3D,IAAM,OAAO,GAAG,IAAI,mBAAmB,EAAE,CAAC;QAE1C,IAAM,cAAc,GAAG,UAAC,WAAiC,EAAE,QAA6B;YACtF,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE;gBACpB,OAAO;aACR;YAED,IAAM,GAAG,GAAG,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;YACvC,IAAM,YAAY,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;YAE9D,IAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,SAAS,CAAC;YACrD,IAAM,IAAI,GAAG,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACtC,IAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAExC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE;gBAC7B,gDAAgD;gBAChD,OAAO;aACR;YAED,YAAG,CAAC,iCAAiC,EAAE,YAAY,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC;YACzE,IAAI,qBAAqB,GAA+B,IAAI,CAAC;YAC7D,IAAI,kBAAkB,GAA4B,IAAI,CAAC;YACvD,IAAI,KAAK,EAAE;gBACT,YAAG,CAAC,OAAO,CAAC,CAAC;gBACb,gCAAgC;gBAChC,IAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,WAAW,CAAC,IAAI,SAAS,CAAC;gBAC5D,IAAM,gBAAgB,GAAG,YAAY,CAAC,GAAG,CAAC,mBAAmB,CAAC,IAAI,SAAS,CAAC;gBAC5E,kBAAkB,GAAG,IAAI,2CAAkB,CACvC,EAAC,KAAK,EAAE,KAAK,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,SAAS,EAAE,QAAQ,EAAE,KAAK,EAAE,KAAK,EAAC,CAAC,CAAC;aAC7F;iBAAM;gBACL,qBAAqB,GAAG,IAAI,8CAAqB,CAAC,EAAC,IAAI,EAAE,IAAK,EAAE,KAAK,EAAE,KAAM,EAAC,CAAC,CAAC;aACjF;YACD,IAAM,gBAAgB,GAAG;gBACvB,OAAO,SAAA;gBACP,QAAQ,EAAE,qBAAqB;gBAC/B,KAAK,EAAE,kBAAkB;aACM,CAAC;YAClC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,yBAAyB,EAAE,gBAAgB,CAAC,CAAC;YAC9E,QAAQ,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;QACjD,CAAC,CAAC;QAEF,IAAI,CAAC,oBAAoB,GAAG,IAAI,OAAO,CAAoC,UAAC,OAAO,EAAE,MAAM;YACzF,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,UAAC,KAAK,IAAK,OAAA,MAAM,CAAC,KAAK,CAAC,EAAb,CAAa,CAAC,CAAC;YAC/E,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,yBAAyB,EAAE,UAAC,MAAW;gBACtE,MAAM,CAAC,KAAK,EAAE,CAAC;gBACf,0BAA0B;gBAC1B,OAAO,CAAC,MAAsC,CAAC,CAAC;gBAChD,8BAA8B;gBAC9B,KAAI,CAAC,sCAAsC,EAAE;qBAC1C,KAAK,CAAC,UAAA,KAAK;oBACV,YAAG,CAAC,0CAA0C,EAAE,KAAK,CAAC,CAAC;gBACzD,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,UAAA,KAAK;YACnC,YAAG,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,IAAI,MAAmB,CAAC;QACxB,OAAO,CAAC,iBAAiB,EAAE;aACtB,IAAI,CAAC;YACJ,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,CAAC,MAAM,CAAC,KAAI,CAAC,cAAc,EAAE,WAAW,EAAE;gBAC9C,IAAM,GAAG,GAAG,KAAI,CAAC,eAAe,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;gBACzD,YAAG,CAAC,sBAAsB,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC;gBAC1C,MAAM,CAAC,GAAG,CAAC,CAAC;gBACZ,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;YAEH,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,UAAC,KAAY;gBAC9B,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,KAAK,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;aACD,KAAK,CAAC,UAAC,KAAK;YACX,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,KAAK,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEP,OAAO,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;YACvC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,UAAC,KAAK;gBACzD,MAAM,CAAC,IAAI,qBAAY,CACrB,0CAAwC,KAAI,CAAC,cAAgB,EAC7D,EAAC,SAAS,EAAE,KAAK,EAAC,CACnB,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,EAAE,cAAM,OAAA,OAAO,EAAE,EAAT,CAAS,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;IACL,CAAC;IAES,uDAA4B,GAAtC;QACE,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE;YAC9B,OAAO,OAAO,CAAC,MAAM,CACjB,wEAAwE,CAAC,CAAC;SAC/E;QAED,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IACH,uBAAC;AAAD,CAAC,AAnHD,CAAsC,2DAA2B,GAmHhE;AAnHY,4CAAgB","sourcesContent":["/*\n * Copyright 2017 Google Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use this file except\n * in compliance with the License. You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under the\n * License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either\n * express or implied. See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as EventEmitter from 'events';\nimport * as Http from 'http';\nimport * as Url from 'url';\nimport {AppAuthError} from '../errors';\nimport {AuthorizationRequest} from '../authorization_request';\nimport {AuthorizationRequestHandler, AuthorizationRequestResponse} from '../authorization_request_handler';\nimport {AuthorizationError, AuthorizationResponse} from '../authorization_response';\nimport {AuthorizationServiceConfiguration} from '../authorization_service_configuration';\nimport {Crypto} from '../crypto_utils';\nimport {log} from '../logger';\nimport {BasicQueryStringUtils, QueryStringUtils} from '../query_string_utils';\nimport {NodeCrypto} from './crypto_utils';\n\n\n// TypeScript typings for `opener` are not correct and do not export it as module\nimport opener = require('opener');\n\nclass ServerEventsEmitter extends EventEmitter {\n  static ON_START = 'start';\n  static ON_UNABLE_TO_START = 'unable_to_start';\n  static ON_AUTHORIZATION_RESPONSE = 'authorization_response';\n}\n\nexport class NodeBasedHandler extends AuthorizationRequestHandler {\n  // the handle to the current authorization request\n  authorizationPromise: Promise<AuthorizationRequestResponse|null>|null = null;\n\n  constructor(\n      // default to port 8000\n      public httpServerPort = 8000,\n      utils: QueryStringUtils = new BasicQueryStringUtils(),\n      crypto: Crypto = new NodeCrypto()) {\n    super(utils, crypto);\n  }\n\n  performAuthorizationRequest(\n      configuration: AuthorizationServiceConfiguration,\n      request: AuthorizationRequest) {\n    // use opener to launch a web browser and start the authorization flow.\n    // start a web server to handle the authorization response.\n    const emitter = new ServerEventsEmitter();\n\n    const requestHandler = (httpRequest: Http.IncomingMessage, response: Http.ServerResponse) => {\n      if (!httpRequest.url) {\n        return;\n      }\n\n      const url = Url.parse(httpRequest.url);\n      const searchParams = new Url.URLSearchParams(url.query || '');\n\n      const state = searchParams.get('state') || undefined;\n      const code = searchParams.get('code');\n      const error = searchParams.get('error');\n\n      if (!state && !code && !error) {\n        // ignore irrelevant requests (e.g. favicon.ico)\n        return;\n      }\n\n      log('Handling Authorization Request ', searchParams, state, code, error);\n      let authorizationResponse: AuthorizationResponse|null = null;\n      let authorizationError: AuthorizationError|null = null;\n      if (error) {\n        log('error');\n        // get additional optional info.\n        const errorUri = searchParams.get('error_uri') || undefined;\n        const errorDescription = searchParams.get('error_description') || undefined;\n        authorizationError = new AuthorizationError(\n            {error: error, error_description: errorDescription, error_uri: errorUri, state: state});\n      } else {\n        authorizationResponse = new AuthorizationResponse({code: code!, state: state!});\n      }\n      const completeResponse = {\n        request,\n        response: authorizationResponse,\n        error: authorizationError\n      } as AuthorizationRequestResponse;\n      emitter.emit(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, completeResponse);\n      response.end('Close your browser to continue');\n    };\n\n    this.authorizationPromise = new Promise<AuthorizationRequestResponse|null>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => reject(error));\n      emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => {\n        server.close();\n        // resolve pending promise\n        resolve(result as AuthorizationRequestResponse);\n        // complete authorization flow\n        this.completeAuthorizationRequestIfPossible()\n          .catch(error => {\n            log('Could not complete authorization request', error);\n          });\n      });\n    });\n\n    this.authorizationPromise.catch(error => {\n      log('Something bad happened ', error);\n    });\n\n    let server: Http.Server;\n    request.setupCodeVerifier()\n        .then(() => {\n          server = Http.createServer(requestHandler);\n          server.listen(this.httpServerPort, '127.0.0.1', () => {\n            const url = this.buildRequestUrl(configuration, request);\n            log('Making a request to ', request, url);\n            opener(url);\n            emitter.emit(ServerEventsEmitter.ON_START);\n          });\n\n          server.on('error', (error: Error) => {\n            emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n          });\n        })\n        .catch((error) => {\n          emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error);\n        });\n\n    return new Promise<void>((resolve, reject) => {\n      emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => {\n        reject(new AppAuthError(\n          `Unable to create HTTP server at port ${this.httpServerPort}`,\n          {origError: error}\n        ));\n      });\n\n      emitter.once(ServerEventsEmitter.ON_START, () => resolve());\n    });\n  }\n\n  protected completeAuthorizationRequest(): Promise<AuthorizationRequestResponse|null> {\n    if (!this.authorizationPromise) {\n      return Promise.reject(\n          'No pending authorization request. Call performAuthorizationRequest() ?');\n    }\n\n    return this.authorizationPromise;\n  }\n}\n"]} \ No newline at end of file diff --git a/src/node_support/node_request_handler.ts b/src/node_support/node_request_handler.ts index 1cb841c..a77ac59 100644 --- a/src/node_support/node_request_handler.ts +++ b/src/node_support/node_request_handler.ts @@ -93,17 +93,24 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { response.end('Close your browser to continue'); }; - this.authorizationPromise = new Promise((resolve) => { - emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, () => resolve(null)); + this.authorizationPromise = new Promise((resolve, reject) => { + emitter.once(ServerEventsEmitter.ON_UNABLE_TO_START, (error) => reject(error)); emitter.once(ServerEventsEmitter.ON_AUTHORIZATION_RESPONSE, (result: any) => { server.close(); // resolve pending promise resolve(result as AuthorizationRequestResponse); // complete authorization flow - this.completeAuthorizationRequestIfPossible(); + this.completeAuthorizationRequestIfPossible() + .catch(error => { + log('Could not complete authorization request', error); + }); }); }); + this.authorizationPromise.catch(error => { + log('Something bad happened ', error); + }); + let server: Http.Server; request.setupCodeVerifier() .then(() => { @@ -116,12 +123,10 @@ export class NodeBasedHandler extends AuthorizationRequestHandler { }); server.on('error', (error: Error) => { - log('Something bad happened ', error); emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); }); }) .catch((error) => { - log('Something bad happened ', error); emitter.emit(ServerEventsEmitter.ON_UNABLE_TO_START, error); });