Merge branch 'develop' into hotfix/get_tickets

This commit is contained in:
Sokurov Idar
2021-05-29 23:59:51 +03:00
35 changed files with 18586 additions and 273 deletions

View File

@@ -0,0 +1,8 @@
{
"presets": ["@babel/preset-env", "@babel/preset-react"],
"plugins": [
["@babel/plugin-transform-runtime", {
"regenerator": true
}], "@babel/plugin-syntax-jsx"
]
}

View File

@@ -0,0 +1,98 @@
import React from "react";
import {render, unmountComponentAtNode} from "react-dom";
import {act} from "react-dom/test-utils";
import {Table} from "../src/control";
import * as test_data from "./test_users.json"
import axios from "axios";
import MockAdapter from "axios-mock-adapter";
let mock = null
let container = null
beforeEach(() => {
mock = new MockAdapter(axios);
mock.onGet('/api/users').reply(200, test_data)
container = document.createElement('div')
container.id = "table"
document.body.appendChild(container)
})
afterEach(() => {
unmountComponentAtNode(container);
mock.restore()
container.remove();
container = null;
});
it("has no rows without axios request", () => {
act(() => {
render(<Table/>, container);
});
let tbody = container.querySelector("#tbody")
expect(tbody.getElementsByTagName('tr').length).toBe(0);
});
it("shows valid number of free workplaces", async () => {
await act(async () => {
render(<Table/>, container)
})
let element = container.querySelector('#licences_remaining')
let licences = Number(element.innerHTML.replace(/Свободных мест: /, ''))
expect(licences).toEqual(1)
});
it("Pretext must be deleted on render", () => {
act(() => {
render(<Table/>, container)
})
expect(document.body).not.toContain(container.querySelector('#loading'))
});
it("has valid number of table rows with axios request", async () => {
await act(async () => {
render(<Table/>, container)
})
let tbody = container.querySelector("#tbody")
expect(tbody.getElementsByTagName('tr').length)
.toEqual(test_data.users.length + test_data.zendesk_users.length)
});
it("show valid number for engineers and light agents", async () => {
await act(async () => {
render(<Table/>, container)
})
let engineers = container.querySelector('#engineers')
let agents = container.querySelector('#agents')
expect(Number(engineers.textContent)).toEqual(test_data.engineers)
expect(Number(agents.textContent)).toEqual(test_data.light_agents)
});
it("called one request on mount", async () => {
let requests = jest.spyOn(Table.prototype, "getUsers")
await act(async () => {
render(<Table/>, container)
})
expect(requests).toHaveBeenCalledTimes(1)
requests.mockRestore()
})
it("checkbox count equals users from db count", async () => {
await act(async () => {
render(<Table/>, container)
})
let tbody = container.querySelector("#tbody")
let checkboxes = tbody.querySelectorAll("input[type='checkbox']")
let users = test_data.users
expect(checkboxes.length).toEqual(users.length)
})
it("requests occur every one minute", async () => {
jest.useFakeTimers()
let requests = jest.spyOn(Table.prototype, "getUsers")
await act(async () => {
render(<Table/>, container)
})
jest.advanceTimersByTime(60000)
expect(requests).toHaveBeenCalledTimes(2)
jest.useRealTimers()
requests.mockRestore()
})

View File

@@ -0,0 +1,32 @@
{
"users": [
{
"user": {
"email": "123@test.ru"
},
"id": 2,
"name": "UserForAccessTest",
"zendesk_role": "light_agent"
}
],
"engineers": 2,
"light_agents": 2,
"zendesk_users": [
{
"name": "Степаненко Ольга s101",
"zendesk_role": "engineer",
"email": "stepanenko_olga@mail.ru"
},
{
"name": "TEST",
"zendesk_role": "engineer",
"email": "akovalev1305@gmail.com"
},
{
"name": "Vasua",
"zendesk_role": "light_agent",
"email": "krav-88@mail.ru"
}
],
"max_agents": 3
}

View File

