Skip to content
Snippets Groups Projects
users.rb 15.7 KiB
Newer Older
  • Learn to ignore specific revisions
  •   class Users < Grape::API
    
        include PaginationParams
    
    
        before do
          allow_access_with_scope :read_user if request.get?
          authenticate!
        end
    
        resource :users, requirements: { uid: /[0-9]*/, id: /[0-9]*/ } do
    
    Robert Schilling's avatar
    Robert Schilling committed
          helpers do
            params :optional_attributes do
              optional :skype, type: String, desc: 'The Skype username'
              optional :linkedin, type: String, desc: 'The LinkedIn username'
              optional :twitter, type: String, desc: 'The Twitter username'
              optional :website_url, type: String, desc: 'The website of the user'
              optional :organization, type: String, desc: 'The organization of the user'
              optional :projects_limit, type: Integer, desc: 'The number of projects a user can create'
    
              optional :extern_uid, type: String, desc: 'The external authentication provider UID'
    
    Robert Schilling's avatar
    Robert Schilling committed
              optional :provider, type: String, desc: 'The external provider'
              optional :bio, type: String, desc: 'The biography of the user'
              optional :location, type: String, desc: 'The location of the user'
              optional :admin, type: Boolean, desc: 'Flag indicating the user is an administrator'
              optional :can_create_group, type: Boolean, desc: 'Flag indicating the user can create groups'
              optional :confirm, type: Boolean, desc: 'Flag indicating the account needs to be confirmed'
              optional :external, type: Boolean, desc: 'Flag indicating the user is an external user'
              all_or_none_of :extern_uid, :provider
            end
          end
    
          desc 'Get the list of users' do
            success Entities::UserBasic
          end
          params do
            optional :username, type: String, desc: 'Get a single user with a specific username'
            optional :search, type: String, desc: 'Search for a username'
            optional :active, type: Boolean, default: false, desc: 'Filters only active users'
            optional :external, type: Boolean, default: false, desc: 'Filters only external users'
            optional :blocked, type: Boolean, default: false, desc: 'Filters only blocked users'
    
            use :pagination
    
    Robert Schilling's avatar
    Robert Schilling committed
          end
    
          get do
    
            unless can?(current_user, :read_users_list, nil)
    
              render_api_error!("Not authorized.", 403)
            end
    
    
            if params[:username].present?
    
    Robert Schilling's avatar
    Robert Schilling committed
              users = User.where(username: params[:username])
    
    Robert Schilling's avatar
    Robert Schilling committed
              users = User.all
              users = users.active if params[:active]
              users = users.search(params[:search]) if params[:search].present?
              users = users.blocked if params[:blocked]
              users = users.external if params[:external] && current_user.is_admin?
    
            entity = current_user.is_admin? ? Entities::UserPublic : Entities::UserBasic
    
    Robert Schilling's avatar
    Robert Schilling committed
            present paginate(users), with: entity
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get a single user' do
            success Entities::UserBasic
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
          end
    
          get ":id" do
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params[:id])
            not_found!('User') unless user
    
    Felipe Artur's avatar
    Felipe Artur committed
            if current_user && current_user.is_admin?
    
              present user, with: Entities::UserPublic
    
    Robert Schilling's avatar
    Robert Schilling committed
            elsif can?(current_user, :read_user, user)
              present user, with: Entities::User
    
            else
              render_api_error!("User not found.", 404)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Create a user. Available only for admins.' do
    
    Robert Schilling's avatar
    Robert Schilling committed
          end
          params do
            requires :email, type: String, desc: 'The email of the user'
    
            optional :password, type: String, desc: 'The password of the new user'
            optional :reset_password, type: Boolean, desc: 'Flag indicating the user will be sent a password reset token'
            at_least_one_of :password, :reset_password
    
    Robert Schilling's avatar
    Robert Schilling committed
            requires :name, type: String, desc: 'The name of the user'
            requires :username, type: String, desc: 'The username of the user'
            use :optional_attributes
          end
    
          post do
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
    
            # Filter out params which are used later
    
            user_params = declared_params(include_missing: false)
            identity_attrs = user_params.slice(:provider, :extern_uid)
    
            confirm = user_params.delete(:confirm)
    
            user = User.new(user_params.except(:extern_uid, :provider, :reset_password))
    
            if user_params.delete(:reset_password)
              user.attributes = {
                force_random_password: true,
                password_expires_at: nil,
                created_by_id: current_user.id
              }
              user.generate_password
              user.generate_reset_token
            end
    
            user.skip_confirmation! unless confirm
    
            if identity_attrs.any?
              user.identities.build(identity_attrs)
            end
    
    
              present user, with: Entities::UserPublic
    
              conflict!('Email has already been taken') if User.
                  where(email: user.email).
                  count > 0
    
              conflict!('Username has already been taken') if User.
                  where(username: user.username).
                  count > 0
    
              render_validation_error!(user)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Update a user. Available only for admins.' do
    
    Robert Schilling's avatar
    Robert Schilling committed
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
            optional :email, type: String, desc: 'The email of the user'
            optional :password, type: String, desc: 'The password of the new user'
            optional :name, type: String, desc: 'The name of the user'
            optional :username, type: String, desc: 'The username of the user'
            use :optional_attributes
            at_least_one_of :email, :password, :name, :username, :skype, :linkedin,
                            :twitter, :website_url, :organization, :projects_limit,
                            :extern_uid, :provider, :bio, :location, :admin,
                            :can_create_group, :confirm, :external
          end
    
          put ":id" do
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params.delete(:id))
    
            not_found!('User') unless user
    
    Robert Schilling's avatar
    Robert Schilling committed
            conflict!('Email has already been taken') if params[:email] &&
                User.where(email: params[:email]).
    
                    where.not(id: user.id).count > 0
    
    
    Robert Schilling's avatar
    Robert Schilling committed
            conflict!('Username has already been taken') if params[:username] &&
                User.where(username: params[:username]).
    
                    where.not(id: user.id).count > 0
    
    
            user_params = declared_params(include_missing: false)
            identity_attrs = user_params.slice(:provider, :extern_uid)
    
            if identity_attrs.any?
              identity = user.identities.find_by(provider: identity_attrs[:provider])
    
              if identity
                identity.update_attributes(identity_attrs)
              else
                identity = user.identities.build(identity_attrs)
                identity.save
              end
            end
    
    
            user_params[:password_expires_at] = Time.now if user_params[:password].present?
    
            if user.update_attributes(user_params.except(:extern_uid, :provider))
    
              present user, with: Entities::UserPublic
    
              render_validation_error!(user)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Add an SSH key to a specified user. Available only for admins.' do
            success Entities::SSHKey
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
            requires :key, type: String, desc: 'The new SSH key'
            requires :title, type: String, desc: 'The title of the new SSH key'
          end
    
    Angus MacArthur's avatar
    Angus MacArthur committed
          post ":id/keys" do
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params.delete(:id))
            not_found!('User') unless user
    
            key = user.keys.new(declared_params(include_missing: false))
    
    
    Angus MacArthur's avatar
    Angus MacArthur committed
            if key.save
              present key, with: Entities::SSHKey
            else
    
              render_validation_error!(key)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get the SSH keys of a specified user. Available only for admins.' do
            success Entities::SSHKey
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
    
    Robert Schilling's avatar
    Robert Schilling committed
          end
          get ':id/keys' do
    
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
    
            user = User.find_by(id: params[:id])
    
            not_found!('User') unless user
    
    
            present paginate(user.keys), with: Entities::SSHKey
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Delete an existing SSH key from a specified user. Available only for admins.' do
            success Entities::SSHKey
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
            requires :key_id, type: Integer, desc: 'The ID of the SSH key'
          end
          delete ':id/keys/:key_id' do
    
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
    
            user = User.find_by(id: params[:id])
    
            not_found!('User') unless user
    
    
    Robert Schilling's avatar
    Robert Schilling committed
            key = user.keys.find_by(id: params[:key_id])
            not_found!('Key') unless key
    
            present key.destroy, with: Entities::SSHKey
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Add an email address to a specified user. Available only for admins.' do
            success Entities::Email
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
            requires :email, type: String, desc: 'The email of the user'
          end
    
          post ":id/emails" do
            authenticated_as_admin!
    
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params.delete(:id))
            not_found!('User') unless user
    
            email = user.emails.new(declared_params(include_missing: false))
    
    
            if email.save
              NotificationService.new.new_email(email)
              present email, with: Entities::Email
            else
              render_validation_error!(email)
            end
          end
    
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get the emails addresses of a specified user. Available only for admins.' do
            success Entities::Email
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
    
    Robert Schilling's avatar
    Robert Schilling committed
          end
          get ':id/emails' do
    
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params[:id])
    
            not_found!('User') unless user
    
    
            present paginate(user.emails), with: Entities::Email
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Delete an email address of a specified user. Available only for admins.' do
            success Entities::Email
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
            requires :email_id, type: Integer, desc: 'The ID of the email'
          end
          delete ':id/emails/:email_id' do
    
            authenticated_as_admin!
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params[:id])
    
            not_found!('User') unless user
    
    
    Robert Schilling's avatar
    Robert Schilling committed
            email = user.emails.find_by(id: params[:email_id])
            not_found!('Email') unless email
    
    Robert Schilling's avatar
    Robert Schilling committed
            email.destroy
            user.update_secondary_emails!
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Delete a user. Available only for admins.' do
            success Entities::Email
          end
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
          end
    
          delete ":id" do
            authenticated_as_admin!
    
    skv's avatar
    skv committed
            user = User.find_by(id: params[:id])
    
    Robert Schilling's avatar
    Robert Schilling committed
            not_found!('User') unless user
    
            ::Users::DestroyService.new(current_user).execute(user)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Block a user. Available only for admins.'
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
          end
    
          post ':id/block' do
    
            authenticated_as_admin!
            user = User.find_by(id: params[:id])
    
    Robert Schilling's avatar
    Robert Schilling committed
            not_found!('User') unless user
    
    Robert Schilling's avatar
    Robert Schilling committed
            if !user.ldap_blocked?
    
              forbidden!('LDAP blocked users cannot be modified by the API')
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Unblock a user. Available only for admins.'
          params do
            requires :id, type: Integer, desc: 'The ID of the user'
          end
    
          post ':id/unblock' do
    
            authenticated_as_admin!
            user = User.find_by(id: params[:id])
    
    Robert Schilling's avatar
    Robert Schilling committed
            not_found!('User') unless user
    
    Robert Schilling's avatar
    Robert Schilling committed
            if user.ldap_blocked?
    
              forbidden!('LDAP blocked users cannot be unblocked by the API')
    
    Gabriel Mazetto's avatar
    Gabriel Mazetto committed
            else
              user.activate
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get the contribution events of a specified user' do
    
            detail 'This feature was introduced in GitLab 8.13.'
            success Entities::Event
          end
          params do
    
    Robert Schilling's avatar
    Robert Schilling committed
            requires :id, type: Integer, desc: 'The ID of the user'
    
            use :pagination
    
          end
          get ':id/events' do
    
    Robert Schilling's avatar
    Robert Schilling committed
            user = User.find_by(id: params[:id])
    
            not_found!('User') unless user
    
    
            events = user.events.
    
              merge(ProjectsFinder.new.execute(current_user)).
              references(:project).
              with_associations.
    
              recent
    
    
            present paginate(events), with: Entities::Event
          end
    
        resource :user do
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get the currently authenticated user' do
    
            present current_user, with: sudo? ? Entities::UserWithPrivateToken : Entities::UserPublic
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc "Get the currently authenticated user's SSH keys" do
            success Entities::SSHKey
          end
    
          params do
            use :pagination
          end
    
          get "keys" do
    
            present paginate(current_user.keys), with: Entities::SSHKey
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get a single key owned by currently authenticated user' do
            success Entities::SSHKey
          end
          params do
            requires :key_id, type: Integer, desc: 'The ID of the SSH key'
          end
          get "keys/:key_id" do
            key = current_user.keys.find_by(id: params[:key_id])
            not_found!('Key') unless key
    
    
            present key, with: Entities::SSHKey
          end
    
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Add a new SSH key to the currently authenticated user' do
            success Entities::SSHKey
          end
          params do
            requires :key, type: String, desc: 'The new SSH key'
            requires :title, type: String, desc: 'The title of the new SSH key'
          end
    
          post "keys" do
    
    Robert Schilling's avatar
    Robert Schilling committed
            key = current_user.keys.new(declared_params)
    
            if key.save
              present key, with: Entities::SSHKey
            else
    
              render_validation_error!(key)
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Delete an SSH key from the currently authenticated user' do
            success Entities::SSHKey
          end
          params do
            requires :key_id, type: Integer, desc: 'The ID of the SSH key'
          end
          delete "keys/:key_id" do
            key = current_user.keys.find_by(id: params[:key_id])
            not_found!('Key') unless key
    
            present key.destroy, with: Entities::SSHKey
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc "Get the currently authenticated user's email addresses" do
            success Entities::Email
          end
    
          params do
            use :pagination
          end
    
            present paginate(current_user.emails), with: Entities::Email
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Get a single email address owned by the currently authenticated user' do
            success Entities::Email
          end
          params do
            requires :email_id, type: Integer, desc: 'The ID of the email'
          end
          get "emails/:email_id" do
            email = current_user.emails.find_by(id: params[:email_id])
            not_found!('Email') unless email
    
    
            present email, with: Entities::Email
          end
    
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Add new email address to the currently authenticated user' do
            success Entities::Email
          end
          params do
            requires :email, type: String, desc: 'The new email'
          end
    
    Robert Schilling's avatar
    Robert Schilling committed
            email = current_user.emails.new(declared_params)
    
    
            if email.save
              NotificationService.new.new_email(email)
              present email, with: Entities::Email
            else
              render_validation_error!(email)
            end
          end
    
    
    Robert Schilling's avatar
    Robert Schilling committed
          desc 'Delete an email address from the currently authenticated user'
          params do
            requires :email_id, type: Integer, desc: 'The ID of the email'
          end
          delete "emails/:email_id" do
            email = current_user.emails.find_by(id: params[:email_id])
            not_found!('Email') unless email
    
    Robert Schilling's avatar
    Robert Schilling committed
            email.destroy
            current_user.update_secondary_emails!