使用 cURL 使用用户名和密码登录 Gitlab [英] Login to Gitlab with username and password using cURL

查看:151
本文介绍了使用 cURL 使用用户名和密码登录 Gitlab的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

为了在 Docker 中针对 Gitlab 实例测试命令行工具,我想使用用户名和密码登录 Gitlab 并获取创建的会话来验证我的 API 请求.

For testing a command line tool against a Gitlab instance in Docker, I would like to login to Gitlab using username and password and grab the created session to authenticate my API requests.

因此我做了以下事情:

  1. 使用 curl -i http://localhost:8080/users/sign_in -s
  2. 卷曲用户登录页面
  3. 从标题中获取 _gitlab_session
  4. 从登录表单中获取 authenticity_token
  5. 发送第二个 curl 请求


curl 'http://localhost:8080/users/sign_in' 
  -H "_gitlab_session=${cookie}" 
  -H 'Origin: http://localhost:8080' 
  -H 'Content-Type: application/x-www-form-urlencoded' 
  -H 'Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8' 
  -H 'Referer: http://localhost:8080/users/sign_in' 
  -H 'Connection: keep-alive' 
  --data-urlencode "authenticity_token=${token}" 
  --data-urlencode "user[login]=root" 
  --data-urlencode "user[password]=12341234" 
  --data-urlencode "user[remember_me]=0"

但我得到的不是有效的用户登录,而是

But instead of a valid user login, I get a

422 - The change you requested was rejected.

在日志文件中,我看到了

And in the logfiles, I see

==> gitlab-rails/production.log <==
Started POST "/users/sign_in" for 172.17.0.1 at 2017-12-23 00:22:16 +0000
Processing by SessionsController#create as HTML
Parameters: {"authenticity_token"=>"[FILTERED]", "user"=>{"login"=>"root", "password"=>"[FILTERED]", "remember_me"=>"0"}}
Can't verify CSRF token authenticity
Completed 422 Unprocessable Entity in 125ms (ActiveRecord: 7.8ms)

==> gitlab-rails/production_json.log <==
{"method":"POST","path":"/users/sign_in","format":"html","controller":"SessionsController",  
"action":"create","status":422,"error":" 
ActionController::InvalidAuthenticityToken:ActionController::InvalidAuthenticityToken",
"duration":126.29,"view":0.0,"db":7.78,"time":"2017-12-23T00:22:16.039Z",
"params":{"authenticity_token":"[FILTERED]","user":{"login":"root","password":"
[FILTERED]","remember_me":"0"}},"remote_ip":"172.17.0.1",
"user_id":1,"username":"root"}

==> gitlab-rails/production.log <==

ActionController::InvalidAuthenticityToken (ActionController::InvalidAuthenticityToken):
  lib/gitlab/middleware/multipart.rb:93:in `call'
  lib/gitlab/request_profiler/middleware.rb:14:in `call'
  lib/gitlab/middleware/go.rb:18:in `call'
  lib/gitlab/etag_caching/middleware.rb:11:in `call'
  lib/gitlab/middleware/read_only.rb:31:in `call'
  lib/gitlab/request_context.rb:18:in `call'
  lib/gitlab/metrics/requests_rack_middleware.rb:27:in `call'

我假设我忘记在第二个请求中传递一个必要的成分 - 但我不知道是哪一个.

I assume that I forgot to pass a necessary ingredient in the second request - but I can't figure out, which one.

推荐答案

通过其他答案和评论的一些有用提示,我终于想出了这个解决方案:

With some helpful tips by other answers and comments, I finally came up with this solution:

gitlab_host="http://localhost:8080"
gitlab_user="root"
gitlab_password="12341234"

# curl for the login page to get a session cookie and the sources with the auth tokens
body_header=$(curl -c cookies.txt -i "${gitlab_host}/users/sign_in" -s)

# grep the auth token for the user login for
#   not sure whether another token on the page will work, too - there are 3 of them
csrf_token=$(echo $body_header | perl -ne 'print "$1
" if /new_user.*?authenticity_token"[[:blank:]]value="(.+?)"/' | sed -n 1p)

# send login credentials with curl, using cookies and token from previous request
curl -b cookies.txt -c cookies.txt -i "${gitlab_host}/users/sign_in" 
    --data "user[login]=${gitlab_user}&user[password]=${gitlab_password}" 
    --data-urlencode "authenticity_token=${csrf_token}"

# send curl GET request to personal access token page to get auth token
body_header=$(curl -H 'user-agent: curl' -b cookies.txt -i "${gitlab_host}/profile/personal_access_tokens" -s)
csrf_token=$(echo $body_header | perl -ne 'print "$1
" if /authenticity_token"[[:blank:]]value="(.+?)"/' | sed -n 1p)

# curl POST request to send the "generate personal access token form"
# the response will be a redirect, so we have to follow using `-L`
body_header=$(curl -L -b cookies.txt "${gitlab_host}/profile/personal_access_tokens" 
    --data-urlencode "authenticity_token=${csrf_token}" 
    --data 'personal_access_token[name]=golab-generated&personal_access_token[expires_at]=&personal_access_token[scopes][]=api')

# Scrape the personal access token from the response HTML
personal_access_token=$(echo $body_header | perl -ne 'print "$1
" if /created-personal-access-token"[[:blank:]]value="(.+?)"/' | sed -n 1p)

根据 GitLab API 文档,您现在可以使用会话 cookie 来验证 API 请求:

According to the GitLab API documentation you can now use the session cookie to authenticate API requests:

curl --header "Private-Token: ${personal_access_token}" https://gitlab.example.com/api/v4/projects

一些提示:

  • 我首先对 curl -c 文件(从标头读取 cookie 并将它们写入文件)与 curl -b 文件(其中使用文件中的 cookie 并将它们与请求一起发送)
  • 不幸的是,我没有找到适用于 sed 的正则表达式,所以我必须在这里使用 perl.
  • 在 Chrome 中使用开发者控制台并将 POST 请求复制为 curl 命令非常有帮助:https://www.alexkras.com/copy-any-api-call-as-curl-request-with-chrome-developer-tools/
  • I first got confused with the curl -c file (which reads cookies from the headers and writes them to the file) vs. the curl -b file (which uses the cookies in the file and sends them with the request)
  • Unfortunately I did not find a regex that worked with sed so I have to use perl here.
  • Using the developer console in Chrome and copying a POST request as curl command was very helpful: https://www.alexkras.com/copy-any-api-call-as-curl-request-with-chrome-developer-tools/

这篇关于使用 cURL 使用用户名和密码登录 Gitlab的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