@@ -0,0 +1,9 @@
module.exports = {
verbose: true,
testPathIgnorePatterns: [
"./node_modules/"
],
roots: [
"./__tests__"
],
}

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,28 @@
{
"name": "control_page",
"version": "1.0.0",
"description": "",
"main": "../static/main/js/control_page/dist/index_bundle.js",
"scripts": {
"test": "jest"
},
"author": "",
"license": "ISC",
"dependencies": {
"@babel/cli": "^7.13.16",
"@babel/core": "^7.13.16",
"@babel/plugin-transform-runtime": "^7.13.15",
"@babel/preset-env": "^7.13.15",
"@babel/preset-react": "^7.13.13",
"axios": "^0.21.1",
"axios-mock-adapter": "^1.19.0",
"babel-loader": "^8.2.2",
"jest": "^26.6.3",
"jsx": "^0.9.89",
"react": "^17.0.2",
"react-dom": "^17.0.2",
"save-dev": "0.0.1-security",
"webpack": "^5.36.2",
"webpack-cli": "^4.6.0"
}
}

View File

@@ -0,0 +1,183 @@
import React from "react";
import axios from "axios";
function FreeWorkplaces(props) {
return (
<div className="new-section">
<p className="row page-description" id="licences_remaining">Свободных мест: {props.count}</p>
</div>
)
}
function WorkersCount(props) {
return (
<div className="row justify-content-center new-section d-flex align-items-center">
<div className="col-5">
<div className="info">
<div className="info-row">
<div className="info-target px-4">Инженеров:</div>
<div className="info-quantity">
<div className="status-circle-small light-green"></div>
<span className="info-quantity-value" id="engineers">{props.engineers}</span>
</div>
</div>
<div className="info-row">
<div className="info-target px-4">Легких агентов:</div>
<div className="info-quantity">
<div className="status-circle-small light-yellow"></div>
<span className="info-quantity-value" id="agents">{props.light_agents}</span>
</div>
</div>
</div>
</div>
<div className="col-5">
<button type="submit" name="engineer" className="btn default-button btn-warning btn-block btn-sm py-3">
Назначить выбранных на роль инженера
</button>
<button type="submit" name="light_agent" className="btn default-button btn-warning btn-block btn-sm py-3">
Назначить выбранных на роль легкого агента
</button>
</div>
</div>
)
}
class ModelUserTableRow extends React.Component {
render() {
return (
<tr className={"table-dark"}>
<td>
<input
type="checkbox"
value={this.props.user.id}
className="form-check-input"
name="users"
/>
</td>
<td>
<a href="#">{this.props.user.name}</a>
</td>
<td>{this.props.user.user.email}</td>
<td>{this.props.user.zendesk_role}</td>
</tr>
);
}
}
class ModelUserTableRows extends React.Component {
render() {
return (
this.props.users.map((user, key) => (
<ModelUserTableRow user={user} key={key} />
))
);
}
}
class ZendeskUserTableRow extends React.Component {
render() {
return (
<tr className={"table-secondary text-secondary"}>
<td></td>
<td>
<a href="#" style={{ color: "grey", fontStyle: "italic" }}>
{this.props.user.name}
</a>
</td>
<td style={{ color: "grey", fontStyle: "italic" }}>
{this.props.user.email}
</td>
<td style={{ color: "grey", fontStyle: "italic" }}>
{this.props.user.zendesk_role}
</td>
</tr>
);
}
}
class ZendeskUserTableRows extends React.Component {
render() {
return (
this.props.users.map((user, key) => (
<ZendeskUserTableRow user={user} key={key} />
))
)
}
}
export class Table extends React.Component {
constructor(props) {
super(props);
this.state = {
users: [],
engineers: null,
light_agents: null,
zendesk_users: [],
max_agents: null,
renderLoad: true
};
}
async getUsers() {
await axios.get("/api/users").then((response) => {
this.setState({
users: response.data.users,
engineers: response.data.engineers,
light_agents: response.data.light_agents,
zendesk_users: response.data.zendesk_users,
max_agents: response.data.max_agents,
renderLoad: false
});
return response
}).catch(reason => {
console.log(reason)
});
}
componentDidMount() {
this.getUsers().then(() => {})
.catch(reason => {
console.log(reason)
});
this.interval = setInterval(() => {
this.getUsers().catch(reason => {
console.log(reason)
})
}, 60000);
}
componentWillUnmount() {
clearInterval(this.interval);
}
render() {
return (
<div>
<FreeWorkplaces count={Math.max(this.state.max_agents - this.state.engineers, 0)}/>
<table className="table table-dark">
<thead>
<tr>
<th>
<input
type="checkbox"
className="form-check-input"
id="head-checkbox"
/>
</th>
<th>Name</th>
<th>Email</th>
<th>Role</th>
</tr>
</thead>
<tbody id="tbody">
<ModelUserTableRows users={this.state.users}/>
<ZendeskUserTableRows users={this.state.zendesk_users}/>
</tbody>
</table>
{this.state.renderLoad === true ? <p id="loading">Данные загружаются...</p> : null}
<WorkersCount engineers={this.state.engineers} light_agents={this.state.light_agents}/>
</div>
);
}
}

