Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍11
Это объект, который имитирует поведение реального объекта в контролируемом и предсказуемом виде. В контексте тестирования, особенно в Python, это используется для создания фиктивных объектов, которые замещают реальные компоненты системы во время выполнения тестов.
Позволяет тестировать компоненты системы независимо от других частей.
Можно моделировать ответы и поведение зависимостей, таких как базы данных или веб-сервисы.
Тесты с mock-объектами выполняются быстрее, так как не требуют взаимодействия с реальными ресурсами.
Легко симулировать ошибки и исключения, которые сложно воспроизвести с реальными объектами.
Могут заменять любые объекты в коде.
Указание, какие значения должны возвращать mock-объекты.
Возможность проверять, как и сколько раз были вызваны методы mock-объекта.
Упрощают замену объектов в тестах на mock.
Предположим, у нас есть функция, которая обращается к внешнему API:
import requests
def get_user_data(user_id):
response = requests.get(f"https://api.example.com/users/{user_id}")
return response.json()
Чтобы протестировать эту функцию без реальных запросов, можно использовать mock:
from unittest.mock import patch
@patch('requests.get')
def test_get_user_data(mock_get):
mock_get.return_value.json.return_value = {"id": 1, "name": "John Doe"}
result = get_user_data(1)
assert result == {"id": 1, "name": "John Doe"}
mock_get.assert_called_once_with("https://api.example.com/users/1")
return_value
Устанавливает возвращаемое значение метода.
side_effect
Устанавливает последовательность возвращаемых значений или исключений.
assert_called_with()
Проверяет, что метод был вызван с определенными аргументами.
assert_called_once_with()
Проверяет, что метод был вызван ровно один раз с определенными аргументами.
import requests
from unittest.mock import patch
def get_status_code(url):
response = requests.get(url)
return response.status_code
def test_get_status_code():
with patch('requests.get') as mock_get:
mock_get.return_value.status_code = 200
assert get_status_code("https://example.com") == 200
mock_get.assert_called_once_with("https://example.com")
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍11❤1
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍6❤5🔥1
Существует несколько видов тестов, каждый из которых имеет свою цель и особенности.
Проверка работы отдельных модулей или функций в изоляции от других частей системы.
Ориентированы на минимальные части кода (функции, методы, классы).
Высокая скорость выполнения.
Простота написания и отладки.
Обычно пишутся разработчиками.
def add(a, b):
return a + b
def test_add():
assert add(1, 2) == 3
Проверка взаимодействия между различными модулями или компонентами системы.
Тестируют комбинации модулей и их взаимодействие.
Более сложные и медленные по сравнению с юнит-тестами.
Могут выявить проблемы в интерфейсах между модулями.
def fetch_data_from_api():
response = requests.get('https://api.example.com/data')
return response.json()
def test_fetch_data_from_api():
data = fetch_data_from_api()
assert 'key' in data
Проверка всей системы целиком на соответствие требованиям.
Тестируют систему в рабочей среде.
Включают проверку всех функциональных и нефункциональных требований.
Могут включать пользовательские сценарии.
Тестирование веб-приложения на основе реальных пользовательских сценариев, включая проверку интерфейса, баз данных и API.
Проверка соответствия системы требованиям и ожиданиям заказчика или конечного пользователя.
Часто выполняются вместе с заказчиком или пользователем.
Фокусируются на бизнес-требованиях и пользовательских сценариях.
Успешное прохождение приемочных тестов является критерием готовности системы к выпуску.
Тестирование нового функционала с участием конечных пользователей для проверки его удобства и соответствия их ожиданиям.
Убедиться, что изменения в коде не вызвали новых ошибок в уже работающем функционале.
Выполняются после внесения изменений в код.
Обычно автоматизируются и включают повторное выполнение всех или части существующих тестов.
Повторное выполнение всех юнит-тестов и интеграционных тестов после рефакторинга кода.
Проверка нефункциональных аспектов системы, таких как производительность, безопасность, удобство использования и др.
Измеряют скорость выполнения, пропускную способность и время отклика системы.
Оценивают защищенность системы от угроз и атак.
Проверяют удобство и интуитивность пользовательского интерфейса.
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍9🔥4
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥6❤1
Это концепция, предложенная Майком Коэном (Mike Cohn), которая описывает оптимальное соотношение разных видов тестов для обеспечения качественного программного обеспечения. Пирамида тестирования визуально представлена в виде треугольника, где каждый уровень соответствует определённому виду тестов, а ширина уровня отражает рекомендуемое количество тестов этого вида.
Расположение: Основание пирамиды.
Цель: Тестирование отдельных функций или методов в изоляции.
Количество: Наибольшее количество тестов.
Особенности: Высокая скорость выполнения, низкая стоимость написания и поддержки.
def add(a, b):
return a + b
def test_add():
assert add(1, 2) == 3
Расположение: Средний уровень пирамиды.
Цель: Тестирование взаимодействия между различными модулями или компонентами.
Количество: Меньше, чем юнит-тестов, но больше, чем системных тестов.
Особенности: Проверка интерфейсов и взаимодействия между модулями, более сложные и медленные, чем юнит-тесты.
def fetch_data_from_api():
response = requests.get('https://api.example.com/data')
return response.json()
def test_fetch_data_from_api():
data = fetch_data_from_api()
assert 'key' in data
Расположение: Вершина пирамиды.
Цель: Тестирование всей системы целиком на соответствие требованиям.
Количество: Наименьшее количество тестов.
Особенности: Включают тестирование пользовательских сценариев, проверку функциональных и нефункциональных требований, самые сложные и медленные.
Юнит-тесты дешевле в написании и выполнении, чем интеграционные и системные тесты.
Большое количество юнит-тестов позволяет быстро находить ошибки на ранних стадиях разработки.
Юнит-тесты способствуют улучшению дизайна кода и повышают его тестируемость.
Хорошее покрытие юнит-тестами позволяет быстро обнаруживать и исправлять ошибки, возникающие после внесения изменений в код.
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍7🔥2
Такие ситуации можно обработать и протестировать несколькими способами. Один из наиболее распространенных подходов — использование mock-объектов для имитации поведения внешних сервисов. Это позволяет изолировать тестируемую функцию и контролировать ее взаимодействие с внешними сервисами.
С помощью библиотеки
unittest.mock
можно подменить вызовы к внешним сервисам и задать их поведение, включая ошибки и исключения.В тестируемой функции следует предусмотреть обработку возможных ошибок, таких как таймауты и HTTP-статусы (например, 404), чтобы она корректно реагировала на эти ситуации.
Предположим, у нас есть функция, которая запрашивает данные с удаленного сервиса
import requests
def fetch_data_from_api(url):
try:
response = requests.get(url, timeout=5)
response.raise_for_status()
return response.json()
except requests.exceptions.Timeout:
return {"error": "Timeout"}
except requests.exceptions.HTTPError as http_err:
return {"error": f"HTTP error: {http_err.response.status_code}"}
except requests.exceptions.RequestException as err:
return {"error": f"Request error: {err}"}
Установка поведения для mock-объекта
Создадим тесты, которые подменяют вызов
requests.get
и задают различные сценарии ошибок.from unittest.mock import patch
import requests
import pytest
def test_fetch_data_timeout():
with patch('requests.get') as mock_get:
mock_get.side_effect = requests.exceptions.Timeout
result = fetch_data_from_api("https://example.com/data")
assert result == {"error": "Timeout"}
def test_fetch_data_404():
with patch('requests.get') as mock_get:
mock_response = mock_get.return_value
mock_response.raise_for_status.side_effect = requests.exceptions.HTTPError(response=mock_response)
mock_response.status_code = 404
result = fetch_data_from_api("https://example.com/data")
assert result == {"error": "HTTP error: 404"}
def test_fetch_data_request_exception():
with patch('requests.get') as mock_get:
mock_get.side_effect = requests.exceptions.RequestException("Connection error")
result = fetch_data_from_api("https://example.com/data")
assert result == {"error": "Request error: Connection error"}
Используем
mock_get.side_effect = requests.exceptions.Timeout
для имитации ошибки таймаута. Проверяем, что функция возвращает ожидаемое значение при таймауте.Устанавливаем
mock_response.raise_for_status.side_effect = requests.exceptions.HTTPError(response=mock_response)
для имитации HTTP ошибки. Устанавливаем код статуса mock_response.status_code = 404
. Проверяем, что функция возвращает ожидаемое значение при HTTP ошибке 404.Используем
mock_get.side_effect = requests.exceptions.RequestException("Connection error")
для имитации общей ошибки запроса. Проверяем, что функция возвращает ожидаемое значение при общей ошибке запроса.Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍7
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍9😁1
Можно использовать мокирование (mocking) зависимостей или стабирование (stubbing), чтобы заменить долгие операции на быстрые фиктивные реализации, которые будут возвращать заранее заданные значения. Также можно рассмотреть параллельное выполнение или кеширование результатов для ускорения тестов.
Если функция выполняет одинаковые операции с одинаковыми входными данными, можно кэшировать результаты этих операций, чтобы не выполнять их повторно.
Используйте mock-объекты для замены дорогих операций фиктивными, которые возвращают заранее определенные результаты.
Фикстуры позволяют подготовить данные или состояния один раз перед выполнением тестов и повторно использовать их в разных тестах.
Параллельное выполнение тестов может значительно сократить общее время тестирования. Допустим, у нас есть функция, которая выполняет дорогую операцию:
import time
def expensive_operation(data):
time.sleep(5) # Имитация долгой операции
return sum(data)
def process_data(data):
result = []
for item in data:
result.append(expensive_operation(item))
return result
Мы можем замокировать
expensive_operation
, чтобы она возвращала заранее определенные результаты без выполнения долгих операций.from unittest.mock import patch
def test_process_data():
data = [[1, 2, 3], [4, 5, 6], [7, 8, 9]]
with patch('__main__.expensive_operation') as mock_expensive:
mock_expensive.side_effect = [6, 15, 24] # Заранее определенные результаты
result = process_data(data)
assert result == [6, 15, 24]
assert mock_expensive.call_count == 3 # Проверка, что функция вызвана 3 раза
Используем кэширование для хранения результатов дорогостоящих операций.
from functools import lru_cache
@lru_cache(maxsize=None)
def expensive_operation(data):
time.sleep(5) # Имитация долгой операции
return sum(data)
def process_data(data):
result = []
for item in data:
result.append(expensive_operation(tuple(item))) # Преобразуем список в кортеж для кэширования
return result
Фикстуры для подготовки данных
Создаем фикстуры для подготовки данных, которые используются в тестах.
import pytest
@pytest.fixture
def sample_data():
return [[1, 2, 3], [4, 5, 6], [7, 8, 9]]
def test_process_data(sample_data):
with patch('__main__.expensive_operation') as mock_expensive:
mock_expensive.side_effect = [6, 15, 24] # Заранее определенные результаты
result = process_data(sample_data)
assert result == [6, 15, 24]
Запуск тестов параллельно
Используйте возможности параллельного выполнения тестов, например, с помощью
pytest-xdist
.pytest -n 4 # Запуск тестов в 4 параллельных потоках
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥9👍7
Проверка взаимодействия между различными модулями или компонентами системы.
Проверяет, как отдельные части системы работают вместе.
Выявляет проблемы, которые могут возникнуть при передаче данных или управлении между модулями.
Проводится после юнит-тестов и перед системными тестами.
Включает взаимодействие с внешними сервисами, базами данных, файлами и другими модулями.
Часто используются mock-объекты для замены реальных зависимостей.
def fetch_data_from_db():
# Имитация запроса к базе данных
return {"user": "John", "email": "john@example.com"}
def send_email(data):
# Имитация отправки электронной почты
print(f"Sending email to {data['email']}")
def process_user_data():
data = fetch_data_from_db()
send_email(data)
Интеграционный тест может выглядеть так
def test_process_user_data():
data = fetch_data_from_db()
assert data["email"] == "john@example.com"
send_email(data)
Проверка функциональности системы на соответствие требованиям.
Проверяет, выполняет ли система свои функции в соответствии с заданными требованиями.
Тестирует систему с точки зрения конечного пользователя.
Может включать юнит-тесты, интеграционные тесты, системные и приемочные тесты.
Не требует знания внутренней структуры или кода системы.
Могут использоваться автоматизированные тестовые фреймворки, такие как Selenium для веб-приложений.
def register_user(username, password):
# Имитация регистрации пользователя
if username and password:
return "Registration successful"
else:
return "Registration failed"
Функциональный тест может выглядеть так:
def test_register_user():
assert register_user("testuser", "securepassword") == "Registration successful"
assert register_user("", "securepassword") == "Registration failed"
Интеграционное тестирование: Проверка взаимодействия между модулями.
Функциональное тестирование: Проверка функциональности системы на соответствие требованиям.
Интеграционное тестирование: Взаимодействие и интерфейсы между модулями.
Функциональное тестирование: Поведение системы с точки зрения пользователя.
Интеграционное тестирование: Промежуточный уровень между юнит-тестами и системными тестами.
Функциональное тестирование: Может охватывать все уровни, включая юнит-тесты, интеграционные, системные и приемочные тесты.
Интеграционное тестирование: Тестирование "серого ящика" (частичное знание внутренней структуры).
Функциональное тестирование: Тестирование "черного ящика" (без знания внутренней структуры).
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍5❤1
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
😁7💊4👍3🔥3
Вьюха обрабатывает запрос (например, получение данных из базы данных, выполнение бизнес-логики). На основе обработки данных вьюха формирует объект ответа. В Django это может быть
HttpResponse
, JsonResponse
и т.д. В Flask это может быть объект Response
или просто строка, которая будет конвертирована в ответ. from django.http import HttpResponse
def my_view(request):
data = "Hello, World!"
return HttpResponse(data)
Пример в Flask
from flask import Flask, Response
app = Flask(__name__)
@app.route('/')
def my_view():
return Response("Hello, World!")
После того как вьюха сформировала ответ, он проходит через цепочку промежуточных слоев (middleware). Middleware могут модифицировать объект ответа, добавлять заголовки, обрабатывать ошибки и т.д. Примеры обработки включают логирование, сжатие контента, обработку сессий и т.д.
class SimpleMiddleware:
def __init__(self, get_response):
self.get_response = get_response
def __call__(self, request):
response = self.get_response(request)
response['X-Custom-Header'] = 'My Custom Header'
return response
После прохождения всех промежуточных слоев ответ передается веб-серверу. Веб-сервер, такой как Nginx или Apache, отправляет сформированный ответ обратно клиенту (например, браузеру). Сервер обрабатывает низкоуровневые детали HTTP-протокола, такие как установка соединения, обработка заголовков и передача данных.
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍11
Это процесс проверки корректности и полноты данных, поступающих в систему, прежде чем они будут использованы или сохранены. Этот процесс может происходить на разных уровнях веб-приложения: клиентская сторона, серверная сторона, и на уровне базы данных.
До отправки данных на сервер. Используются HTML5 атрибуты и JavaScript. Снижение нагрузки на сервер, мгновенная обратная связь для пользователя. Не является надежной, так как пользователь может обойти ее.
<form>
<input type="text" name="username" required minlength="3">
<input type="email" name="email" required>
<input type="submit">
</form>
Пример JavaScript валидации
document.querySelector('form').addEventListener('submit', function(event) {
const username = document.querySelector('input[name="username"]').value;
if (username.length < 3) {
alert('Username must be at least 3 characters long.');
event.preventDefault();
}
});
После получения данных от клиента. Используются встроенные или сторонние библиотеки и фреймворки. Надежная защита, обязательна даже при наличии клиентской валидации. Добавляет нагрузку на сервер, может увеличить время отклика.
from django import forms
class UserForm(forms.Form):
username = forms.CharField(min_length=3, max_length=100)
email = forms.EmailField()
def my_view(request):
if request.method == 'POST':
form = UserForm(request.POST)
if form.is_valid():
# Обработка данных
pass
else:
# Обработка ошибок
pass
else:
form = UserForm()
return render(request, 'my_template.html', {'form': form})
Пример в Flask
from flask import Flask, request, render_template_string
from wtforms import Form, StringField, validators
app = Flask(__name__)
class UserForm(Form):
username = StringField('Username', [validators.Length(min=3, max=100)])
email = StringField('Email', [validators.Email()])
@app.route('/', methods=['GET', 'POST'])
def my_view():
form = UserForm(request.form)
if request.method == 'POST' and form.validate():
# Обработка данных
pass
return render_template_string('<form method="POST">{{ form.csrf_token }}{{ form.username }}{{ form.email }}<input type="submit"></form>', form=form)
При сохранении данных в базу данных. Используются ограничения базы данных (constraints) и триггеры. Гарантия целостности данных на уровне хранения. Может вызвать ошибки, если данные уже были приняты сервером, но не прошли валидацию на уровне БД.
Пример ограничения в SQL
CREATE TABLE users (
id SERIAL PRIMARY KEY,
username VARCHAR(100) NOT NULL CHECK (LENGTH(username) >= 3),
email VARCHAR(100) NOT NULL UNIQUE
);
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥5👍2💊2😁1
Это стандартный интерфейс между веб-серверами и веб-приложениями или фреймворками, написанными на языке Python. Он определяет, как веб-сервер должен взаимодействовать с веб-приложением, и позволяет различным веб-серверам работать с любыми Python-приложениями, поддерживающими WSGI.
Позволяет любому WSGI-совместимому веб-серверу запускать любое WSGI-совместимое приложение.
Обеспечивает возможность объединения нескольких промежуточных компонентов (middleware), которые могут выполнять задачи до или после обработки запросов основным приложением.
Упрощает разработку и развертывание веб-приложений, делая их независимыми от конкретных веб-серверов.
Программное обеспечение, которое принимает HTTP-запросы от клиента (например, браузера), преобразует их в формат, понятный приложению, и затем отправляет HTTP-ответы обратно клиенту. Примеры WSGI-серверов: Gunicorn, uWSGI.
Любое Python-приложение или фреймворк, которое реализует WSGI-интерфейс. Примеры фреймворков: Django, Flask.
WSGI определяет простой интерфейс, который веб-сервер использует для передачи запросов в приложение и получения ответов от него. Этот интерфейс основан на вызове функции или вызываемого объекта (callable), который принимает два аргумента:
environ
и start_response
.Словарь, содержащий все данные о запросе, такие как заголовки, параметры и другие метаданные.
Функция, которую приложение вызывает для начала формирования ответа. Она принимает статус ответа и заголовки.
def application(environ, start_response):
status = '200 OK'
headers = [('Content-Type', 'text/plain')]
start_response(status, headers)
return [b"Hello, World!"]
# Запуск приложения с использованием встроенного WSGI-сервера wsgiref
if __name__ == "__main__":
from wsgiref.simple_server import make_server
server = make_server('localhost', 8051, application)
print("Serving on http://localhost:8051...")
server.serve_forever()
Современные Python веб-фреймворки, такие как Django и Flask, автоматически поддерживают WSGI. Это позволяет легко развертывать приложения, используя WSGI-серверы.
from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello():
return "Hello, World!"
if __name__ == "__main__":
app.run()
Для развертывания этого приложения с использованием WSGI-сервера, например, Gunicorn, можно выполнить команду. Где
myapp
— это имя файла без расширения .py
, а app
— это объект приложения Flask.gunicorn myapp:app
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4❤1
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8
Django и микрофреймворки, такие как FastAPI, предназначены для разработки веб-приложений на Python, но они имеют различия в подходе, функциональности и областях применения.
Django предоставляет полный набор инструментов для разработки веб-приложений, включая ORM (Object-Relational Mapping), систему шаблонов, формы, аутентификацию, админ-панель и другие компоненты. Подходит для разработки сложных и крупных приложений, требующих множества встроенных функциональностей.
Django следует принципу "The web framework for perfectionists with deadlines", предлагая стандартные способы решения большинства задач. Многое делается "из коробки", что снижает необходимость принимать решения по настройке и интеграции.
Django создавался с идеей "все включено", что означает, что все основные компоненты встроены и работают вместе. Это упрощает создание приложений, но может быть избыточным для небольших или простых проектов.
Django ORM позволяет удобно работать с различными базами данных, абстрагируясь от специфики SQL. Включает мощные средства для миграций баз данных.
from django.shortcuts import render
from django.http import HttpResponse
def hello_world(request):
return HttpResponse("Hello, World!")
FastAPI является легковесным фреймворком, предназначенным для создания высокопроизводительных веб-API. Включает минимальное количество компонентов, предоставляя только то, что необходимо для обработки запросов и ответов.
FastAPI изначально поддерживает асинхронное программирование с помощью
async
и await
, что позволяет создавать высокопроизводительные приложения. Построен на базе Starlette и Pydantic, что обеспечивает скорость работы и удобную валидацию данных.FastAPI следует принципу "минимальных зависимостей", позволяя разработчику добавлять только необходимые компоненты и библиотеки. Легко интегрируется с различными базами данных и другими инструментами по мере необходимости.
FastAPI использует аннотации типов Python для автоматической генерации документации OpenAPI и валидации данных. Это упрощает разработку и тестирование API, делая код более читаемым и поддерживаемым.
from fastapi import FastAPI
app = FastAPI()
@app.get("/")
async def hello_world():
return {"message": "Hello, World!"}
Django: Предоставляет все необходимые инструменты для разработки полного веб-приложения, включая работу с базами данных, формами, аутентификацией и админ-панелью.
FastAPI: Фокусируется на создании быстрых и производительных веб-API, оставляя выбор других инструментов за разработчиком.
Django: Поддержка асинхронного программирования была добавлена в более поздних версиях, но изначально он был ориентирован на синхронное программирование.
FastAPI: Изначально построен с поддержкой асинхронного программирования, что позволяет создавать высокопроизводительные приложения.
Ставь 👍 и забирай 📚 Базу знаний
Please open Telegram to view this post
VIEW IN TELEGRAM
👍22❤3