# -*- coding: utf-8 -*- # # Copyright (C) 2014 Mika Mäenpää , # Tampere University of Technology # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with this program. If not, see . import os import pickle import tempfile import json import unittest from httmock import HTTMock # noqa from httmock import response # noqa from httmock import urlmatch # noqa import requests import gitlab from gitlab import * # noqa from gitlab.v4.objects import * # noqa import pytest valid_config = b"""[global] default = one ssl_verify = true timeout = 2 [one] url = http://one.url private_token = ABCDEF """ class TestSanitize(unittest.TestCase): def test_do_nothing(self): assert 1 == gitlab._sanitize(1) assert 1.5 == gitlab._sanitize(1.5) assert "foo" == gitlab._sanitize("foo") def test_slash(self): assert "foo%2Fbar" == gitlab._sanitize("foo/bar") def test_dict(self): source = {"url": "foo/bar", "id": 1} expected = {"url": "foo%2Fbar", "id": 1} assert expected == gitlab._sanitize(source) class TestGitlabList(unittest.TestCase): def setUp(self): self.gl = Gitlab( "http://localhost", private_token="private_token", api_version=4 ) def test_build_list(self): @urlmatch(scheme="http", netloc="localhost", path="/api/v4/tests", method="get") def resp_1(url, request): headers = { "content-type": "application/json", "X-Page": 1, "X-Next-Page": 2, "X-Per-Page": 1, "X-Total-Pages": 2, "X-Total": 2, "Link": ( ";" ' rel="next"' ), } content = '[{"a": "b"}]' return response(200, content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/tests", method="get", query=r".*page=2", ) def resp_2(url, request): headers = { "content-type": "application/json", "X-Page": 2, "X-Next-Page": 2, "X-Per-Page": 1, "X-Total-Pages": 2, "X-Total": 2, } content = '[{"c": "d"}]' return response(200, content, headers, None, 5, request) with HTTMock(resp_1): obj = self.gl.http_list("/tests", as_list=False) assert len(obj) == 2 assert obj._next_url == "http://localhost/api/v4/tests?per_page=1&page=2" assert obj.current_page == 1 assert obj.prev_page == None assert obj.next_page == 2 assert obj.per_page == 1 assert obj.total_pages == 2 assert obj.total == 2 with HTTMock(resp_2): l = list(obj) assert len(l) == 2 assert l[0]["a"] == "b" assert l[1]["c"] == "d" def test_all_omitted_when_as_list(self): @urlmatch(scheme="http", netloc="localhost", path="/api/v4/tests", method="get") def resp(url, request): headers = { "content-type": "application/json", "X-Page": 2, "X-Next-Page": 2, "X-Per-Page": 1, "X-Total-Pages": 2, "X-Total": 2, } content = '[{"c": "d"}]' return response(200, content, headers, None, 5, request) with HTTMock(resp): result = self.gl.http_list("/tests", as_list=False, all=True) assert isinstance(result, GitlabList) class TestGitlabHttpMethods(unittest.TestCase): def setUp(self): self.gl = Gitlab( "http://localhost", private_token="private_token", api_version=4 ) def test_build_url(self): r = self.gl._build_url("http://localhost/api/v4") assert r == "http://localhost/api/v4" r = self.gl._build_url("https://localhost/api/v4") assert r == "https://localhost/api/v4" r = self.gl._build_url("/projects") assert r == "http://localhost/api/v4/projects" def test_http_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '[{"name": "project1"}]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): http_r = self.gl.http_request("get", "/projects") http_r.json() assert http_r.status_code == 200 def test_http_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="get" ) def resp_cont(url, request): content = {"Here is wh it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_request("get", "/not_there") def test_get_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '{"name": "project1"}' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_get("/projects") assert isinstance(result, dict) assert result["name"] == "project1" def test_get_request_raw(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/octet-stream"} content = "content" return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_get("/projects") assert result.content.decode("utf-8") == "content" def test_get_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="get" ) def resp_cont(url, request): content = {"Here is wh it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_get("/not_there") def test_get_request_invalid_data(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '["name": "project1"]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabParsingError): self.gl.http_get("/projects") def test_list_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/json", "X-Total": 1} content = '[{"name": "project1"}]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_list("/projects", as_list=True) assert isinstance(result, list) assert len(result) == 1 with HTTMock(resp_cont): result = self.gl.http_list("/projects", as_list=False) assert isinstance(result, GitlabList) assert len(result) == 1 with HTTMock(resp_cont): result = self.gl.http_list("/projects", all=True) assert isinstance(result, list) assert len(result) == 1 def test_list_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="get" ) def resp_cont(url, request): content = {"Here is why it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_list("/not_there") def test_list_request_invalid_data(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="get" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '["name": "project1"]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabParsingError): self.gl.http_list("/projects") def test_post_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="post" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '{"name": "project1"}' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_post("/projects") assert isinstance(result, dict) assert result["name"] == "project1" def test_post_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="post" ) def resp_cont(url, request): content = {"Here is wh it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_post("/not_there") def test_post_request_invalid_data(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="post" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '["name": "project1"]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabParsingError): self.gl.http_post("/projects") def test_put_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="put" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '{"name": "project1"}' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_put("/projects") assert isinstance(result, dict) assert result["name"] == "project1" def test_put_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="put" ) def resp_cont(url, request): content = {"Here is wh it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_put("/not_there") def test_put_request_invalid_data(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="put" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = '["name": "project1"]' return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabParsingError): self.gl.http_put("/projects") def test_delete_request(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects", method="delete" ) def resp_cont(url, request): headers = {"content-type": "application/json"} content = "true" return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): result = self.gl.http_delete("/projects") assert isinstance(result, requests.Response) assert result.json() == True def test_delete_request_404(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/not_there", method="delete" ) def resp_cont(url, request): content = {"Here is wh it failed"} return response(404, content, {}, None, 5, request) with HTTMock(resp_cont): with pytest.raises(GitlabHttpError): self.gl.http_delete("/not_there") class TestGitlabStripBaseUrl(unittest.TestCase): def setUp(self): self.gl = Gitlab( "http://localhost/", private_token="private_token", api_version=4 ) def test_strip_base_url(self): assert self.gl.url == "http://localhost" def test_strip_api_url(self): assert self.gl.api_url == "http://localhost/api/v4" def test_build_url(self): r = self.gl._build_url("/projects") assert r == "http://localhost/api/v4/projects" class TestGitlabAuth(unittest.TestCase): def test_invalid_auth_args(self): with pytest.raises(ValueError): Gitlab( "http://localhost", api_version="4", private_token="private_token", oauth_token="bearer", ) with pytest.raises(ValueError): Gitlab( "http://localhost", api_version="4", oauth_token="bearer", http_username="foo", http_password="bar", ) with pytest.raises(ValueError): Gitlab( "http://localhost", api_version="4", private_token="private_token", http_password="bar", ) with pytest.raises(ValueError): Gitlab( "http://localhost", api_version="4", private_token="private_token", http_username="foo", ) def test_private_token_auth(self): gl = Gitlab("http://localhost", private_token="private_token", api_version="4") assert gl.private_token == "private_token" assert gl.oauth_token == None assert gl.job_token == None assert gl._http_auth == None assert "Authorization" not in gl.headers assert gl.headers["PRIVATE-TOKEN"] == "private_token" assert "JOB-TOKEN" not in gl.headers def test_oauth_token_auth(self): gl = Gitlab("http://localhost", oauth_token="oauth_token", api_version="4") assert gl.private_token == None assert gl.oauth_token == "oauth_token" assert gl.job_token == None assert gl._http_auth == None assert gl.headers["Authorization"] == "Bearer oauth_token" assert "PRIVATE-TOKEN" not in gl.headers assert "JOB-TOKEN" not in gl.headers def test_job_token_auth(self): gl = Gitlab("http://localhost", job_token="CI_JOB_TOKEN", api_version="4") assert gl.private_token == None assert gl.oauth_token == None assert gl.job_token == "CI_JOB_TOKEN" assert gl._http_auth == None assert "Authorization" not in gl.headers assert "PRIVATE-TOKEN" not in gl.headers assert gl.headers["JOB-TOKEN"] == "CI_JOB_TOKEN" def test_http_auth(self): gl = Gitlab( "http://localhost", private_token="private_token", http_username="foo", http_password="bar", api_version="4", ) assert gl.private_token == "private_token" assert gl.oauth_token == None assert gl.job_token == None assert isinstance(gl._http_auth, requests.auth.HTTPBasicAuth) assert gl.headers["PRIVATE-TOKEN"] == "private_token" assert "Authorization" not in gl.headers class TestGitlab(unittest.TestCase): def setUp(self): self.gl = Gitlab( "http://localhost", private_token="private_token", ssl_verify=True, api_version=4, ) def test_pickability(self): original_gl_objects = self.gl._objects pickled = pickle.dumps(self.gl) unpickled = pickle.loads(pickled) assert isinstance(unpickled, Gitlab) assert hasattr(unpickled, "_objects") assert unpickled._objects == original_gl_objects def test_token_auth(self, callback=None): name = "username" id_ = 1 @urlmatch(scheme="http", netloc="localhost", path="/api/v4/user", method="get") def resp_cont(url, request): headers = {"content-type": "application/json"} content = '{{"id": {0:d}, "username": "{1:s}"}}'.format(id_, name).encode( "utf-8" ) return response(200, content, headers, None, 5, request) with HTTMock(resp_cont): self.gl.auth() assert self.gl.user.username == name assert self.gl.user.id == id_ assert isinstance(self.gl.user, CurrentUser) def test_hooks(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/hooks/1", method="get" ) def resp_get_hook(url, request): headers = {"content-type": "application/json"} content = '{"url": "testurl", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_hook): data = self.gl.hooks.get(1) assert isinstance(data, Hook) assert data.url == "testurl" assert data.id == 1 def test_projects(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1", method="get" ) def resp_get_project(url, request): headers = {"content-type": "application/json"} content = '{"name": "name", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_project): data = self.gl.projects.get(1) assert isinstance(data, Project) assert data.name == "name" assert data.id == 1 def test_project_environments(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1$", method="get" ) def resp_get_project(url, request): headers = {"content-type": "application/json"} content = '{"name": "name", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/environments/1", method="get", ) def resp_get_environment(url, request): headers = {"content-type": "application/json"} content = '{"name": "environment_name", "id": 1, "last_deployment": "sometime"}'.encode( "utf-8" ) return response(200, content, headers, None, 5, request) with HTTMock(resp_get_project, resp_get_environment): project = self.gl.projects.get(1) environment = project.environments.get(1) assert isinstance(environment, ProjectEnvironment) assert environment.id == 1 assert environment.last_deployment == "sometime" assert environment.name == "environment_name" def test_project_additional_statistics(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1$", method="get" ) def resp_get_project(url, request): headers = {"content-type": "application/json"} content = '{"name": "name", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/statistics", method="get", ) def resp_get_environment(url, request): headers = {"content-type": "application/json"} content = """{"fetches": {"total": 50, "days": [{"count": 10, "date": "2018-01-10"}]}}""".encode( "utf-8" ) return response(200, content, headers, None, 5, request) with HTTMock(resp_get_project, resp_get_environment): project = self.gl.projects.get(1) statistics = project.additionalstatistics.get() assert isinstance(statistics, ProjectAdditionalStatistics) assert statistics.fetches["total"] == 50 def test_project_issues_statistics(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1$", method="get" ) def resp_get_project(url, request): headers = {"content-type": "application/json"} content = '{"name": "name", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/issues_statistics", method="get", ) def resp_get_environment(url, request): headers = {"content-type": "application/json"} content = """{"statistics": {"counts": {"all": 20, "closed": 5, "opened": 15}}}""".encode( "utf-8" ) return response(200, content, headers, None, 5, request) with HTTMock(resp_get_project, resp_get_environment): project = self.gl.projects.get(1) statistics = project.issuesstatistics.get() assert isinstance(statistics, ProjectIssuesStatistics) assert statistics.statistics["counts"]["all"] == 20 def test_issues(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/issues", method="get" ) def resp_get_issue(url, request): headers = {"content-type": "application/json"} content = '[{"name": "name", "id": 1}, ' '{"name": "other_name", "id": 2}]' content = content.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_issue): data = self.gl.issues.list() assert data[1].id == 2 assert data[1].name == "other_name" @urlmatch(scheme="http", netloc="localhost", path="/api/v4/users/1", method="get") def resp_get_user(self, url, request): headers = {"content-type": "application/json"} content = ( '{"name": "name", "id": 1, "password": "password", ' '"username": "username", "email": "email"}' ) content = content.encode("utf-8") return response(200, content, headers, None, 5, request) def test_users(self): with HTTMock(self.resp_get_user): user = self.gl.users.get(1) assert isinstance(user, User) assert user.name == "name" assert user.id == 1 def test_user_memberships(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/users/1/memberships", method="get", ) def resp_get_user_memberships(url, request): headers = {"content-type": "application/json"} content = """[ { "source_id": 1, "source_name": "Project one", "source_type": "Project", "access_level": "20" }, { "source_id": 3, "source_name": "Group three", "source_type": "Namespace", "access_level": "20" } ]""" content = content.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_user_memberships): user = self.gl.users.get(1, lazy=True) memberships = user.memberships.list() assert isinstance(memberships[0], UserMembership) assert memberships[0].source_type == "Project" def test_user_status(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/users/1/status", method="get", ) def resp_get_user_status(url, request): headers = {"content-type": "application/json"} content = '{"message": "test", "message_html": "