View File

@@ -0,0 +1,17 @@
import {Table} from "./control"
import ReactDOM from "react-dom";
import React from "react";
function headCheckbox() {
let headCheckbox = document.getElementById("head-checkbox");
headCheckbox.addEventListener("click", () => {
let checkboxes = document.getElementsByName("users");
for (let checkbox of checkboxes)
checkbox.checked = headCheckbox.checked;
});
}
ReactDOM.render(<Table />, document.getElementById("table"));
headCheckbox();

View File

@@ -0,0 +1,31 @@
const path = require('path')
module.exports = {
entry: './src/index.js',
module: {
rules: [
{
test: /\.(js)$/,
exclude: path.resolve(__dirname, 'node_modules/'),
use: {
loader: 'babel-loader',
options: {
presets: ['@babel/preset-env', "@babel/preset-react"],
plugins: [["@babel/plugin-transform-runtime", {"regenerator": true}]],
}
}
}
]
},
resolve: {
extensions: [
'.js',
'.jsx'
]
},
output: {
path: path.resolve('../../static/main/js/control_page', 'dist'),
filename: 'index_bundle.js'
},
mode: 'development',
}

View File

@@ -6,6 +6,8 @@ from datetime import timedelta, date
from typing import Union, Optional
from django.contrib.auth import get_user_model
from django.contrib.auth.models import Permission
from django.contrib.contenttypes.models import ContentType
from django.core.exceptions import ObjectDoesNotExist
from django.core.handlers.wsgi import WSGIRequest
from django.http import HttpResponseRedirect, HttpResponsePermanentRedirect
@@ -123,6 +125,7 @@ def update_profile(user_profile: UserProfile) -> None:
:return: Обновленный, в соответствие с текущими данными в Zendesk, профиль пользователя
"""
user = zenpy.get_user(user_profile.user.email)
update_permission(user_profile, user)
user_profile.name = user.name
user_profile.role = user.role
user_profile.custom_role_id = user.custom_role_id if user.custom_role_id else 0
@@ -130,6 +133,52 @@ def update_profile(user_profile: UserProfile) -> None:
user_profile.save()
def update_permission(user_profile: UserProfile, user: ZenpyUser):
"""
Функция обновляет права доступа пользователя в БД.
:param user_profile: Профиль пользователя
:param user: Данные пользователя в Zendesk
"""
if user_profile.role != user.role:
user_profile.role = user.role
user_profile.save()
set_permission(user_profile.user)
del_permission(user_profile.user)
def set_permission(user: get_user_model()) -> None:
"""
Функция дает разрешение на просмотр страница администратора, если пользователь имеет роль admin.
:param user: Авторизованный пользователь (получает разрешение, имея роль "admin")
"""
if user.userprofile.role == 'admin':
content_type = ContentType.objects.get_for_model(UserProfile)
permission = Permission.objects.get(
codename='has_control_access',
content_type=content_type,
)
user.user_permissions.add(permission)
user.save()
def del_permission(user: get_user_model()) -> None:
"""
Функция забирает разрешение на просмотр страница администратора, если пользователь не имеет роль admin.
:param user: Авторизованный пользователь (теряет разрешение, не имея роль "admin")
"""
if user.userprofile.role == 'agent' and user.has_perm('main.has_control_access'):
content_type = ContentType.objects.get_for_model(UserProfile)
permission = Permission.objects.get(
codename='has_control_access',
content_type=content_type,
)
user.user_permissions.remove(permission)
user.save()
def check_user_exist(email: str) -> bool:
"""
Функция проверяет, существует ли пользователь.
@@ -180,6 +229,7 @@ def update_user_in_model(profile: UserProfile, zendesk_user: ZenpyUser) -> None:
:param zendesk_user: Данные пользователя в Zendesk
:return: Обновленный профиль пользователя
"""
update_permission(profile, zendesk_user)
profile.name = zendesk_user.name
profile.role = zendesk_user.role
profile.image = zendesk_user.photo['content_url'] if zendesk_user.photo else None
@@ -262,6 +312,7 @@ class DatabaseHandler(logging.Handler):
"""
Класс записи изменений ролей в базу данных.
"""
def __init__(self):
logging.Handler.__init__(self)
@@ -295,6 +346,7 @@ class CsvFormatter(logging.Formatter):
"""
Класс преобразования смены ролей пользователей в строковый формат.
"""
def __init__(self):
logging.Formatter.__init__(self)
@@ -346,7 +398,7 @@ def log(user: get_user_model(), admin: get_user_model() = None) -> None:
def set_session_params_for_work_page(request: WSGIRequest, count: int = None, is_confirm: bool = True) -> \
Union[HttpResponsePermanentRedirect, HttpResponseRedirect]:
Union[HttpResponsePermanentRedirect, HttpResponseRedirect]:
"""
Функция для страницы получения прав, устанавливает данные сессии о успешности запроса и количестве
назначенных тикетов.

