CartoDB/cartodb20

View on GitHub
lib/carto/oauth/google/api.rb

Summary

Maintainability
A
25 mins
Test Coverage
require 'typhoeus'
require_dependency 'carto/oauth/google/config'
require_dependency 'carto/oauth/api'

module Carto
  module Oauth
    module Google
      class Api < Carto::Oauth::Api
        include Carto::EmailCleaner

        USERINFO_ENDPOINT = 'https://openidconnect.googleapis.com/v1/userinfo'.freeze

        def student?
          false
        end

        def user_params
          {
            username: email.split('@')[0],
            email: email,
            name: first_name,
            last_name: last_name,
            google_sign_in: true
          }
        end

        def user
          User.where(email: clean_email(email)).first
        end

        def hidden_fields
          {
            oauth_provider: 'google',
            oauth_access_token: access_token,
            'user[google_sign_in]': true
          }
        end

        def valid?(user)
          email == user.email
        end

        private

        def id
          user_data['sub']
        end

        def first_name
          user_data.fetch('given_name', nil)
        end

        def last_name
          user_data.fetch('family_name', nil)
        end

        def email
          user_data['email']
        end

        def user_data
          @user_data ||= get_user_data
        rescue StandardError => e
          log_error(message: 'Error obtaining Google user data', exception: e, access_token: access_token)
          nil
        end

        def get_user_data
          response = Typhoeus::Request.new(
            "#{USERINFO_ENDPOINT}?access_token=#{access_token}",
            method: 'GET',
            ssl_verifypeer: true,
            timeout: 600
          ).run

          raise 'Invalid response code' unless response.code == 200
          JSON.parse(response.body)
        rescue StandardError => e
          trace_info = {
            message: 'Error in request to Google', exception: e
          }
          if response
            trace_info.merge!(response: {
              code: response.code,
              headers: response.headers,
              body: response.body,
              status: response.return_code
            })
          end
          log_error(trace_info)
          nil
        end
      end
    end
  end
end