require_relative 'group_presenter' module Carto module Api class UserPresenter include AccountTypeHelper def initialize(user, fetch_groups: false, current_viewer: nil, fetch_db_size: true, fetch_basemaps: false, fetch_profile: true) @user = user @fetch_groups = fetch_groups @current_viewer = current_viewer @fetch_db_size = fetch_db_size @fetch_basemaps = fetch_basemaps @fetch_profile = fetch_profile end def to_poro return {} if @user.nil? return to_public_poro unless current_viewer && @user.viewable_by?(current_viewer) poro = { id: @user.id, name: @user.name, last_name: @user.last_name, username: @user.username, email: @user.email, avatar_url: @user.avatar_url, website: @user.website, description: @user.description, location: @user.location, twitter_username: @user.twitter_username, disqus_shortname: @user.disqus_shortname, available_for_hire: @user.available_for_hire, base_url: @user.public_url, google_maps_query_string: @user.google_maps_query_string, quota_in_bytes: @user.quota_in_bytes, table_count: @user.table_count, viewer: @user.viewer?, role_display: @user.role_display, org_admin: @user.organization_admin?, public_visualization_count: @user.public_visualization_count, all_visualization_count: @user.all_visualization_count, org_user: @user.organization_id.present?, remove_logo: @user.remove_logo?, } if fetch_groups poro[:groups] = @user.groups ? @user.groups.map { |g| Carto::Api::GroupPresenter.new(g).to_poro } : [] end poro[:basemaps] = @user.basemaps if fetch_basemaps poro[:db_size_in_bytes] = @user.db_size_in_bytes if fetch_db_size if fetch_profile poro[:industry] = @user.industry poro[:company_employees] = @user.company_employees poro[:use_case] = @user.use_case poro[:company] = @user.company poro[:phone] = @user.phone poro[:job_role] = @user.job_role end poro end def to_eumapi_poro presentation = to_poro presentation.delete(:id) presentation[:soft_geocoding_limit] = @user.soft_geocoding_limit presentation[:api_key] = @user.api_key presentation end def to_public_poro return {} if @user.nil? poro = { id: @user.id, username: @user.username, name: @user.name, last_name: @user.last_name, avatar_url: @user.avatar_url, base_url: @user.public_url, google_maps_query_string: @user.google_maps_query_string, disqus_shortname: @user.disqus_shortname, viewer: @user.viewer?, org_admin: @user.organization_admin?, org_user: @user.organization_id.present?, remove_logo: @user.remove_logo? } if fetch_groups poro[:groups] = @user.groups ? @user.groups.map { |g| Carto::Api::GroupPresenter.new(g).to_poro } : [] end poro end def data(options = {}) return {} if @user.nil? db_size_in_bytes = @user.db_size_in_bytes data = { id: @user.id, email: @user.email, name: @user.name, last_name: @user.last_name, created_at: @user.created_at, username: @user.username, state: @user.state, account_type: @user.account_type, account_type_display_name: plan_name(@user.account_type), table_quota: @user.table_quota, public_map_quota: @user.public_map_quota, public_dataset_quota: @user.public_dataset_quota, private_map_quota: @user.private_map_quota, regular_api_key_quota: @user.regular_api_key_quota, table_count: @user.table_count, viewer: @user.viewer?, role_display: @user.role_display, industry: @user.industry, company_employees: @user.company_employees, use_case: @user.use_case, company: @user.company, phone: @user.phone, job_role: @user.job_role, org_admin: @user.organization_admin?, public_visualization_count: @user.public_visualization_count, public_privacy_map_count: @user.public_privacy_visualization_count, link_privacy_map_count: @user.link_privacy_visualization_count, password_privacy_map_count: @user.password_privacy_visualization_count, private_privacy_map_count: @user.private_privacy_visualization_count, owned_visualization_count: @user.owned_visualization_count, all_visualization_count: @user.all_visualization_count, visualization_count: @user.visualization_count, failed_import_count: failed_import_count, success_import_count: success_import_count, import_count: import_count, last_visualization_created_at: last_visualization_created_at, quota_in_bytes: @user.quota_in_bytes, # TODO: To be deprecated in favor of memory -> quota_in_bytes db_size_in_bytes: db_size_in_bytes, # TODO: To be deprecated in favor of memory -> db_size_in_bytes db_size_in_megabytes: db_size_in_bytes.present? ? (db_size_in_bytes / (1024.0 * 1024.0)).round(2) : nil, remaining_table_quota: @user.remaining_table_quota, remaining_byte_quota: @user.remaining_quota(db_size_in_bytes).to_f, storage: { # Total quota, including premium subscriptions addons: quota_in_bytes: @user.quota_in_bytes, # Total DB storage used, including premium and public synchronized datasets: db_size_in_bytes: db_size_in_bytes, # DB storage used by public subscriptions: subscriptions_public_size_in_bytes: @user.subscriptions_public_size_in_bytes, # DB storage used by premium subscriptions: subscriptions_premium_size_in_bytes: @user.subscriptions_premium_size_in_bytes, # Estimated premium datasets size: subscriptions_premium_estimated_size_in_bytes: @user.subscriptions_premium_estimated_size_in_bytes }, map_views: @user.organization_user? ? @user.organization.map_views_count : @user.map_views_count, map_views_quota: @user.organization_user? ? @user.organization.map_views_quota : @user.map_views_quota, unverified: @user.unverified?, geocoding: { quota: @user.organization_user? ? @user.organization.geocoding_quota : @user.geocoding_quota, block_price: @user.organization_user? ? @user.organization.geocoding_block_price : @user.geocoding_block_price, monthly_use: @user.organization_user? ? @user.organization.get_geocoding_calls : @user.get_geocoding_calls, hard_limit: @user.hard_geocoding_limit? }, here_isolines: { quota: @user.organization_user? ? @user.organization.here_isolines_quota : @user.here_isolines_quota, block_price: @user.organization_user? ? @user.organization.here_isolines_block_price : @user.here_isolines_block_price, monthly_use: @user.organization_user? ? @user.organization.get_here_isolines_calls : @user.get_here_isolines_calls, hard_limit: @user.hard_here_isolines_limit? }, mapzen_routing: { quota: @user.organization_user? ? @user.organization.mapzen_routing_quota : @user.mapzen_routing_quota, block_price: @user.organization_user? ? @user.organization.mapzen_routing_block_price : @user.mapzen_routing_block_price, monthly_use: @user.organization_user? ? @user.organization.get_mapzen_routing_calls : @user.get_mapzen_routing_calls, hard_limit: @user.hard_mapzen_routing_limit? }, geocoder_provider: @user.geocoder_provider, isolines_provider: @user.isolines_provider, routing_provider: @user.routing_provider, twitter: { enabled: @user.twitter_datasource_enabled, quota: @user.organization_user? ? @user.organization.twitter_datasource_quota : @user.twitter_datasource_quota, block_price: @user.organization_user? ? @user.organization.twitter_datasource_block_price : @user.twitter_datasource_block_price, block_size: @user.organization_user? ? @user.organization.twitter_datasource_block_size : @user.twitter_datasource_block_size, monthly_use: @user.organization_user? ? @user.organization.twitter_imports_count : @user.twitter_imports_count, hard_limit: @user.hard_twitter_datasource_limit, customized_config: CartoDB::Datasources::DatasourcesFactory.customized_config?(CartoDB::Datasources::Search::Twitter::DATASOURCE_NAME, @user) }, salesforce: { enabled: @user.organization_user? ? @user.organization.salesforce_datasource_enabled : @user.salesforce_datasource_enabled }, mailchimp: { enabled: Carto::AccountType.new.mailchimp?(@user) }, billing_period: @user.last_billing_cycle, next_billing_period: @user.next_billing_cycle, api_key: @user.api_key, layers: @user.layers.map { |layer| Carto::Api::LayerPresenter.new(layer).to_poro }, trial_ends_at: @user.trial_ends_at, upgraded_at: @user.upgraded_at, show_trial_reminder: @user.show_trial_reminder?, show_upgraded_message: (@user.account_type.downcase != 'free' && @user.upgraded_at && @user.upgraded_at + 15.days > Date.today ? true : false), actions: { private_tables: @user.private_tables_enabled, private_maps: @user.private_maps_enabled?, remove_logo: @user.remove_logo?, sync_tables: @user.sync_tables_enabled, google_maps_geocoder_enabled: @user.google_maps_geocoder_enabled?, google_maps_enabled: @user.google_maps_enabled?, engine_enabled: @user.engine_enabled?, builder_enabled: @user.builder_enabled?, mobile_sdk_enabled: @user.mobile_sdk_enabled? }, limits: { concurrent_syncs: CartoDB::PlatformLimits::Importer::UserConcurrentSyncsAmount::MAX_SYNCS_PER_USER, concurrent_imports: @user.max_concurrent_import_count, import_file_size: @user.max_import_file_size, import_table_rows: @user.max_import_table_row_count, max_layers: @user.max_layers }, notification: @user.notification, email_notifications: @user.decorate_email_notifications, avatar_url: @user.avatar, feature_flags: @user.feature_flags_names, base_url: @user.public_url, needs_password_confirmation: @user.needs_password_confirmation?, description: @user.description, website: @user.website, twitter_username: @user.twitter_username, disqus_shortname: @user.disqus_shortname, available_for_hire: @user.available_for_hire, location: @user.location, mfa_configured: @user.multifactor_authentication_configured?, is_enterprise: @user.enterprise?, do_enabled: @user.do_enabled?, do_bq_project: @user&.gcloud_settings&.[](:bq_project), do_bq_dataset: @user&.gcloud_settings&.[](:bq_dataset), } if @user.google_maps_geocoder_enabled? && (!@user.organization.present? || @user.organization_owner?) data[:google_maps_private_key] = @user.google_maps_private_key end if @user.organization.present? data[:organization] = Carto::Api::OrganizationPresenter.new(@user.organization).to_poro data[:organization][:available_quota_for_user] = @user.organization.unassigned_quota + @user.quota_in_bytes end if !@user.groups.nil? data[:groups] = @user.groups.map { |g| Carto::Api::GroupPresenter.new(g).to_poro } end if @user.mobile_sdk_enabled? data[:mobile_apps] = { mobile_xamarin: @user.mobile_xamarin, mobile_custom_watermark: @user.mobile_custom_watermark, mobile_offline_maps: @user.mobile_offline_maps, mobile_gis_extension: @user.mobile_gis_extension, mobile_max_open_users: @user.mobile_max_open_users, mobile_max_private_users: @user.mobile_max_private_users } end if options[:extended] # TODO: This fields are pending migration data.merge({ :real_table_count => @user.real_tables.size, :last_active_time => @user.get_last_active_time }) else data end end private attr_reader :current_viewer, :current_user, :fetch_groups, :fetch_db_size, :fetch_basemaps, :fetch_profile def failed_import_count Carto::DataImport.where(user_id: @user.id, state: 'failure').count end def success_import_count Carto::DataImport.where(user_id: @user.id, state: 'complete').count end def import_count DataImport.where(user_id: @user.id).count end def last_visualization_created_at row_data = Carto::VisualizationQueryBuilder.new .with_user_id(@user.id) .with_order(:created_at, :desc) .build_paged(1, 1) .pluck(:created_at) row_data.nil? ? nil : row_data[0] end end end end