├── .gitignore ├── Drivers ├── Mixer.js ├── Patreon.js ├── Twitch.js └── Vk.js ├── LICENSE ├── README.md ├── ServiceProvider.js ├── instructions.md ├── package-lock.json └── package.json /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (http://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # Typescript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | -------------------------------------------------------------------------------- /Drivers/Mixer.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /* 4 | * adonis-ally Mixer driver 5 | * 6 | * (c) Vladyslav Gaysyuk 7 | * 8 | * For the full copyright and license information, please view the LICENSE 9 | * file that was distributed with this source code. 10 | */ 11 | 12 | const CE = require('@adonisjs/ally/src/Exceptions') 13 | const OAuth2Scheme = require('@adonisjs/ally/src/Schemes/OAuth2') 14 | const AllyUser = require('@adonisjs/ally/src/AllyUser') 15 | const got = require('got') 16 | const utils = require('@adonisjs/ally/lib/utils') 17 | const _ = require('lodash') 18 | 19 | class Mixer extends OAuth2Scheme { 20 | 21 | constructor (Config) { 22 | const config = Config.get('services.ally.mixer') 23 | 24 | utils.validateDriverConfig('mixer', config, ['clientId', 'clientSecret', 'redirectUri']) 25 | 26 | super(config.clientId, config.clientSecret, config.headers) 27 | 28 | /** 29 | * Oauth specific values to be used when creating the redirect 30 | * url or fetching user profile. 31 | */ 32 | this._scope = this._getInitialScopes(config.scope) 33 | this._redirectUri = config.redirectUri 34 | this.config = config 35 | this._redirectUriOptions = _.merge({response_type: 'code'}, config.options) 36 | } 37 | 38 | /** 39 | * Injections to be made by the IoC container 40 | * 41 | * @return {Array} 42 | */ 43 | static get inject () { 44 | return ['Adonis/Src/Config'] 45 | } 46 | 47 | /** 48 | * Scope seperator for seperating multiple 49 | * scopes. 50 | * 51 | * @return {String} 52 | */ 53 | get scopeSeperator () { 54 | return ' ' 55 | } 56 | 57 | /** 58 | * Base url to be used for constructing 59 | * mixer oauth urls. 60 | * 61 | * @return {String} 62 | */ 63 | get baseUrl () { 64 | return 'https://mixer.com' 65 | } 66 | 67 | /** 68 | * Relative url to be used for redirecting 69 | * user. 70 | * 71 | * @return {String} [description] 72 | */ 73 | get authorizeUrl () { 74 | return 'oauth/authorize' 75 | } 76 | 77 | /** 78 | * Relative url to be used for exchanging 79 | * access token. 80 | * 81 | * @return {String} 82 | */ 83 | get accessTokenUrl () { 84 | return `api/v1/oauth/token` 85 | } 86 | 87 | /** 88 | * API url to be used for getting VKontakte user's profile 89 | * 90 | * @return {String} 91 | */ 92 | get apiUrl () { 93 | return 'https://mixer.com/api/v1' 94 | } 95 | 96 | /** 97 | * Returns initial scopes to be used right from the 98 | * config file. Otherwise it will fallback to the 99 | * commonly used scopes 100 | * 101 | * @param {Array} scopes 102 | * 103 | * @return {Array} 104 | * 105 | * @private 106 | */ 107 | _getInitialScopes (scopes) { 108 | return _.size(scopes) ? scopes : ["user:details:self"] 109 | } 110 | 111 | /** 112 | * Returns the user profile as an object using the 113 | * access token 114 | * 115 | * @param {String} accessToken 116 | * @param {Array} [fields] 117 | * 118 | * @return {Object} 119 | * 120 | * @private 121 | */ 122 | async _getUserProfile (accessToken, fields) { 123 | const response = await got(`${this.apiUrl}/users/current`, { 124 | headers: { 125 | 'Authorization': accessToken?'Bearer ' + accessToken : undefined 126 | }, 127 | json: true 128 | }) 129 | return response.body 130 | } 131 | 132 | /** 133 | * Returns the redirect url for a given provider. 134 | * 135 | * @param {Array} scope 136 | * 137 | * @return {String} 138 | */ 139 | async getRedirectUrl (scope) { 140 | scope = _.size(scope) ? scope : this._scope 141 | return this.getUrl(this._redirectUri, scope, this._redirectUriOptions) 142 | } 143 | 144 | /** 145 | * Parses provider error by fetching error message 146 | * from nested data property. 147 | * 148 | * @param {Object} error 149 | * 150 | * @return {Error} 151 | */ 152 | parseProviderError (error) { 153 | console.log(error) 154 | const parsedError = _.isString(error.data) ? JSON.parse(error.data) : null 155 | const message = _.get(parsedError, 'message', error) 156 | return CE.OAuthException.tokenExchangeException(message, error.statusCode, parsedError) 157 | } 158 | 159 | /** 160 | * Parses the redirect errors returned by mixer 161 | * and returns the error message. 162 | * 163 | * @param {Object} queryParams 164 | * 165 | * @return {String} 166 | */ 167 | parseRedirectError (queryParams) { 168 | return queryParams.error_message || 'Oauth failed during redirect' 169 | } 170 | 171 | /** 172 | * Returns the user profile with it's access token, refresh token 173 | * and token expiry 174 | * 175 | * @param {Object} queryParams 176 | * @param {Array} [fields] 177 | * 178 | * @return {Object} 179 | */ 180 | async getUser (queryParams, fields) { 181 | const code = queryParams.code 182 | /** 183 | * Throw an exception when query string does not have 184 | * code. 185 | */ 186 | if (!code) { 187 | const errorMessage = this.parseRedirectError(queryParams) 188 | throw CE.OAuthException.tokenExchangeException(errorMessage, null, errorMessage) 189 | } 190 | const accessTokenResponse = await this.getAccessToken(code, this._redirectUri, { 191 | grant_type: 'authorization_code' 192 | }) 193 | const userProfile = await this._getUserProfile(accessTokenResponse.accessToken, fields) 194 | 195 | const user = new AllyUser() 196 | user 197 | .setOriginal(userProfile) 198 | .setFields( 199 | userProfile.id, 200 | userProfile.username, 201 | userProfile.email, 202 | null, 203 | userProfile.avatarUrl 204 | ) 205 | .setToken( 206 | accessTokenResponse.accessToken, 207 | accessTokenResponse.refreshToken, 208 | null, 209 | Number(accessTokenResponse.result.expires_in) 210 | ) 211 | 212 | return user 213 | } 214 | } 215 | 216 | module.exports = Mixer 217 | -------------------------------------------------------------------------------- /Drivers/Patreon.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /* 4 | * adonis-ally Patreon driver 5 | * 6 | * (c) Jonas Fleur-Aime 7 | * 8 | * For the full copyright and license information, please view the LICENSE 9 | * file that was distributed with this source code. 10 | */ 11 | 12 | const CE = require('@adonisjs/ally/src/Exceptions') 13 | const OAuth2Scheme = require('@adonisjs/ally/src/Schemes/OAuth2') 14 | const AllyUser = require('@adonisjs/ally/src/AllyUser') 15 | const got = require('got') 16 | const utils = require('@adonisjs/ally/lib/utils') 17 | const _ = require('lodash') 18 | 19 | class Patreon extends OAuth2Scheme { 20 | 21 | constructor (Config) { 22 | const config = Config.get('services.ally.patreon') 23 | 24 | utils.validateDriverConfig('patreon', config, ['clientId', 'clientSecret', 'redirectUri']) 25 | 26 | super(config.clientId, config.clientSecret, config.headers) 27 | 28 | /** 29 | * Oauth specific values to be used when creating the redirect 30 | * url or fetching user profile. 31 | */ 32 | this._scope = this._getInitialScopes(config.scope) 33 | this._redirectUri = config.redirectUri 34 | this.config = config 35 | this._redirectUriOptions = _.merge({response_type: 'code'}, config.options) 36 | } 37 | 38 | /** 39 | * Injections to be made by the IoC container 40 | * 41 | * @return {Array} 42 | */ 43 | static get inject () { 44 | return ['Adonis/Src/Config'] 45 | } 46 | 47 | /** 48 | * Scope seperator for seperating multiple 49 | * scopes. 50 | * 51 | * @return {String} 52 | */ 53 | get scopeSeperator () { 54 | return ' ' 55 | } 56 | 57 | /** 58 | * Base url to be used for constructing 59 | * mixer oauth urls. 60 | * 61 | * @return {String} 62 | */ 63 | get baseUrl () { 64 | return 'https://www.patreon.com/' 65 | } 66 | 67 | /** 68 | * Relative url to be used for redirecting 69 | * user. 70 | * 71 | * @return {String} [description] 72 | */ 73 | get authorizeUrl () { 74 | return 'oauth2/authorize' 75 | } 76 | 77 | /** 78 | * Relative url to be used for exchanging 79 | * access token. 80 | * 81 | * @return {String} 82 | */ 83 | get accessTokenUrl () { 84 | return 'api/oauth2/token' 85 | } 86 | 87 | /** 88 | * API url to be used for getting Patreon user's profile 89 | * 90 | * @return {String} 91 | */ 92 | get apiUrl () { 93 | return 'https://www.patreon.com/api/oauth2/api' 94 | } 95 | 96 | /** 97 | * Returns initial scopes to be used right from the 98 | * config file. Otherwise it will fallback to the 99 | * commonly used scopes 100 | * 101 | * @param {Array} scopes 102 | * 103 | * @return {Array} 104 | * 105 | * @private 106 | */ 107 | _getInitialScopes (scopes) { 108 | return _.size(scopes) ? scopes : ['users pledges-to-me my-campaign'] 109 | } 110 | 111 | /** 112 | * Returns the user profile as an object using the 113 | * access token 114 | * 115 | * @param {String} accessToken 116 | * @param {Array} [fields] 117 | * 118 | * @return {Object} 119 | * 120 | * @private 121 | */ 122 | async _getUserProfile (accessToken, fields) { 123 | const response = await got(`${this.apiUrl}/current_user`, { 124 | headers: { 125 | 'Authorization': accessToken ? 'Bearer ' + accessToken : undefined 126 | }, 127 | json: true 128 | }) 129 | return response.body 130 | } 131 | 132 | /** 133 | * Returns the redirect url for a given provider. 134 | * 135 | * @param {Array} scope 136 | * 137 | * @return {String} 138 | */ 139 | async getRedirectUrl (scope) { 140 | scope = _.size(scope) ? scope : this._scope 141 | return this.getUrl(this._redirectUri, scope, this._redirectUriOptions) 142 | } 143 | 144 | /** 145 | * Parses provider error by fetching error message 146 | * from nested data property. 147 | * 148 | * @param {Object} error 149 | * 150 | * @return {Error} 151 | */ 152 | parseProviderError (error) { 153 | console.log(error) 154 | const parsedError = _.isString(error.data) ? JSON.parse(error.data) : null 155 | const message = _.get(parsedError, 'message', error) 156 | return CE.OAuthException.tokenExchangeException(message, error.statusCode, parsedError) 157 | } 158 | 159 | /** 160 | * Parses the redirect errors returned by mixer 161 | * and returns the error message. 162 | * 163 | * @param {Object} queryParams 164 | * 165 | * @return {String} 166 | */ 167 | parseRedirectError (queryParams) { 168 | return queryParams.error_message || 'Oauth failed during redirect' 169 | } 170 | 171 | /** 172 | * Returns the user profile with it's access token, refresh token 173 | * and token expiry 174 | * 175 | * @param {Object} queryParams 176 | * @param {Array} [fields] 177 | * 178 | * @return {Object} 179 | */ 180 | async getUser (queryParams, fields) { 181 | const code = queryParams.code 182 | /** 183 | * Throw an exception when query string does not have 184 | * code. 185 | */ 186 | if (!code) { 187 | const errorMessage = this.parseRedirectError(queryParams) 188 | throw CE.OAuthException.tokenExchangeException(errorMessage, null, errorMessage) 189 | } 190 | 191 | const accessTokenResponse = await this.getAccessToken(code, this._redirectUri, { 192 | 'grant_type': 'authorization_code' 193 | }) 194 | 195 | const userProfile = await this._getUserProfile(accessTokenResponse.accessToken) 196 | 197 | const user = new AllyUser() 198 | user 199 | .setOriginal(userProfile) 200 | .setFields( 201 | userProfile.data.id, 202 | userProfile.data.attributes.vanity, 203 | userProfile.data.attributes.email, 204 | userProfile.data.attributes.full_name, 205 | userProfile.data.attributes.image_url 206 | ) 207 | .setToken( 208 | accessTokenResponse.accessToken, 209 | accessTokenResponse.refreshToken, 210 | null, 211 | Number(accessTokenResponse.result.expires_in) 212 | ) 213 | 214 | return user 215 | } 216 | } 217 | 218 | module.exports = Patreon -------------------------------------------------------------------------------- /Drivers/Twitch.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /* 4 | * adonis-ally twitch driver 5 | * 6 | * (c) Vladyslav Gaysyuk 7 | * 8 | * For the full copyright and license information, please view the LICENSE 9 | * file that was distributed with this source code. 10 | */ 11 | 12 | const CE = require('@adonisjs/ally/src/Exceptions') 13 | const OAuth2Scheme = require('@adonisjs/ally/src/Schemes/OAuth2') 14 | const AllyUser = require('@adonisjs/ally/src/AllyUser') 15 | const got = require('got') 16 | const utils = require('@adonisjs/ally/lib/utils') 17 | const _ = require('lodash') 18 | 19 | class Twitch extends OAuth2Scheme { 20 | 21 | constructor (Config) { 22 | const config = Config.get('services.ally.twitch') 23 | 24 | utils.validateDriverConfig('twitch', config, ['clientId', 'clientSecret', 'redirectUri']) 25 | 26 | super(config.clientId, config.clientSecret, config.headers) 27 | 28 | /** 29 | * Oauth specific values to be used when creating the redirect 30 | * url or fetching user profile. 31 | */ 32 | this._scope = this._getInitialScopes(config.scope) 33 | this._api_version = config.api_version || 'v5' 34 | this._redirectUri = config.redirectUri 35 | this.config = config 36 | this._redirectUriOptions = _.merge({response_type: 'code'}, config.options) 37 | } 38 | 39 | /** 40 | * Injections to be made by the IoC container 41 | * 42 | * @return {Array} 43 | */ 44 | static get inject () { 45 | return ['Adonis/Src/Config'] 46 | } 47 | 48 | /** 49 | * Scope seperator for seperating multiple 50 | * scopes. 51 | * 52 | * @return {String} 53 | */ 54 | get scopeSeperator () { 55 | return ' ' 56 | } 57 | 58 | /** 59 | * Base url to be used for constructing 60 | * twitch oauth urls. 61 | * 62 | * @return {String} 63 | */ 64 | get baseUrl () { 65 | return 'https://api.twitch.tv/kraken' 66 | } 67 | 68 | /** 69 | * Relative url to be used for redirecting 70 | * user. 71 | * 72 | * @return {String} [description] 73 | */ 74 | get authorizeUrl () { 75 | return '/oauth2/authorize' 76 | } 77 | 78 | /** 79 | * Relative url to be used for exchanging 80 | * access token. 81 | * 82 | * @return {String} 83 | */ 84 | get accessTokenUrl () { 85 | return '/oauth2/token' 86 | } 87 | 88 | /** 89 | * API url to be used for getting VKontakte user's profile 90 | * 91 | * @return {String} 92 | */ 93 | get apiUrl () { 94 | return 'https://api.twitch.tv/kraken' 95 | } 96 | 97 | /** 98 | * Returns initial scopes to be used right from the 99 | * config file. Otherwise it will fallback to the 100 | * commonly used scopes 101 | * 102 | * @param {Array} scopes 103 | * 104 | * @return {Array} 105 | * 106 | * @private 107 | */ 108 | _getInitialScopes (scopes) { 109 | return _.size(scopes) ? scopes : ['user_read'] 110 | } 111 | 112 | /** 113 | * Returns the user profile as an object using the 114 | * access token 115 | * 116 | * @param {String} accessToken 117 | * @param {Array} [fields] 118 | * 119 | * @return {Object} 120 | * 121 | * @private 122 | */ 123 | async _getUserProfile (accessToken, fields) { 124 | const response = await got(`${this.apiUrl}/user`, { 125 | headers: { 126 | 'Authorization': accessToken?'OAuth ' + accessToken : undefined, 127 | 'Accept': `Accept: application/vnd.twitchtv.${this._api_version}+json`, 128 | 'Client-ID': this.config.clientId 129 | }, 130 | json: true 131 | }) 132 | return response.body 133 | } 134 | 135 | /** 136 | * Returns the redirect url for a given provider. 137 | * 138 | * @param {Array} scope 139 | * 140 | * @return {String} 141 | */ 142 | async getRedirectUrl (scope) { 143 | scope = _.size(scope) ? scope : this._scope 144 | return this.getUrl(this._redirectUri, scope, this._redirectUriOptions) 145 | } 146 | 147 | /** 148 | * Parses provider error by fetching error message 149 | * from nested data property. 150 | * 151 | * @param {Object} error 152 | * 153 | * @return {Error} 154 | */ 155 | parseProviderError (error) { 156 | const parsedError = _.isString(error.data) ? JSON.parse(error.data) : null 157 | const message = _.get(parsedError, 'message', error) 158 | return CE.OAuthException.tokenExchangeException(message, error.statusCode, parsedError) 159 | } 160 | 161 | /** 162 | * Parses the redirect errors returned by twitch 163 | * and returns the error message. 164 | * 165 | * @param {Object} queryParams 166 | * 167 | * @return {String} 168 | */ 169 | parseRedirectError (queryParams) { 170 | return queryParams.error_message || 'Oauth failed during redirect' 171 | } 172 | 173 | /** 174 | * Returns the user profile with it's access token, refresh token 175 | * and token expiry 176 | * 177 | * @param {Object} queryParams 178 | * @param {Array} [fields] 179 | * 180 | * @return {Object} 181 | */ 182 | async getUser (queryParams, fields) { 183 | const code = queryParams.code 184 | 185 | /** 186 | * Throw an exception when query string does not have 187 | * code. 188 | */ 189 | if (!code) { 190 | const errorMessage = this.parseRedirectError(queryParams) 191 | throw CE.OAuthException.tokenExchangeException(errorMessage, null, errorMessage) 192 | } 193 | const accessTokenResponse = await this.getAccessToken(code, this._redirectUri, { 194 | grant_type: 'authorization_code' 195 | }) 196 | const userProfile = await this._getUserProfile(accessTokenResponse.accessToken, fields) 197 | 198 | const user = new AllyUser() 199 | user 200 | .setOriginal(userProfile) 201 | .setFields( 202 | userProfile._id, 203 | userProfile.display_name, 204 | userProfile.email, 205 | userProfile.name, 206 | userProfile.logo 207 | ) 208 | .setToken( 209 | accessTokenResponse.accessToken, 210 | accessTokenResponse.refreshToken, 211 | null, 212 | Number(accessTokenResponse.result.expires_in) 213 | ) 214 | 215 | return user 216 | } 217 | } 218 | 219 | module.exports = Twitch 220 | -------------------------------------------------------------------------------- /Drivers/Vk.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /* 4 | * adonis-ally vkontakte driver 5 | * 6 | * (c) Oleg Kovalev 7 | * 8 | * For the full copyright and license information, please view the LICENSE 9 | * file that was distributed with this source code. 10 | */ 11 | 12 | const CE = require('@adonisjs/ally/src/Exceptions') 13 | const OAuth2Scheme = require('@adonisjs/ally/src/Schemes/OAuth2') 14 | const AllyUser = require('@adonisjs/ally/src/AllyUser') 15 | const got = require('got') 16 | const utils = require('@adonisjs/ally/lib/utils') 17 | const _ = require('lodash') 18 | 19 | class VKontakte extends OAuth2Scheme { 20 | 21 | constructor (Config) { 22 | const config = Config.get('services.ally.vk') 23 | 24 | utils.validateDriverConfig('vk', config, ['clientId', 'clientSecret', 'redirectUri']) 25 | 26 | super(config.clientId, config.clientSecret, config.headers) 27 | 28 | /** 29 | * Oauth specific values to be used when creating the redirect 30 | * url or fetching user profile. 31 | */ 32 | this._scope = this._getInitialScopes(config.scope) 33 | this._fields = this._getInitialFields(config.fields) 34 | this._api_version = this._getInitialFields(config.api_version) || '5.65' 35 | this._redirectUri = config.redirectUri 36 | this._redirectUriOptions = _.merge({ 37 | response_type: 'code' 38 | }, config.options) 39 | } 40 | 41 | /** 42 | * Injections to be made by the IoC container 43 | * 44 | * @return {Array} 45 | */ 46 | static get inject () { 47 | return ['Adonis/Src/Config'] 48 | } 49 | 50 | /** 51 | * Scope seperator for seperating multiple 52 | * scopes. 53 | * 54 | * @return {String} 55 | */ 56 | get scopeSeperator () { 57 | return ',' 58 | } 59 | 60 | /** 61 | * Base url to be used for constructing 62 | * vk oauth urls. 63 | * 64 | * @return {String} 65 | */ 66 | get baseUrl () { 67 | return 'https://oauth.vk.com' 68 | } 69 | 70 | /** 71 | * Relative url to be used for redirecting 72 | * user. 73 | * 74 | * @return {String} [description] 75 | */ 76 | get authorizeUrl () { 77 | return 'authorize' 78 | } 79 | 80 | /** 81 | * Relative url to be used for exchanging 82 | * access token. 83 | * 84 | * @return {String} 85 | */ 86 | get accessTokenUrl () { 87 | return 'access_token' 88 | } 89 | 90 | /** 91 | * API url to be used for getting VKontakte user's profile 92 | * 93 | * @return {String} 94 | */ 95 | get apiUrl () { 96 | return 'https://api.vk.com/method' 97 | } 98 | 99 | /** 100 | * Returns initial scopes to be used right from the 101 | * config file. Otherwise it will fallback to the 102 | * commonly used scopes 103 | * 104 | * @param {Array} scopes 105 | * 106 | * @return {Array} 107 | * 108 | * @private 109 | */ 110 | _getInitialScopes (scopes) { 111 | return _.size(scopes) ? scopes : ['email'] 112 | } 113 | 114 | /** 115 | * Returns the initial fields to be used right from the 116 | * config file. Otherwise it will fallback to the 117 | * commonly used fields. 118 | * 119 | * @param {Array} fields 120 | * 121 | * @return {Array} 122 | * 123 | * @private 124 | */ 125 | _getInitialFields (fields) { 126 | return _.size(fields) ? fields : ['uid', 'first_name', 'screen_name', 'last_name', 'has_photo', 'photo', 'city'] 127 | } 128 | 129 | /** 130 | * Returns the user profile as an object using the 131 | * access token 132 | * 133 | * @param {String} accessToken 134 | * @param {Array} [fields] 135 | * 136 | * @return {Object} 137 | * 138 | * @private 139 | */ 140 | async _getUserProfile (accessToken, fields) { 141 | fields = _.size(fields) ? fields : this._fields 142 | const profileUrl = `${this.apiUrl}/users.get?access_token=${accessToken}&fields=${fields.join(',')}&https=1&v=${this._api_version}` 143 | const response = await got(profileUrl, { 144 | headers: { 145 | 'Accept': 'application/json' 146 | }, 147 | json: true 148 | }) 149 | return response.body 150 | } 151 | 152 | /** 153 | * Returns the redirect url for a given provider. 154 | * 155 | * @param {Array} scope 156 | * 157 | * @return {String} 158 | */ 159 | async getRedirectUrl (scope) { 160 | scope = _.size(scope) ? scope : this._scope 161 | return this.getUrl(this._redirectUri, scope, this._redirectUriOptions) 162 | } 163 | 164 | /** 165 | * Parses provider error by fetching error message 166 | * from nested data property. 167 | * 168 | * @param {Object} error 169 | * 170 | * @return {Error} 171 | */ 172 | parseProviderError (error) { 173 | const parsedError = _.isString(error.data) ? JSON.parse(error.data) : null 174 | const message = _.get(parsedError, 'error.message', error) 175 | return CE.OAuthException.tokenExchangeException(message, error.statusCode, parsedError) 176 | } 177 | 178 | /** 179 | * Parses the redirect errors returned by vk 180 | * and returns the error message. 181 | * 182 | * @param {Object} queryParams 183 | * 184 | * @return {String} 185 | */ 186 | parseRedirectError (queryParams) { 187 | return queryParams.error_message || 'Oauth failed during redirect' 188 | } 189 | 190 | /** 191 | * Returns the user profile with it's access token, refresh token 192 | * and token expiry 193 | * 194 | * @param {Object} queryParams 195 | * @param {Array} [fields] 196 | * 197 | * @return {Object} 198 | */ 199 | async getUser (queryParams, fields) { 200 | const code = queryParams.code 201 | 202 | /** 203 | * Throw an exception when query string does not have 204 | * code. 205 | */ 206 | if (!code) { 207 | const errorMessage = this.parseRedirectError(queryParams) 208 | throw CE.OAuthException.tokenExchangeException(errorMessage, null, errorMessage) 209 | } 210 | 211 | const accessTokenResponse = await this.getAccessToken(code, this._redirectUri, { 212 | grant_type: 'authorization_code' 213 | }) 214 | const userProfile = await this._getUserProfile(accessTokenResponse.accessToken, fields) 215 | const user = new AllyUser() 216 | const avatarUrl = userProfile.response[0].photo 217 | user 218 | .setOriginal(userProfile) 219 | .setFields( 220 | accessTokenResponse.result.user_id, 221 | `${userProfile.response[0].first_name} ${userProfile.response[0].last_name}`, 222 | accessTokenResponse.result.email, 223 | userProfile.response[0].screen_name, 224 | avatarUrl 225 | ) 226 | .setToken( 227 | accessTokenResponse.accessToken, 228 | accessTokenResponse.refreshToken, 229 | null, 230 | Number(accessTokenResponse.result.expires_in) 231 | ) 232 | 233 | return user 234 | } 235 | } 236 | 237 | module.exports = VKontakte 238 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 Vladyslav Gaysyuk 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 |

Welcome to Adonis Ally Extended 👋

2 |

3 | Version 4 | 5 | Documentation 6 | 7 | 8 | Maintenance 9 | 10 | 11 | License: MIT 12 | 13 | 14 | Twitter: AdmiralMiki 15 | 16 |

17 | 18 | This package gives you additional social services to auth your users. 19 | 20 | Any of service that accepts OAuth can be added. If you want an additional service - please create a issue and I will take my time to add it. 21 | Avaiable services for now are: 22 | * Twitch.tv 23 | * Mixer.com 24 | * Vk.com (Thanks to [@oddie](https://github.com/oddie)) 25 | * Patreon.com (Thanks to [@mindofjonas](https://github.com/mindofjonas)) 26 | 27 | ## Install 28 | 29 | ```sh 30 | adonis install @mikield/adonis-ally-extended 31 | ``` 32 | 33 | #### The provider will be registered inside `start/app.js` file. 34 | 35 | ```js 36 | const providers = [ 37 | '@mikield/adonis-ally-extended/ServiceProvider' 38 | ] 39 | ``` 40 | 41 | #### Add additional fields to `config/services.js` file. 42 | 43 | ```js 44 | ... 45 | /* 46 | |-------------------------------------------------------------------------- 47 | | Vk Configuration 48 | |-------------------------------------------------------------------------- 49 | | 50 | | You can access your application credentials from the vk developers 51 | | page. https://vk.com/apps?act=manage 52 | | 53 | */ 54 | vk: { 55 | clientId: Env.get('VK_CLIENT_ID'), 56 | clientSecret: Env.get('VK_CLIENT_SECRET'), 57 | redirectUri: `${Env.get('APP_URL')}/authenticated/vk` 58 | }, 59 | 60 | /* 61 | |-------------------------------------------------------------------------- 62 | | Twitch Configuration 63 | |-------------------------------------------------------------------------- 64 | | 65 | | You can access your application credentials from the twitch developers 66 | | dashboard. https://dev.twitch.tv/dashboard 67 | | 68 | */ 69 | twitch: { 70 | clientId: Env.get('TWITCH_CLIENT_ID'), 71 | clientSecret: Env.get('TWITCH_CLIENT_SECRET'), 72 | redirectUri: `${Env.get('APP_URL')}/authenticated/twitch` 73 | }, 74 | 75 | /* 76 | |-------------------------------------------------------------------------- 77 | | Mixer Configuration 78 | |-------------------------------------------------------------------------- 79 | | 80 | | You can access your application credentials from the Mixer developers 81 | | lab. https://mixer.com/lab/oauth 82 | | 83 | */ 84 | mixer: { 85 | clientId: Env.get('MIXER_CLIENT_ID'), 86 | clientSecret: Env.get('MIXER_CLIENT_SECRET'), 87 | redirectUri: `${Env.get('APP_URL')}/authenticated/mixer` 88 | }, 89 | 90 | /* 91 | |-------------------------------------------------------------------------- 92 | | Patreon Configuration 93 | |-------------------------------------------------------------------------- 94 | | 95 | | You can access your application credentials from the Patreon developers 96 | | lab. https://www.patreon.com/developers 97 | | 98 | */ 99 | patreon: { 100 | clientId: Env.get('MIXER_CLIENT_ID'), 101 | clientSecret: Env.get('MIXER_CLIENT_SECRET'), 102 | redirectUri: `${Env.get('APP_URL')}/authenticated/mixer` 103 | } 104 | ... 105 | ``` 106 | 107 | 108 | ## Usage 109 | 110 | Now you can access, the `ally` object on each HTTP request 111 | 112 | ```js 113 | Route.get('/:service', async ({ request, ally }) => { 114 | let {service} = await request.all() 115 | await ally.driver(service).redirect() 116 | }) 117 | 118 | Route.get('authenticated/:service', async ({ request, ally }) => { 119 | let {service} = await request.all() 120 | const user = await ally.driver(service).getUser() 121 | 122 | return user 123 | }) 124 | ``` 125 | 126 | ## Author 127 | 128 | 👤 **Vladyslav Gaysyuk ** 129 | 130 | * Website: https://mikield.rocks 131 | * Twitter: [@AdmiralMiki](https://twitter.com/AdmiralMiki) 132 | * Github: [@mikield](https://github.com/mikield) 133 | * LinkedIn: [@mikield](https://linkedin.com/in/mikield) 134 | 135 | ## 🤝 Contributing 136 | 137 | Contributions, issues and feature requests are welcome!
Feel free to check [issues page](https://github.com/mikield/adonis-ally-extended/issues). You can also take a look at the [contributing guide](https://github.com/mikield/adonis-ally-extended/blob/master/CONTRIBUTING.md). 138 | 139 | ## Show your support 140 | 141 | Give a ⭐️ if this project helped you! 142 | 143 | 144 | 145 | 146 | 147 | ## 📝 License 148 | 149 | Copyright © 2020 [Vladyslav Gaysyuk](https://github.com/mikield).
150 | This project is [MIT](https://github.com/mikield/adonis-ally-extended/blob/master/LICENSE) licensed. 151 | 152 | *** 153 | _This README was generated with ❤️ by [readme-md-generator](https://github.com/kefranabg/readme-md-generator)_ -------------------------------------------------------------------------------- /ServiceProvider.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /* 4 | * Adonis Ally Extended 5 | * 6 | * (c) Vladyslav Gaysyuk 7 | * 8 | * For the full copyright and license information, please view the LICENSE 9 | * file that was distributed with this source code. 10 | */ 11 | 12 | const { ServiceProvider } = require('@adonisjs/fold') 13 | 14 | const Drivers = [ 15 | { name: "vk", driver: require('./Drivers/Vk') }, 16 | { name: "twitch", driver: require('./Drivers/Twitch') }, 17 | { name: "mixer", driver: require('./Drivers/Mixer') }, 18 | { name: "patreon", driver: require('./Drivers/Patreon') } 19 | ] 20 | 21 | class AllyServiceProvider extends ServiceProvider { 22 | 23 | /** 24 | * The register method called by ioc container 25 | * as a life-cycle method 26 | * 27 | * @method register 28 | * 29 | * @return {void} 30 | */ 31 | register() { 32 | Drivers.forEach(({ name, driver }) => { 33 | this.app.extend('Adonis/Addons/Ally', name, () => { 34 | return driver 35 | }) 36 | }); 37 | } 38 | 39 | boot() {} 40 | 41 | 42 | } 43 | 44 | module.exports = AllyServiceProvider 45 | -------------------------------------------------------------------------------- /instructions.md: -------------------------------------------------------------------------------- 1 | 2 | ## Registering provider 3 | 4 | The provider will be registered inside `start/app.js` file. 5 | 6 | ```js 7 | const providers = [ 8 | '@mikield/adonis-ally-extended/ServiceProvider' 9 | ] 10 | ``` 11 | 12 | Add additional fields to `config/services.js` file. 13 | 14 | ```js 15 | ... 16 | /* 17 | |-------------------------------------------------------------------------- 18 | | Vk Configuration 19 | |-------------------------------------------------------------------------- 20 | | 21 | | You can access your application credentials from the vk developers 22 | | page. https://vk.com/apps?act=manage 23 | | 24 | */ 25 | vk: { 26 | clientId: Env.get('VK_CLIENT_ID'), 27 | clientSecret: Env.get('VK_CLIENT_SECRET'), 28 | redirectUri: `${Env.get('APP_URL')}/authenticated/vk` 29 | }, 30 | 31 | /* 32 | |-------------------------------------------------------------------------- 33 | | Twitch Configuration 34 | |-------------------------------------------------------------------------- 35 | | 36 | | You can access your application credentials from the twitch developers 37 | | dashboard. https://dev.twitch.tv/dashboard 38 | | 39 | */ 40 | twitch: { 41 | clientId: Env.get('TWITCH_CLIENT_ID'), 42 | clientSecret: Env.get('TWITCH_CLIENT_SECRET'), 43 | redirectUri: `${Env.get('APP_URL')}/authenticated/twitch` 44 | }, 45 | 46 | /* 47 | |-------------------------------------------------------------------------- 48 | | Mixer Configuration 49 | |-------------------------------------------------------------------------- 50 | | 51 | | You can access your application credentials from the Mixer developers 52 | | lab. https://mixer.com/lab/oauth 53 | | 54 | */ 55 | mixer: { 56 | clientId: Env.get('MIXER_CLIENT_ID'), 57 | clientSecret: Env.get('MIXER_CLIENT_SECRET'), 58 | redirectUri: `${Env.get('APP_URL')}/authenticated/mixer` 59 | }, 60 | 61 | /* 62 | |-------------------------------------------------------------------------- 63 | | Mixer Configuration 64 | |-------------------------------------------------------------------------- 65 | | 66 | | You can access your application credentials from the Patreon developers 67 | | lab. https://www.patreon.com/developers 68 | | 69 | */ 70 | patreon: { 71 | clientId: Env.get('MIXER_CLIENT_ID'), 72 | clientSecret: Env.get('MIXER_CLIENT_SECRET'), 73 | redirectUri: `${Env.get('APP_URL')}/authenticated/mixer` 74 | } 75 | ... 76 | ``` 77 | 78 | ## Usage 79 | 80 | Now you can access, the `ally` object on each HTTP request 81 | 82 | ```js 83 | Route.get('/:service', async ({ request, ally }) => { 84 | let {service} = await request.all() 85 | await ally.driver(service).redirect() 86 | }) 87 | 88 | Route.get('authenticated/:service', async ({ request, ally }) => { 89 | let {service} = await request.all() 90 | const user = await ally.driver(service).getUser() 91 | 92 | return user 93 | }) 94 | ``` 95 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@mikield/adonis-ally-extended", 3 | "version": "0.7.2", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@adonisjs/ally": { 8 | "version": "2.1.3", 9 | "resolved": "https://registry.npmjs.org/@adonisjs/ally/-/ally-2.1.3.tgz", 10 | "integrity": "sha512-Dl3/YHPHJy+g0DfY/jWEHF2hyRBemPLq3ca/ybzjmfMARaUsTNciDtc72Fx/3upCLjaL8q8YzTDpI0DFBENa2Q==", 11 | "dev": true, 12 | "requires": { 13 | "@adonisjs/generic-exceptions": "^2.0.1", 14 | "debug": "^4.0.1", 15 | "got": "8.3.2", 16 | "lodash": "^4.17.11", 17 | "oauth": "^0.9.15", 18 | "uuid": "^3.3.2" 19 | } 20 | }, 21 | "@adonisjs/fold": { 22 | "version": "4.0.9", 23 | "resolved": "https://registry.npmjs.org/@adonisjs/fold/-/fold-4.0.9.tgz", 24 | "integrity": "sha512-eH6048Ug32BvYvvvfRThy+IDE8lcRtqExca2TfE/Gw5ZP51rVEYqPd1yy3ioB4R5XI8VUS0hjOt5l7tKUh4Sww==", 25 | "dev": true, 26 | "requires": { 27 | "@adonisjs/generic-exceptions": "^2.0.1", 28 | "caller": "^1.0.1", 29 | "debug": "^3.1.0", 30 | "lodash": "^4.17.10", 31 | "require-stack": "^1.0.2" 32 | }, 33 | "dependencies": { 34 | "debug": { 35 | "version": "3.2.6", 36 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", 37 | "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", 38 | "dev": true, 39 | "requires": { 40 | "ms": "^2.1.1" 41 | } 42 | } 43 | } 44 | }, 45 | "@adonisjs/generic-exceptions": { 46 | "version": "2.0.1", 47 | "resolved": "https://registry.npmjs.org/@adonisjs/generic-exceptions/-/generic-exceptions-2.0.1.tgz", 48 | "integrity": "sha512-ZIPnj7vlRZKaAyZ4c2SUFCpJ6Yk+xzR+STjsze9unmZQncpQmVq1K8r20pXX3Z9rnxKVlfwO58HTxMMWaX9t9A==", 49 | "dev": true, 50 | "requires": { 51 | "node-exceptions": "^3.0.0", 52 | "upcast": "^2.1.1" 53 | } 54 | }, 55 | "@sindresorhus/is": { 56 | "version": "0.7.0", 57 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.7.0.tgz", 58 | "integrity": "sha512-ONhaKPIufzzrlNbqtWFFd+jlnemX6lJAgq9ZeiZtS7I1PIf/la7CW4m83rTXRnVnsMbW2k56pGYu7AUFJD9Pow==", 59 | "dev": true 60 | }, 61 | "acorn": { 62 | "version": "7.1.1", 63 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.1.1.tgz", 64 | "integrity": "sha512-add7dgA5ppRPxCFJoAGfMDi7PIBXq1RtGo7BhbLaxwrXPOmw8gq48Y9ozT01hUKy9byMjlR20EJhu5zlkErEkg==", 65 | "dev": true 66 | }, 67 | "acorn-node": { 68 | "version": "1.8.2", 69 | "resolved": "https://registry.npmjs.org/acorn-node/-/acorn-node-1.8.2.tgz", 70 | "integrity": "sha512-8mt+fslDufLYntIoPAaIMUe/lrbrehIiwmR3t2k9LljIzoigEPF27eLk2hy8zSGzmR/ogr7zbRKINMo1u0yh5A==", 71 | "dev": true, 72 | "requires": { 73 | "acorn": "^7.0.0", 74 | "acorn-walk": "^7.0.0", 75 | "xtend": "^4.0.2" 76 | } 77 | }, 78 | "acorn-walk": { 79 | "version": "7.1.1", 80 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.1.1.tgz", 81 | "integrity": "sha512-wdlPY2tm/9XBr7QkKlq0WQVgiuGTX6YWPyRyBviSoScBuLfTVQhvwg6wJ369GJ/1nPfTLMfnrFIfjqVg6d+jQQ==", 82 | "dev": true 83 | }, 84 | "cacheable-request": { 85 | "version": "2.1.4", 86 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-2.1.4.tgz", 87 | "integrity": "sha1-DYCIAbY0KtM8kd+dC0TcCbkeXD0=", 88 | "dev": true, 89 | "requires": { 90 | "clone-response": "1.0.2", 91 | "get-stream": "3.0.0", 92 | "http-cache-semantics": "3.8.1", 93 | "keyv": "3.0.0", 94 | "lowercase-keys": "1.0.0", 95 | "normalize-url": "2.0.1", 96 | "responselike": "1.0.2" 97 | }, 98 | "dependencies": { 99 | "lowercase-keys": { 100 | "version": "1.0.0", 101 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.0.tgz", 102 | "integrity": "sha1-TjNms55/VFfjXxMkvfb4jQv8cwY=", 103 | "dev": true 104 | } 105 | } 106 | }, 107 | "caller": { 108 | "version": "1.0.1", 109 | "resolved": "https://registry.npmjs.org/caller/-/caller-1.0.1.tgz", 110 | "integrity": "sha1-uFGGD3Dhlds9J3OVqhp+I+ow7PU=", 111 | "dev": true 112 | }, 113 | "clone-response": { 114 | "version": "1.0.2", 115 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 116 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 117 | "dev": true, 118 | "requires": { 119 | "mimic-response": "^1.0.0" 120 | } 121 | }, 122 | "core-util-is": { 123 | "version": "1.0.2", 124 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 125 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", 126 | "dev": true 127 | }, 128 | "cross-env": { 129 | "version": "5.2.1", 130 | "resolved": "https://registry.npmjs.org/cross-env/-/cross-env-5.2.1.tgz", 131 | "integrity": "sha512-1yHhtcfAd1r4nwQgknowuUNfIT9E8dOMMspC36g45dN+iD1blloi7xp8X/xAIDnjHWyt1uQ8PHk2fkNaym7soQ==", 132 | "dev": true, 133 | "requires": { 134 | "cross-spawn": "^6.0.5" 135 | } 136 | }, 137 | "cross-spawn": { 138 | "version": "6.0.5", 139 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", 140 | "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", 141 | "dev": true, 142 | "requires": { 143 | "nice-try": "^1.0.4", 144 | "path-key": "^2.0.1", 145 | "semver": "^5.5.0", 146 | "shebang-command": "^1.2.0", 147 | "which": "^1.2.9" 148 | } 149 | }, 150 | "debug": { 151 | "version": "4.1.1", 152 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", 153 | "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", 154 | "dev": true, 155 | "requires": { 156 | "ms": "^2.1.1" 157 | } 158 | }, 159 | "decode-uri-component": { 160 | "version": "0.2.0", 161 | "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", 162 | "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", 163 | "dev": true 164 | }, 165 | "decompress-response": { 166 | "version": "3.3.0", 167 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 168 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 169 | "dev": true, 170 | "requires": { 171 | "mimic-response": "^1.0.0" 172 | } 173 | }, 174 | "duplexer3": { 175 | "version": "0.1.4", 176 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 177 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 178 | "dev": true 179 | }, 180 | "from2": { 181 | "version": "2.3.0", 182 | "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", 183 | "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", 184 | "dev": true, 185 | "requires": { 186 | "inherits": "^2.0.1", 187 | "readable-stream": "^2.0.0" 188 | } 189 | }, 190 | "get-stream": { 191 | "version": "3.0.0", 192 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", 193 | "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", 194 | "dev": true 195 | }, 196 | "got": { 197 | "version": "8.3.2", 198 | "resolved": "https://registry.npmjs.org/got/-/got-8.3.2.tgz", 199 | "integrity": "sha512-qjUJ5U/hawxosMryILofZCkm3C84PLJS/0grRIpjAwu+Lkxxj5cxeCU25BG0/3mDSpXKTyZr8oh8wIgLaH0QCw==", 200 | "dev": true, 201 | "requires": { 202 | "@sindresorhus/is": "^0.7.0", 203 | "cacheable-request": "^2.1.1", 204 | "decompress-response": "^3.3.0", 205 | "duplexer3": "^0.1.4", 206 | "get-stream": "^3.0.0", 207 | "into-stream": "^3.1.0", 208 | "is-retry-allowed": "^1.1.0", 209 | "isurl": "^1.0.0-alpha5", 210 | "lowercase-keys": "^1.0.0", 211 | "mimic-response": "^1.0.0", 212 | "p-cancelable": "^0.4.0", 213 | "p-timeout": "^2.0.1", 214 | "pify": "^3.0.0", 215 | "safe-buffer": "^5.1.1", 216 | "timed-out": "^4.0.1", 217 | "url-parse-lax": "^3.0.0", 218 | "url-to-options": "^1.0.1" 219 | } 220 | }, 221 | "has-symbol-support-x": { 222 | "version": "1.4.2", 223 | "resolved": "https://registry.npmjs.org/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz", 224 | "integrity": "sha512-3ToOva++HaW+eCpgqZrCfN51IPB+7bJNVT6CUATzueB5Heb8o6Nam0V3HG5dlDvZU1Gn5QLcbahiKw/XVk5JJw==", 225 | "dev": true 226 | }, 227 | "has-to-string-tag-x": { 228 | "version": "1.4.1", 229 | "resolved": "https://registry.npmjs.org/has-to-string-tag-x/-/has-to-string-tag-x-1.4.1.tgz", 230 | "integrity": "sha512-vdbKfmw+3LoOYVr+mtxHaX5a96+0f3DljYd8JOqvOLsf5mw2Otda2qCDT9qRqLAhrjyQ0h7ual5nOiASpsGNFw==", 231 | "dev": true, 232 | "requires": { 233 | "has-symbol-support-x": "^1.4.1" 234 | } 235 | }, 236 | "http-cache-semantics": { 237 | "version": "3.8.1", 238 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", 239 | "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", 240 | "dev": true 241 | }, 242 | "inherits": { 243 | "version": "2.0.4", 244 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 245 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 246 | "dev": true 247 | }, 248 | "into-stream": { 249 | "version": "3.1.0", 250 | "resolved": "https://registry.npmjs.org/into-stream/-/into-stream-3.1.0.tgz", 251 | "integrity": "sha1-lvsKk2wSur1v8XUqF9BWFqvQlMY=", 252 | "dev": true, 253 | "requires": { 254 | "from2": "^2.1.1", 255 | "p-is-promise": "^1.1.0" 256 | } 257 | }, 258 | "is-object": { 259 | "version": "1.0.1", 260 | "resolved": "https://registry.npmjs.org/is-object/-/is-object-1.0.1.tgz", 261 | "integrity": "sha1-iVJojF7C/9awPsyF52ngKQMINHA=", 262 | "dev": true 263 | }, 264 | "is-plain-obj": { 265 | "version": "1.1.0", 266 | "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", 267 | "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", 268 | "dev": true 269 | }, 270 | "is-retry-allowed": { 271 | "version": "1.2.0", 272 | "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz", 273 | "integrity": "sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==", 274 | "dev": true 275 | }, 276 | "isarray": { 277 | "version": "1.0.0", 278 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 279 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 280 | "dev": true 281 | }, 282 | "isexe": { 283 | "version": "2.0.0", 284 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 285 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 286 | "dev": true 287 | }, 288 | "isurl": { 289 | "version": "1.0.0", 290 | "resolved": "https://registry.npmjs.org/isurl/-/isurl-1.0.0.tgz", 291 | "integrity": "sha512-1P/yWsxPlDtn7QeRD+ULKQPaIaN6yF368GZ2vDfv0AL0NwpStafjWCDDdn0k8wgFMWpVAqG7oJhxHnlud42i9w==", 292 | "dev": true, 293 | "requires": { 294 | "has-to-string-tag-x": "^1.2.0", 295 | "is-object": "^1.0.1" 296 | } 297 | }, 298 | "json-buffer": { 299 | "version": "3.0.0", 300 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 301 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 302 | "dev": true 303 | }, 304 | "keyv": { 305 | "version": "3.0.0", 306 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.0.0.tgz", 307 | "integrity": "sha512-eguHnq22OE3uVoSYG0LVWNP+4ppamWr9+zWBe1bsNcovIMy6huUJFPgy4mGwCd/rnl3vOLGW1MTlu4c57CT1xA==", 308 | "dev": true, 309 | "requires": { 310 | "json-buffer": "3.0.0" 311 | } 312 | }, 313 | "lodash": { 314 | "version": "4.17.19", 315 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz", 316 | "integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==", 317 | "dev": true 318 | }, 319 | "lowercase-keys": { 320 | "version": "1.0.1", 321 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 322 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 323 | "dev": true 324 | }, 325 | "mimic-response": { 326 | "version": "1.0.1", 327 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 328 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 329 | "dev": true 330 | }, 331 | "ms": { 332 | "version": "2.1.2", 333 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 334 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 335 | "dev": true 336 | }, 337 | "nice-try": { 338 | "version": "1.0.5", 339 | "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", 340 | "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", 341 | "dev": true 342 | }, 343 | "node-exceptions": { 344 | "version": "3.0.0", 345 | "resolved": "https://registry.npmjs.org/node-exceptions/-/node-exceptions-3.0.0.tgz", 346 | "integrity": "sha512-pFhMAqdN1avrFwtZs66HxYiVnbnH9wjXB4m8IKs5Z9+r7U5voqxT+EDbVkRfge+V7JnkOgKhN4HfKBn1o5g9Wg==", 347 | "dev": true 348 | }, 349 | "normalize-url": { 350 | "version": "2.0.1", 351 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-2.0.1.tgz", 352 | "integrity": "sha512-D6MUW4K/VzoJ4rJ01JFKxDrtY1v9wrgzCX5f2qj/lzH1m/lW6MhUZFKerVsnyjOhOsYzI9Kqqak+10l4LvLpMw==", 353 | "dev": true, 354 | "requires": { 355 | "prepend-http": "^2.0.0", 356 | "query-string": "^5.0.1", 357 | "sort-keys": "^2.0.0" 358 | } 359 | }, 360 | "oauth": { 361 | "version": "0.9.15", 362 | "resolved": "https://registry.npmjs.org/oauth/-/oauth-0.9.15.tgz", 363 | "integrity": "sha1-vR/vr2hslrdUda7VGWQS/2DPucE=", 364 | "dev": true 365 | }, 366 | "object-assign": { 367 | "version": "4.1.1", 368 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 369 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", 370 | "dev": true 371 | }, 372 | "p-cancelable": { 373 | "version": "0.4.1", 374 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-0.4.1.tgz", 375 | "integrity": "sha512-HNa1A8LvB1kie7cERyy21VNeHb2CWJJYqyyC2o3klWFfMGlFmWv2Z7sFgZH8ZiaYL95ydToKTFVXgMV/Os0bBQ==", 376 | "dev": true 377 | }, 378 | "p-finally": { 379 | "version": "1.0.0", 380 | "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", 381 | "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", 382 | "dev": true 383 | }, 384 | "p-is-promise": { 385 | "version": "1.1.0", 386 | "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-1.1.0.tgz", 387 | "integrity": "sha1-nJRWmJ6fZYgBewQ01WCXZ1w9oF4=", 388 | "dev": true 389 | }, 390 | "p-timeout": { 391 | "version": "2.0.1", 392 | "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-2.0.1.tgz", 393 | "integrity": "sha512-88em58dDVB/KzPEx1X0N3LwFfYZPyDc4B6eF38M1rk9VTZMbxXXgjugz8mmwpS9Ox4BDZ+t6t3QP5+/gazweIA==", 394 | "dev": true, 395 | "requires": { 396 | "p-finally": "^1.0.0" 397 | } 398 | }, 399 | "path-key": { 400 | "version": "2.0.1", 401 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", 402 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", 403 | "dev": true 404 | }, 405 | "pify": { 406 | "version": "3.0.0", 407 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 408 | "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", 409 | "dev": true 410 | }, 411 | "prepend-http": { 412 | "version": "2.0.0", 413 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 414 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 415 | "dev": true 416 | }, 417 | "process-nextick-args": { 418 | "version": "2.0.1", 419 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 420 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", 421 | "dev": true 422 | }, 423 | "query-string": { 424 | "version": "5.1.1", 425 | "resolved": "https://registry.npmjs.org/query-string/-/query-string-5.1.1.tgz", 426 | "integrity": "sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==", 427 | "dev": true, 428 | "requires": { 429 | "decode-uri-component": "^0.2.0", 430 | "object-assign": "^4.1.0", 431 | "strict-uri-encode": "^1.0.0" 432 | } 433 | }, 434 | "readable-stream": { 435 | "version": "2.3.7", 436 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 437 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 438 | "dev": true, 439 | "requires": { 440 | "core-util-is": "~1.0.0", 441 | "inherits": "~2.0.3", 442 | "isarray": "~1.0.0", 443 | "process-nextick-args": "~2.0.0", 444 | "safe-buffer": "~5.1.1", 445 | "string_decoder": "~1.1.1", 446 | "util-deprecate": "~1.0.1" 447 | }, 448 | "dependencies": { 449 | "safe-buffer": { 450 | "version": "5.1.2", 451 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 452 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 453 | "dev": true 454 | } 455 | } 456 | }, 457 | "require-stack": { 458 | "version": "1.0.2", 459 | "resolved": "https://registry.npmjs.org/require-stack/-/require-stack-1.0.2.tgz", 460 | "integrity": "sha1-4A7jSL+Wy1w+LUwntJ5BR24Ill0=", 461 | "dev": true, 462 | "requires": { 463 | "syntax-error": "^1.1.4" 464 | } 465 | }, 466 | "responselike": { 467 | "version": "1.0.2", 468 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 469 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 470 | "dev": true, 471 | "requires": { 472 | "lowercase-keys": "^1.0.0" 473 | } 474 | }, 475 | "safe-buffer": { 476 | "version": "5.2.0", 477 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", 478 | "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==", 479 | "dev": true 480 | }, 481 | "semver": { 482 | "version": "5.7.1", 483 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 484 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 485 | "dev": true 486 | }, 487 | "shebang-command": { 488 | "version": "1.2.0", 489 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", 490 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", 491 | "dev": true, 492 | "requires": { 493 | "shebang-regex": "^1.0.0" 494 | } 495 | }, 496 | "shebang-regex": { 497 | "version": "1.0.0", 498 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", 499 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", 500 | "dev": true 501 | }, 502 | "sort-keys": { 503 | "version": "2.0.0", 504 | "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-2.0.0.tgz", 505 | "integrity": "sha1-ZYU1WEhh7JfXMNbPQYIuH1ZoQSg=", 506 | "dev": true, 507 | "requires": { 508 | "is-plain-obj": "^1.0.0" 509 | } 510 | }, 511 | "strict-uri-encode": { 512 | "version": "1.1.0", 513 | "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", 514 | "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", 515 | "dev": true 516 | }, 517 | "string_decoder": { 518 | "version": "1.1.1", 519 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 520 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 521 | "dev": true, 522 | "requires": { 523 | "safe-buffer": "~5.1.0" 524 | }, 525 | "dependencies": { 526 | "safe-buffer": { 527 | "version": "5.1.2", 528 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 529 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 530 | "dev": true 531 | } 532 | } 533 | }, 534 | "syntax-error": { 535 | "version": "1.4.0", 536 | "resolved": "https://registry.npmjs.org/syntax-error/-/syntax-error-1.4.0.tgz", 537 | "integrity": "sha512-YPPlu67mdnHGTup2A8ff7BC2Pjq0e0Yp/IyTFN03zWO0RcK07uLcbi7C2KpGR2FvWbaB0+bfE27a+sBKebSo7w==", 538 | "dev": true, 539 | "requires": { 540 | "acorn-node": "^1.2.0" 541 | } 542 | }, 543 | "timed-out": { 544 | "version": "4.0.1", 545 | "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", 546 | "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", 547 | "dev": true 548 | }, 549 | "upcast": { 550 | "version": "2.1.2", 551 | "resolved": "https://registry.npmjs.org/upcast/-/upcast-2.1.2.tgz", 552 | "integrity": "sha512-c+ueM175OVWv9vr1SYA3rI1ao0bxq6Y7l6u5Sac25Hi0yWz9Lz341zt9/sVOa5+7lYcvHMb0xxaacFy/2lVF2w==", 553 | "dev": true, 554 | "requires": { 555 | "cross-env": "^5.1.0" 556 | } 557 | }, 558 | "url-parse-lax": { 559 | "version": "3.0.0", 560 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 561 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 562 | "dev": true, 563 | "requires": { 564 | "prepend-http": "^2.0.0" 565 | } 566 | }, 567 | "url-to-options": { 568 | "version": "1.0.1", 569 | "resolved": "https://registry.npmjs.org/url-to-options/-/url-to-options-1.0.1.tgz", 570 | "integrity": "sha1-FQWgOiiaSMvXpDTvuu7FBV9WM6k=", 571 | "dev": true 572 | }, 573 | "util-deprecate": { 574 | "version": "1.0.2", 575 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 576 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", 577 | "dev": true 578 | }, 579 | "uuid": { 580 | "version": "3.4.0", 581 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", 582 | "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", 583 | "dev": true 584 | }, 585 | "which": { 586 | "version": "1.3.1", 587 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 588 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 589 | "dev": true, 590 | "requires": { 591 | "isexe": "^2.0.0" 592 | } 593 | }, 594 | "xtend": { 595 | "version": "4.0.2", 596 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 597 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", 598 | "dev": true 599 | } 600 | } 601 | } 602 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@mikield/adonis-ally-extended", 3 | "version": "0.7.2", 4 | "description": "Additional Services for AdonisJS Ally package", 5 | "author": "Vladyslav Gaysyuk ", 6 | "license": "MIT", 7 | "scripts": { 8 | "test": "echo \"Error: no test specified\" && exit 1" 9 | }, 10 | "keywords": [ 11 | "adonisjs", 12 | "ally", 13 | "oauth2", 14 | "vk", 15 | "mixer", 16 | "twitch" 17 | ], 18 | "repository": { 19 | "type": "git", 20 | "url": "git+https://github.com/mikield/adonis-ally-extended.git" 21 | }, 22 | "bugs": { 23 | "url": "https://github.com/mikield/adonis-ally-extended/issues" 24 | }, 25 | "homepage": "https://github.com/mikield/adonis-ally-extended#readme", 26 | "devDependencies": { 27 | "@adonisjs/ally": "^2.1.3", 28 | "@adonisjs/fold": "^4.0.9" 29 | } 30 | } 31 | --------------------------------------------------------------------------------