Message

", "emoji": "thumbsup"}' content = content.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(self.resp_get_user): user = self.gl.users.get(1) with HTTMock(resp_get_user_status): status = user.status.get() assert isinstance(status, UserStatus) assert status.message == "test" assert status.emoji == "thumbsup" def test_todo(self): with open(os.path.dirname(__file__) + "/data/todo.json", "r") as json_file: todo_content = json_file.read() json_content = json.loads(todo_content) encoded_content = todo_content.encode("utf-8") @urlmatch(scheme="http", netloc="localhost", path="/api/v4/todos", method="get") def resp_get_todo(url, request): headers = {"content-type": "application/json"} return response(200, encoded_content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/todos/102/mark_as_done", method="post", ) def resp_mark_as_done(url, request): headers = {"content-type": "application/json"} single_todo = json.dumps(json_content[0]) content = single_todo.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_todo): todo = self.gl.todos.list()[0] assert isinstance(todo, Todo) assert todo.id == 102 assert todo.target_type == "MergeRequest" assert todo.target["assignee"]["username"] == "root" with HTTMock(resp_mark_as_done): todo.mark_as_done() def test_todo_mark_all_as_done(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/todos/mark_as_done", method="post", ) def resp_mark_all_as_done(url, request): headers = {"content-type": "application/json"} return response(204, {}, headers, None, 5, request) with HTTMock(resp_mark_all_as_done): self.gl.todos.mark_all_as_done() def test_deployment(self): content = '{"id": 42, "status": "success", "ref": "master"}' json_content = json.loads(content) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/deployments", method="post", ) def resp_deployment_create(url, request): headers = {"content-type": "application/json"} return response(200, json_content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/deployments/42", method="put", ) def resp_deployment_update(url, request): headers = {"content-type": "application/json"} return response(200, json_content, headers, None, 5, request) with HTTMock(resp_deployment_create): deployment = self.gl.projects.get(1, lazy=True).deployments.create( { "environment": "Test", "sha": "1agf4gs", "ref": "master", "tag": False, "status": "created", } ) assert deployment.id == 42 assert deployment.status == "success" assert deployment.ref == "master" with HTTMock(resp_deployment_update): json_content["status"] = "failed" deployment.status = "failed" deployment.save() assert deployment.status == "failed" def test_user_activate_deactivate(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/users/1/activate", method="post", ) def resp_activate(url, request): headers = {"content-type": "application/json"} return response(201, {}, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/users/1/deactivate", method="post", ) def resp_deactivate(url, request): headers = {"content-type": "application/json"} return response(201, {}, headers, None, 5, request) with HTTMock(resp_activate), HTTMock(resp_deactivate): self.gl.users.get(1, lazy=True).activate() self.gl.users.get(1, lazy=True).deactivate() def test_update_submodule(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1$", method="get" ) def resp_get_project(url, request): headers = {"content-type": "application/json"} content = '{"name": "name", "id": 1}'.encode("utf-8") return response(200, content, headers, None, 5, request) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/repository/submodules/foo%2Fbar", method="put", ) def resp_update_submodule(url, request): headers = {"content-type": "application/json"} content = """{ "id": "ed899a2f4b50b4370feeea94676502b42383c746", "short_id": "ed899a2f4b5", "title": "Message", "author_name": "Author", "author_email": "author@example.com", "committer_name": "Author", "committer_email": "author@example.com", "created_at": "2018-09-20T09:26:24.000-07:00", "message": "Message", "parent_ids": [ "ae1d9fb46aa2b07ee9836d49862ec4e2c46fbbba" ], "committed_date": "2018-09-20T09:26:24.000-07:00", "authored_date": "2018-09-20T09:26:24.000-07:00", "status": null}""" content = content.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_get_project): project = self.gl.projects.get(1) assert isinstance(project, Project) assert project.name == "name" assert project.id == 1 with HTTMock(resp_update_submodule): ret = project.update_submodule( submodule="foo/bar", branch="master", commit_sha="4c3674f66071e30b3311dac9b9ccc90502a72664", commit_message="Message", ) assert isinstance(ret, dict) assert ret["message"] == "Message" assert ret["id"] == "ed899a2f4b50b4370feeea94676502b42383c746" def test_applications(self): content = '{"name": "test_app", "redirect_uri": "http://localhost:8080", "scopes": ["api", "email"]}' json_content = json.loads(content) @urlmatch( scheme="http", netloc="localhost", path="/api/v4/applications", method="post", ) def resp_application_create(url, request): headers = {"content-type": "application/json"} return response(200, json_content, headers, None, 5, request) with HTTMock(resp_application_create): application = self.gl.applications.create( { "name": "test_app", "redirect_uri": "http://localhost:8080", "scopes": ["api", "email"], "confidential": False, } ) assert application.name == "test_app" assert application.redirect_uri == "http://localhost:8080" assert application.scopes == ["api", "email"] def test_deploy_tokens(self): @urlmatch( scheme="http", netloc="localhost", path="/api/v4/projects/1/deploy_tokens", method="post", ) def resp_deploy_token_create(url, request): headers = {"content-type": "application/json"} content = """{ "id": 1, "name": "test_deploy_token", "username": "custom-user", "expires_at": "2022-01-01T00:00:00.000Z", "token": "jMRvtPNxrn3crTAGukpZ", "scopes": [ "read_repository" ]}""" content = content.encode("utf-8") return response(200, content, headers, None, 5, request) with HTTMock(resp_deploy_token_create): deploy_token = self.gl.projects.get(1, lazy=True).deploytokens.create( { "name": "test_deploy_token", "expires_at": "2022-01-01T00:00:00.000Z", "username": "custom-user", "scopes": ["read_repository"], } ) assert isinstance(deploy_token, ProjectDeployToken) assert deploy_token.id == 1 assert deploy_token.expires_at == "2022-01-01T00:00:00.000Z" assert deploy_token.username == "custom-user" assert deploy_token.scopes == ["read_repository"] def _default_config(self): fd, temp_path = tempfile.mkstemp() os.write(fd, valid_config) os.close(fd) return temp_path def test_from_config(self): config_path = self._default_config() gitlab.Gitlab.from_config("one", [config_path]) os.unlink(config_path) def test_subclass_from_config(self): class MyGitlab(gitlab.Gitlab): pass config_path = self._default_config() gl = MyGitlab.from_config("one", [config_path]) assert isinstance(gl, MyGitlab) os.unlink(config_path)