View File

@@ -9,6 +9,8 @@
<title>{% block title %}{% endblock %}</title>
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/5.0.0-alpha1/css/bootstrap.min.css"
integrity="sha384-r4NyP46KrjDleawBgD5tp8Y7UzmLA05oM1iAEQ17CSuDqnUK2+k9luXQOfXJCJ4I" crossorigin="anonymous">
<link rel="icon" href="{% static 'main/favs/favicon.ico'%}" type="image/x-icon">
<style>
.bd-placeholder-img {

View File

@@ -42,6 +42,8 @@
href="{{ work_url }}">Запрос прав</a>
{% endif %}
<a class="btn btn-secondary" href="{% url 'logout' %}">Выйти</a>
<a class="btn btn-secondary"
href="https://actrl.cazzzer.com/docs/index.html"> Документация</a>
</div>
{% else %}
<div class="btn-group" role="group" aria-label="Basic example" style="margin-right: 9px">
@@ -59,6 +61,9 @@
class="btn btn-secondary"
{% endif %}
href="{{ registration_url }}">Зарегистрироваться</a>
{% url 'documentation' as documentation_url %}
<a class="btn btn-secondary"
href="{{ documentation_url }}">Документация</a>
</div>
{% endif %}
</nav>

View File

@@ -12,22 +12,12 @@
{% endblock %}
{% block extra_scripts %}
<script src="https://unpkg.com/react@17/umd/react.development.js" crossorigin></script>
<script src="https://unpkg.com/react-dom@17/umd/react-dom.development.js" crossorigin></script>
<script src="https://unpkg.com/babel-standalone@6/babel.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
<script src="{% static 'main/js/control.js'%}" type="text/babel"></script>
<script src="{% static 'main/js/notifications.js' %}"></script> {# Для #}
<script src="{% static 'modules/notifications/dist/notifications.js' %}"></script> {# Уведомлений #}
{% endblock%}
{% block content %}
<div class="container-md">
<div class="new-section">
<p class="row page-description" id="licences_remaining">Свободных Мест:</p>
</div>
{% for message in messages %}
<script>create_notification('{{message}}','','{{message.tags}}',2000)</script>
{% endfor %}
@@ -35,78 +25,21 @@
{% block form %}
<form method="post">
{% csrf_token %}
<div class="row justify-content-center new-section">
<div class="col-10">
<h3 class="py-4 text-center font-weight-bold">Список сотрудников</h3>
<h3 class="py-4 text-center font-weight-bold mb-0">Список сотрудников</h3>
{% block table %}
<table class="table table-dark">
<thead>
<th>
<input
type="checkbox"
class="form-check-input"
id="head-checkbox"
/>
</th>
<th>Name</th>
<th>Email</th>
<th>Role</th>
</thead>
<tbody id="tbody"></tbody>
</table>
<p id="loading">Данные загружаются...</p>
<div id="table"></div>
{% endblock %}
</div>
</div>
{% block count %}
<div class="row justify-content-center new-section">
<div class="col-5">
<div class="info">
<div class="info-row">
<div class="info-target px-4"><h6>Инженеров:</h6></div>
<div class="info-quantity">
<div class="status-circle-small light-green"></div>
<span class="info-quantity-value">{{ engineers }}</span>
</div>
</div>
<div class="info-row">
<div class="info-target px-4"><h6>Легких агентов:</h6></div>
<div class="info-quantity">
<div class="status-circle-small light-yellow"></div>
<span class="info-quantity-value">{{ light_agents }}</span>
</div>
</div>
</div>
</div>
{% endblock %}
{% block buttons %}
<div class="col-5">
<button type="submit" name="engineer" class="btn default-button btn-warning btn-block btn-sm py-3">
Назначить выбранных на роль инженера
</button>
<button type="submit" name="light_agent" class="btn default-button btn-warning btn-block btn-sm py-3">
Назначить выбранных на роль легкого агента
</button>
</div>
{% endblock %}
</div>
</form>
{% endblock %}
</div>
<script src="{% static 'main/js/control_page/dist/index_bundle.js' %}"></script>
{% endblock %}

View File

@@ -0,0 +1,252 @@
<!DOCTYPE html>
<html class="writer-html5" lang="ru" >
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Документация контроллера прав доступа &mdash; документация ZenDesk Access Controller v0.01</title>
<link rel="stylesheet" href="_static/css/theme.css" type="text/css" />
<link rel="stylesheet" href="_static/pygments.css" type="text/css" />
<link rel="stylesheet" href="_static/graphviz.css" type="text/css" />
<!--[if lt IE 9]>
<script src="_static/js/html5shiv.min.js"></script>
<![endif]-->
<script type="text/javascript" id="documentation_options" data-url_root="./" src="_static/documentation_options.js"></script>
<script src="_static/jquery.js"></script>
<script src="_static/underscore.js"></script>
<script src="_static/doctools.js"></script>
<script src="_static/translations.js"></script>
<script type="text/javascript" src="_static/js/theme.js"></script>
<link rel="index" title="Алфавитный указатель" href="genindex.html" />
<link rel="search" title="Поиск" href="search.html" />
<link rel="next" title="Документация пользователя" href="overview.html" />
</head>
<body class="wy-body-for-nav">
<div class="wy-grid-for-nav">
<nav data-toggle="wy-nav-shift" class="wy-nav-side">
<div class="wy-side-scroll">
<div class="wy-side-nav-search" >
<a href="#" class="icon icon-home"> ZenDesk Access Controller
</a>
<div role="search">
<form id="rtd-search-form" class="wy-form" action="search.html" method="get">
<input type="text" name="q" placeholder="Поиск в документации" />
<input type="hidden" name="check_keywords" value="yes" />
<input type="hidden" name="area" value="default" />
</form>
</div>
</div>
<div class="wy-menu wy-menu-vertical" data-spy="affix" role="navigation" aria-label="main navigation">
<p class="caption"><span class="caption-text">Contents:</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="overview.html">Документация пользователя</a></li>
<li class="toctree-l1"><a class="reference internal" href="code.html">Документация разработчика</a></li>
<li class="toctree-l1"><a class="reference internal" href="readme.html">READ.me</a></li>
<li class="toctree-l1"><a class="reference internal" href="todo.html">Что необходимо доделать?</a></li>
</ul>
</div>
</div>
</nav>
<section data-toggle="wy-nav-shift" class="wy-nav-content-wrap">
<nav class="wy-nav-top" aria-label="top navigation">
<i data-toggle="wy-nav-top" class="fa fa-bars"></i>
<a href="#">ZenDesk Access Controller</a>
</nav>
<div class="wy-nav-content">
<div class="rst-content">
<div role="navigation" aria-label="breadcrumbs navigation">
<ul class="wy-breadcrumbs">
<li><a href="#" class="icon icon-home"></a> &raquo;</li>
<li>Документация контроллера прав доступа</li>
<li class="wy-breadcrumbs-aside">
<a href="_sources/index.rst.txt" rel="nofollow"> Просмотреть исходный код страницы</a>
</li>
</ul>
<hr/>
</div>
<div role="main" class="document" itemscope="itemscope" itemtype="http://schema.org/Article">
<div itemprop="articleBody">
<div class="section" id="id1">
<h1>Документация контроллера прав доступа<a class="headerlink" href="#id1" title="Ссылка на этот заголовок"></a></h1>
<div class="toctree-wrapper compound">
<p class="caption"><span class="caption-text">Contents:</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="overview.html">Документация пользователя</a><ul>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id2">Управление правами доступа</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id3">Главная страница</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id4">Регистрация</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id5">Авторизация</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id6">Профиль</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id7">Запрос прав доступа</a></li>
<li class="toctree-l2"><a class="reference internal" href="overview.html#id8">Управление правами доступа администратором</a></li>
</ul>
</li>
<li class="toctree-l1"><a class="reference internal" href="code.html">Документация разработчика</a><ul>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.models">Models</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.forms">Forms</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.extra_func">Extra Functions</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.serializers">Serializers</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.views">Views</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.requester">Обработка тикетов</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.statistic_data">Обработка статистики</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.zendesk_admin">Функционал администратора Zendesk</a></li>
<li class="toctree-l2"><a class="reference internal" href="code.html#module-main.tests">Тесты</a></li>
</ul>
</li>
<li class="toctree-l1"><a class="reference internal" href="readme.html">READ.me</a></li>
<li class="toctree-l1"><a class="reference internal" href="todo.html">Что необходимо доделать?</a><ul>
<li class="toctree-l2"><a class="reference internal" href="todo.html#todos">TODOs</a></li>
</ul>
</li>
</ul>
</div>
</div>
<div class="section" id="indices-and-tables">
<h1>Indices and tables<a class="headerlink" href="#indices-and-tables" title="Ссылка на этот заголовок"></a></h1>
<ul class="simple">
<li><p><a class="reference internal" href="genindex.html"><span class="std std-ref">Index</span></a></p></li>
<li><p><a class="reference internal" href="py-modindex.html"><span class="std std-ref">Module Index</span></a></p></li>
<li><p><a class="reference internal" href="search.html"><span class="std std-ref">Search Page</span></a></p></li>
</ul>
</div>
</div>
</div>
<footer>
<div class="rst-footer-buttons" role="navigation" aria-label="footer navigation">
<a href="overview.html" class="btn btn-neutral float-right" title="Документация пользователя" accesskey="n" rel="next">Следующая <span class="fa fa-arrow-circle-right" aria-hidden="true"></span></a>
</div>
<hr/>
<div role="contentinfo">
<p>
&#169; Copyright 2021, SHP S101, group 2.
</p>
</div>
Собрано при помощи <a href="https://www.sphinx-doc.org/">Sphinx</a> с использованием
<a href="https://github.com/readthedocs/sphinx_rtd_theme">темы,</a>
предоставленной <a href="https://readthedocs.org">Read the Docs</a>.
</footer>
</div>
</div>
</section>
</div>
<script type="text/javascript">
jQuery(function () {
SphinxRtdTheme.Navigation.enable(true);
});
</script>
</body>
</html>

View File

@@ -88,7 +88,11 @@
<tr>
<td scope="col">&nbsp;</td>
{% for date in log_stats.keys %}
{% if interval == 'days' %}
<td scope="col">{{ date | date:'d.m' }}</td>
{% else %}
<td scope="col">{{ date.1 | date:'F' }}</td>
{% endif %}
{% endfor %}
</tr>
</thead>

View File

@@ -11,6 +11,7 @@ from django.core import mail
from django.http import HttpResponseRedirect
from django.template.loader import render_to_string
from django.test import TestCase, Client
from django.urls import reverse, reverse_lazy
from django.utils import translation, timezone
@@ -896,3 +897,24 @@ class LoggingTestCase(UsersBaseTestCase):
file_output = self.get_file_output()
self.assertEqual(file_output, f'UserForAccessTest,light_agent,'
f'{str(timezone.now().today())[:16]},UserForAccessTest\n')
class ControlAccessTests(TestCase):
"""
Класс тестов для проверки доступа к странице управления
"""
fixtures = ['fixtures/data.json']
def setUp(self) -> None:
self.User = get_user_model()
self.client = Client()
def test_admin_has_perm(self):
self.client.force_login(self.User.objects.get(email='admin@gmail.com'))
self.response = self.client.get(reverse('control'))
self.assertEqual(self.response.status_code, 200)
def test_engineer_doesnt_have_perm(self):
self.client.force_login(self.User.objects.get(email='123@test.ru'))
self.response = self.client.get(reverse('control'))
self.assertEqual(self.response.status_code, 403)

View File

@@ -29,7 +29,7 @@ from rest_framework.response import Response
from access_controller.settings import DEFAULT_FROM_EMAIL, ZENDESK_ROLES, ZENDESK_MAX_AGENTS, ZENDESK_GROUPS
from main.extra_func import check_user_exist, update_profile, get_user_organization, \
make_engineer, make_light_agent, get_users_list, update_users_in_model, count_users, \
set_session_params_for_work_page, get_tickets_list_for_group
set_session_params_for_work_page, get_tickets_list_for_group, set_permission
from main.forms import AdminPageUsers, CustomRegistrationForm, CustomAuthenticationForm, \
StatisticForm, WorkGetTicketsForm
from main.serializers import ProfileSerializer, ZendeskUserSerializer
@@ -118,7 +118,7 @@ class CustomRegistrationView(RegistrationView):
)
try:
update_profile(user.userprofile)
self.set_permission(user)
set_permission(user)
forms.save(**opts)
return user
except SMTPException:
@@ -131,21 +131,6 @@ class CustomRegistrationView(RegistrationView):
self.redirect_url = 'invalid_zendesk_email'
return None
@staticmethod
def set_permission(user: get_user_model()) -> None:
"""
Функция дает разрешение на просмотр страница администратора, если пользователь имеет роль admin.
:param user: Авторизованный пользователь (получает разрешение, имея роль "admin")
"""
if user.userprofile.role == 'admin':
content_type = ContentType.objects.get_for_model(UserProfile)
permission = Permission.objects.get(
codename='has_control_access',
content_type=content_type,
)
user.user_permissions.add(permission)
def get_success_url(self, user: get_user_model() = None) -> Dict:
"""
Функция возвращает url-адрес страницы, куда нужно перейти после успешной/не успешной регистрации.
@@ -447,6 +432,7 @@ def statistic_page(request: WSGIRequest) -> HttpResponse:
context['errors'] = data.errors
if data.warnings:
context['warnings'] = data.warnings
context['interval'] = data.interval
context['log_stats'] = stats if not context['errors'] else None
elif request.method == 'GET':
form = StatisticForm()
@@ -458,3 +444,5 @@ def registration_failed(request: WSGIRequest) -> HttpResponse:
Функция отображения страницы "Регистрация закрыта".
"""
return render(request, 'pages/registration_failed.html')