mirror of
https://github.com/yiGmMk/wxocr.git
synced 2026-04-22 16:17:09 +08:00
add uv & page & testcase & docker
This commit is contained in:
@@ -0,0 +1,3 @@
|
||||
.venv
|
||||
__pycache__
|
||||
temp/*
|
||||
@@ -0,0 +1 @@
|
||||
3.12
|
||||
@@ -11,6 +11,7 @@ COPY wcocr.cpython-312-x86_64-linux-gnu.so /app/wcocr.cpython-312-x86_64-linux-g
|
||||
COPY wx /app/wx
|
||||
|
||||
COPY main.py /app/main.py
|
||||
COPY templates /app/templates
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
|
||||
@@ -0,0 +1,15 @@
|
||||
version: "3.8"
|
||||
|
||||
services:
|
||||
wechat-ocr-api:
|
||||
build: .
|
||||
ports:
|
||||
- "5000:5000"
|
||||
container_name: wechat-ocr-api
|
||||
|
||||
# 使用预构建的镜像
|
||||
# wechat-ocr-api:
|
||||
# image: golangboyme/wxocr
|
||||
# ports:
|
||||
# - "5000:5000"
|
||||
# container_name: wechat-ocr-api
|
||||
Binary file not shown.
|
After Width: | Height: | Size: 65 KiB |
@@ -2,34 +2,39 @@ import wcocr
|
||||
import os
|
||||
import uuid
|
||||
import base64
|
||||
from flask import Flask, request, jsonify
|
||||
from flask import Flask, request, jsonify, render_template, send_from_directory
|
||||
|
||||
app = Flask(__name__)
|
||||
wcocr.init("/app/wx/opt/wechat/wxocr", "/app/wx/opt/wechat")
|
||||
wcocr.init("./wx/opt/wechat/wxocr", "./wx/opt/wechat")
|
||||
|
||||
@app.route('/ocr', methods=['POST'])
|
||||
|
||||
@app.route("/ocr", methods=["POST"])
|
||||
def ocr():
|
||||
try:
|
||||
# Get base64 image from request
|
||||
image_data = request.json.get('image')
|
||||
image_data = request.json.get("image")
|
||||
if not image_data:
|
||||
return jsonify({'error': 'No image data provided'}), 400
|
||||
return jsonify({"error": "No image data provided"}), 400
|
||||
# Extract image type from base64 data
|
||||
image_type, base64_data = extract_image_type(image_data)
|
||||
if not image_type:
|
||||
return jsonify({"error": "Invalid base64 image data"}), 400
|
||||
|
||||
# Create temp directory if not exists
|
||||
temp_dir = 'temp'
|
||||
temp_dir = "temp"
|
||||
if not os.path.exists(temp_dir):
|
||||
os.makedirs(temp_dir)
|
||||
|
||||
# Generate unique filename and save image
|
||||
filename = os.path.join(temp_dir, f"{str(uuid.uuid4())}.png")
|
||||
filename = os.path.join(temp_dir, f"{str(uuid.uuid4())}.{image_type}")
|
||||
try:
|
||||
image_bytes = base64.b64decode(image_data)
|
||||
with open(filename, 'wb') as f:
|
||||
image_bytes = base64.b64decode(base64_data)
|
||||
with open(filename, "wb") as f:
|
||||
f.write(image_bytes)
|
||||
|
||||
# Process image with OCR
|
||||
result = wcocr.ocr(filename)
|
||||
return jsonify({'result': result})
|
||||
return jsonify({"result": result})
|
||||
|
||||
finally:
|
||||
# Clean up temp file
|
||||
@@ -37,7 +42,44 @@ def ocr():
|
||||
os.remove(filename)
|
||||
|
||||
except Exception as e:
|
||||
return jsonify({'error': str(e)}), 500
|
||||
return jsonify({"error": str(e)}), 500
|
||||
|
||||
if __name__ == '__main__':
|
||||
app.run(host='0.0.0.0', port=5000, threaded=True)
|
||||
|
||||
# 创建静态文件夹
|
||||
static_dir = os.path.join(os.path.dirname(os.path.abspath(__file__)), "static")
|
||||
if not os.path.exists(static_dir):
|
||||
os.makedirs(static_dir)
|
||||
|
||||
|
||||
def extract_image_type(base64_data):
|
||||
# Check if the base64 data has the expected prefix
|
||||
if base64_data.startswith("data:image/"):
|
||||
# Extract the image type from the prefix
|
||||
prefix_end = base64_data.find(";base64,")
|
||||
if prefix_end != -1:
|
||||
return (
|
||||
base64_data[len("data:image/") : prefix_end],
|
||||
base64_data.split(";base64,")[-1],
|
||||
)
|
||||
return "png", base64_data
|
||||
|
||||
|
||||
@app.route("/")
|
||||
def index():
|
||||
return render_template("index.html")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
# 确保templates目录存在
|
||||
templates_dir = os.path.join(
|
||||
os.path.dirname(os.path.abspath(__file__)), "templates"
|
||||
)
|
||||
if not os.path.exists(templates_dir):
|
||||
os.makedirs(templates_dir)
|
||||
|
||||
# 确保temp目录存在
|
||||
temp_dir = os.path.join(os.path.dirname(os.path.abspath(__file__)), "temp")
|
||||
if not os.path.exists(temp_dir):
|
||||
os.makedirs(temp_dir)
|
||||
|
||||
app.run(host="0.0.0.0", port=5000, threaded=True)
|
||||
|
||||
@@ -0,0 +1,9 @@
|
||||
[project]
|
||||
name = "wxocr"
|
||||
version = "0.1.0"
|
||||
description = "Add your description here"
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.11"
|
||||
dependencies = [
|
||||
"flask>=3.1.0",
|
||||
]
|
||||
@@ -0,0 +1,441 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>微信OCR文字识别工具</title>
|
||||
<style>
|
||||
:root {
|
||||
--primary-color: #07c160;
|
||||
--secondary-color: #576b95;
|
||||
}
|
||||
|
||||
body {
|
||||
font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
padding: 20px;
|
||||
background: #f5f5f5;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.container {
|
||||
background: white;
|
||||
border-radius: 12px;
|
||||
padding: 30px;
|
||||
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
h1 {
|
||||
color: var(--primary-color);
|
||||
text-align: center;
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.upload-section {
|
||||
border: 2px dashed #ddd;
|
||||
border-radius: 8px;
|
||||
padding: 30px;
|
||||
text-align: center;
|
||||
margin: 20px 0;
|
||||
transition: all 0.3s;
|
||||
}
|
||||
|
||||
.upload-section:hover {
|
||||
border-color: var(--primary-color);
|
||||
background: #f8fff9;
|
||||
}
|
||||
|
||||
.preview-image {
|
||||
max-width: 100%;
|
||||
max-height: 400px;
|
||||
margin: 20px 0;
|
||||
border-radius: 8px;
|
||||
display: none;
|
||||
}
|
||||
|
||||
.input-group {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
input[type="file"],
|
||||
input[type="text"] {
|
||||
width: 100%;
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: 6px;
|
||||
margin: 10px 0;
|
||||
}
|
||||
|
||||
button {
|
||||
background: var(--primary-color);
|
||||
color: white;
|
||||
border: none;
|
||||
padding: 12px 30px;
|
||||
border-radius: 6px;
|
||||
cursor: pointer;
|
||||
font-size: 16px;
|
||||
transition: all 0.3s;
|
||||
}
|
||||
|
||||
button:hover {
|
||||
opacity: 0.9;
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.result-table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
margin-top: 20px;
|
||||
display: none;
|
||||
}
|
||||
|
||||
.result-table th,
|
||||
.result-table td {
|
||||
padding: 12px;
|
||||
border: 1px solid #eee;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.result-table th {
|
||||
background: var(--secondary-color);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.loading {
|
||||
display: none;
|
||||
text-align: center;
|
||||
color: var(--primary-color);
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.error {
|
||||
color: #ff4d4f;
|
||||
margin: 10px 0;
|
||||
display: none;
|
||||
}
|
||||
|
||||
.image-container {
|
||||
position: relative;
|
||||
margin: 20px 0;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.text-box {
|
||||
position: absolute;
|
||||
border: 2px solid var(--primary-color);
|
||||
background-color: rgba(7, 193, 96, 0.1);
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.text-tooltip {
|
||||
position: absolute;
|
||||
background: white;
|
||||
border: 1px solid #ddd;
|
||||
padding: 8px;
|
||||
border-radius: 4px;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
|
||||
z-index: 100;
|
||||
display: none;
|
||||
max-width: 300px;
|
||||
word-break: break-word;
|
||||
}
|
||||
|
||||
.confidence {
|
||||
color: var(--primary-color);
|
||||
font-weight: bold;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="container">
|
||||
<h1>微信OCR文字识别工具</h1>
|
||||
|
||||
<!-- 上传区域 -->
|
||||
<div class="upload-section">
|
||||
<div class="input-group">
|
||||
<input type="file" id="fileInput" accept="image/*">
|
||||
<p>或拖拽图片到此区域</p>
|
||||
<input type="text" id="urlInput" placeholder="输入图片URL地址">
|
||||
</div>
|
||||
<button onclick="processImage()">开始识别</button>
|
||||
</div>
|
||||
|
||||
<!-- 图片预览 -->
|
||||
<div class="image-container" id="imageContainer">
|
||||
<img id="preview" class="preview-image">
|
||||
<!-- 文本框将在这里动态添加 -->
|
||||
</div>
|
||||
|
||||
<!-- 加载状态 -->
|
||||
<div class="loading" id="loading">识别中,请稍候...</div>
|
||||
|
||||
<!-- 错误提示 -->
|
||||
<div class="error" id="error"></div>
|
||||
|
||||
<!-- 结果显示 -->
|
||||
<table class="result-table" id="resultTable">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>文本内容</th>
|
||||
<th>置信度</th>
|
||||
<th>位置信息 (左, 上, 右, 下)</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody id="resultBody"></tbody>
|
||||
</table>
|
||||
|
||||
<!-- 使用说明 -->
|
||||
<h2>API接口说明</h2>
|
||||
<h3>请求方式</h3>
|
||||
<pre>POST /ocr</pre>
|
||||
|
||||
<h3>请求示例</h3>
|
||||
<pre>
|
||||
{
|
||||
"image": "BASE64_ENCODED_IMAGE_DATA"
|
||||
}</pre>
|
||||
|
||||
<h3>返回示例</h3>
|
||||
<pre id="responseSample"></pre>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
// 默认的API地址
|
||||
const API_ENDPOINT = window.location.origin + '/ocr';
|
||||
|
||||
// 初始化拖放功能
|
||||
const uploadSection = document.querySelector('.upload-section');
|
||||
uploadSection.addEventListener('dragover', (e) => {
|
||||
e.preventDefault();
|
||||
uploadSection.style.backgroundColor = '#f0fff0';
|
||||
});
|
||||
|
||||
uploadSection.addEventListener('drop', (e) => {
|
||||
e.preventDefault();
|
||||
uploadSection.style.backgroundColor = '';
|
||||
const file = e.dataTransfer.files[0];
|
||||
handleFile(file);
|
||||
});
|
||||
|
||||
// 处理文件选择
|
||||
document.getElementById('fileInput').addEventListener('change', function (e) {
|
||||
handleFile(e.target.files[0]);
|
||||
});
|
||||
|
||||
// 处理文件上传
|
||||
async function handleFile(file) {
|
||||
if (!file) return;
|
||||
if (!file.type.startsWith('image/')) {
|
||||
showError('请上传图片文件');
|
||||
return;
|
||||
}
|
||||
|
||||
// 显示预览图片
|
||||
const reader = new FileReader();
|
||||
reader.onload = (e) => {
|
||||
document.getElementById('preview').src = e.target.result;
|
||||
document.getElementById('preview').style.display = 'block';
|
||||
|
||||
// 清除之前的文本框
|
||||
const imageContainer = document.getElementById('imageContainer');
|
||||
const existingBoxes = imageContainer.querySelectorAll('.text-box, .text-tooltip');
|
||||
existingBoxes.forEach(box => box.remove());
|
||||
};
|
||||
reader.readAsDataURL(file);
|
||||
}
|
||||
|
||||
// 开始处理图像
|
||||
async function processImage() {
|
||||
const file = document.getElementById('fileInput').files[0];
|
||||
const url = document.getElementById('urlInput').value;
|
||||
let base64Data = '';
|
||||
|
||||
try {
|
||||
showLoading();
|
||||
clearError();
|
||||
|
||||
if (file) {
|
||||
base64Data = await fileToBase64(file);
|
||||
} else if (url) {
|
||||
base64Data = await urlToBase64(url);
|
||||
} else {
|
||||
showError('请选择图片或输入图片URL');
|
||||
return;
|
||||
}
|
||||
|
||||
// 发送请求
|
||||
const response = await fetch(API_ENDPOINT, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
body: JSON.stringify({ image: base64Data })
|
||||
});
|
||||
|
||||
const data = await response.json();
|
||||
handleResponse(data);
|
||||
} catch (error) {
|
||||
showError(`请求失败:${error.message}`);
|
||||
} finally {
|
||||
hideLoading();
|
||||
}
|
||||
}
|
||||
|
||||
// 处理响应数据
|
||||
function handleResponse(data) {
|
||||
// 处理新的响应结构
|
||||
const resultData = data.result || data;
|
||||
|
||||
if (resultData.errcode !== 0) {
|
||||
showError(`识别失败,错误码:${resultData.errcode}`);
|
||||
return;
|
||||
}
|
||||
|
||||
// 显示结果表格
|
||||
const tbody = document.getElementById('resultBody');
|
||||
tbody.innerHTML = '';
|
||||
resultData.ocr_response.forEach(item => {
|
||||
const row = document.createElement('tr');
|
||||
row.innerHTML = `
|
||||
<td>${item.text}</td>
|
||||
<td>${(item.rate * 100).toFixed(2)}%</td>
|
||||
<td>(${item.left.toFixed(1)}, ${item.top.toFixed(1)},
|
||||
${item.right.toFixed(1)}, ${item.bottom.toFixed(1)})</td>
|
||||
`;
|
||||
tbody.appendChild(row);
|
||||
});
|
||||
document.getElementById('resultTable').style.display = 'table';
|
||||
|
||||
// 在图片上绘制识别区域
|
||||
drawTextBoxes(resultData.ocr_response, resultData.width, resultData.height);
|
||||
}
|
||||
|
||||
// 在图片上绘制文本框
|
||||
function drawTextBoxes(ocrResults, originalWidth, originalHeight) {
|
||||
const imageContainer = document.getElementById('imageContainer');
|
||||
const preview = document.getElementById('preview');
|
||||
|
||||
// 清除之前的文本框
|
||||
const existingBoxes = imageContainer.querySelectorAll('.text-box, .text-tooltip');
|
||||
existingBoxes.forEach(box => box.remove());
|
||||
|
||||
// 获取图片的实际显示尺寸和位置
|
||||
const imgRect = preview.getBoundingClientRect();
|
||||
const containerRect = imageContainer.getBoundingClientRect();
|
||||
|
||||
// 计算图片相对于容器的偏移
|
||||
const offsetX = imgRect.left - containerRect.left;
|
||||
const offsetY = imgRect.top - containerRect.top;
|
||||
|
||||
// 计算缩放比例
|
||||
const scaleX = imgRect.width / originalWidth;
|
||||
const scaleY = imgRect.height / originalHeight;
|
||||
|
||||
// 为每个识别结果创建文本框
|
||||
ocrResults.forEach((item, index) => {
|
||||
// 创建文本框
|
||||
const textBox = document.createElement('div');
|
||||
textBox.className = 'text-box';
|
||||
|
||||
// 精确定位文本框,考虑图片在容器中的偏移
|
||||
const left = item.left * scaleX + offsetX;
|
||||
const top = item.top * scaleY + offsetY;
|
||||
const width = (item.right - item.left) * scaleX;
|
||||
const height = (item.bottom - item.top) * scaleY;
|
||||
|
||||
// 设置文本框位置和大小
|
||||
textBox.style.left = `${left}px`;
|
||||
textBox.style.top = `${top}px`;
|
||||
textBox.style.width = `${width}px`;
|
||||
textBox.style.height = `${height}px`;
|
||||
textBox.dataset.index = index;
|
||||
|
||||
// 创建提示框
|
||||
const tooltip = document.createElement('div');
|
||||
tooltip.className = 'text-tooltip';
|
||||
tooltip.innerHTML = `
|
||||
<div>${item.text}</div>
|
||||
<div class="confidence">置信度: ${(item.rate * 100).toFixed(2)}%</div>
|
||||
`;
|
||||
|
||||
// 添加鼠标事件
|
||||
textBox.addEventListener('mouseenter', function (e) {
|
||||
tooltip.style.left = `${e.pageX - imageContainer.offsetLeft + 10}px`;
|
||||
tooltip.style.top = `${e.pageY - imageContainer.offsetTop + 10}px`;
|
||||
tooltip.style.display = 'block';
|
||||
});
|
||||
|
||||
textBox.addEventListener('mousemove', function (e) {
|
||||
tooltip.style.left = `${e.pageX - imageContainer.offsetLeft + 10}px`;
|
||||
tooltip.style.top = `${e.pageY - imageContainer.offsetTop + 10}px`;
|
||||
});
|
||||
|
||||
textBox.addEventListener('mouseleave', function () {
|
||||
tooltip.style.display = 'none';
|
||||
});
|
||||
|
||||
imageContainer.appendChild(textBox);
|
||||
imageContainer.appendChild(tooltip);
|
||||
});
|
||||
}
|
||||
|
||||
// 工具函数
|
||||
function fileToBase64(file) {
|
||||
return new Promise((resolve, reject) => {
|
||||
const reader = new FileReader();
|
||||
reader.onload = () => resolve(reader.result);
|
||||
reader.onerror = error => reject(error);
|
||||
reader.readAsDataURL(file);
|
||||
});
|
||||
}
|
||||
|
||||
async function urlToBase64(url) {
|
||||
const response = await fetch(url);
|
||||
const blob = await response.blob();
|
||||
return fileToBase64(blob);
|
||||
}
|
||||
|
||||
function showLoading() {
|
||||
document.getElementById('loading').style.display = 'block';
|
||||
}
|
||||
|
||||
function hideLoading() {
|
||||
document.getElementById('loading').style.display = 'none';
|
||||
}
|
||||
|
||||
function showError(message) {
|
||||
const errorDiv = document.getElementById('error');
|
||||
errorDiv.textContent = message;
|
||||
errorDiv.style.display = 'block';
|
||||
}
|
||||
|
||||
function clearError() {
|
||||
document.getElementById('error').style.display = 'none';
|
||||
}
|
||||
|
||||
// 初始化示例响应显示
|
||||
document.getElementById('responseSample').textContent = JSON.stringify({
|
||||
"result": {
|
||||
"errcode": 0,
|
||||
"height": 258,
|
||||
"imgpath": "temp/0cfbda36-a05d-4cba-9a72-cec6833d305d.png",
|
||||
"ocr_response": [
|
||||
{
|
||||
"bottom": 41.64999771118164,
|
||||
"left": 33.6875,
|
||||
"rate": 0.9951504468917847,
|
||||
"right": 164.76248168945312,
|
||||
"text": "API接口说明",
|
||||
"top": 18.98750114440918
|
||||
}
|
||||
],
|
||||
"width": 392
|
||||
}
|
||||
}, null, 2);
|
||||
</script>
|
||||
</body>
|
||||
|
||||
</html>
|
||||
@@ -0,0 +1,141 @@
|
||||
version = 1
|
||||
revision = 1
|
||||
requires-python = ">=3.11"
|
||||
|
||||
[[package]]
|
||||
name = "blinker"
|
||||
version = "1.9.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/21/28/9b3f50ce0e048515135495f198351908d99540d69bfdc8c1d15b73dc55ce/blinker-1.9.0.tar.gz", hash = "sha256:b4ce2265a7abece45e7cc896e98dbebe6cead56bcf805a3d23136d145f5445bf", size = 22460 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/10/cb/f2ad4230dc2eb1a74edf38f1a38b9b52277f75bef262d8908e60d957e13c/blinker-1.9.0-py3-none-any.whl", hash = "sha256:ba0efaa9080b619ff2f3459d1d500c57bddea4a6b424b60a91141db6fd2f08bc", size = 8458 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "click"
|
||||
version = "8.1.8"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "colorama", marker = "sys_platform == 'win32'" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/b9/2e/0090cbf739cee7d23781ad4b89a9894a41538e4fcf4c31dcdd705b78eb8b/click-8.1.8.tar.gz", hash = "sha256:ed53c9d8990d83c2a27deae68e4ee337473f6330c040a31d4225c9574d16096a", size = 226593 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/7e/d4/7ebdbd03970677812aac39c869717059dbb71a4cfc033ca6e5221787892c/click-8.1.8-py3-none-any.whl", hash = "sha256:63c132bbbed01578a06712a2d1f497bb62d9c1c0d329b7903a866228027263b2", size = 98188 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "colorama"
|
||||
version = "0.4.6"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "flask"
|
||||
version = "3.1.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "blinker" },
|
||||
{ name = "click" },
|
||||
{ name = "itsdangerous" },
|
||||
{ name = "jinja2" },
|
||||
{ name = "werkzeug" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/89/50/dff6380f1c7f84135484e176e0cac8690af72fa90e932ad2a0a60e28c69b/flask-3.1.0.tar.gz", hash = "sha256:5f873c5184c897c8d9d1b05df1e3d01b14910ce69607a117bd3277098a5836ac", size = 680824 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/af/47/93213ee66ef8fae3b93b3e29206f6b251e65c97bd91d8e1c5596ef15af0a/flask-3.1.0-py3-none-any.whl", hash = "sha256:d667207822eb83f1c4b50949b1623c8fc8d51f2341d65f72e1a1815397551136", size = 102979 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "itsdangerous"
|
||||
version = "2.2.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/9c/cb/8ac0172223afbccb63986cc25049b154ecfb5e85932587206f42317be31d/itsdangerous-2.2.0.tar.gz", hash = "sha256:e0050c0b7da1eea53ffaf149c0cfbb5c6e2e2b69c4bef22c81fa6eb73e5f6173", size = 54410 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/04/96/92447566d16df59b2a776c0fb82dbc4d9e07cd95062562af01e408583fc4/itsdangerous-2.2.0-py3-none-any.whl", hash = "sha256:c6242fc49e35958c8b15141343aa660db5fc54d4f13a1db01a3f5891b98700ef", size = 16234 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jinja2"
|
||||
version = "3.1.6"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "markupsafe" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "markupsafe"
|
||||
version = "3.0.2"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/b2/97/5d42485e71dfc078108a86d6de8fa46db44a1a9295e89c5d6d4a06e23a62/markupsafe-3.0.2.tar.gz", hash = "sha256:ee55d3edf80167e48ea11a923c7386f4669df67d7994554387f84e7d8b0a2bf0", size = 20537 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/6b/28/bbf83e3f76936960b850435576dd5e67034e200469571be53f69174a2dfd/MarkupSafe-3.0.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9025b4018f3a1314059769c7bf15441064b2207cb3f065e6ea1e7359cb46db9d", size = 14353 },
|
||||
{ url = "https://files.pythonhosted.org/packages/6c/30/316d194b093cde57d448a4c3209f22e3046c5bb2fb0820b118292b334be7/MarkupSafe-3.0.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:93335ca3812df2f366e80509ae119189886b0f3c2b81325d39efdb84a1e2ae93", size = 12392 },
|
||||
{ url = "https://files.pythonhosted.org/packages/f2/96/9cdafba8445d3a53cae530aaf83c38ec64c4d5427d975c974084af5bc5d2/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2cb8438c3cbb25e220c2ab33bb226559e7afb3baec11c4f218ffa7308603c832", size = 23984 },
|
||||
{ url = "https://files.pythonhosted.org/packages/f1/a4/aefb044a2cd8d7334c8a47d3fb2c9f328ac48cb349468cc31c20b539305f/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a123e330ef0853c6e822384873bef7507557d8e4a082961e1defa947aa59ba84", size = 23120 },
|
||||
{ url = "https://files.pythonhosted.org/packages/8d/21/5e4851379f88f3fad1de30361db501300d4f07bcad047d3cb0449fc51f8c/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1e084f686b92e5b83186b07e8a17fc09e38fff551f3602b249881fec658d3eca", size = 23032 },
|
||||
{ url = "https://files.pythonhosted.org/packages/00/7b/e92c64e079b2d0d7ddf69899c98842f3f9a60a1ae72657c89ce2655c999d/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d8213e09c917a951de9d09ecee036d5c7d36cb6cb7dbaece4c71a60d79fb9798", size = 24057 },
|
||||
{ url = "https://files.pythonhosted.org/packages/f9/ac/46f960ca323037caa0a10662ef97d0a4728e890334fc156b9f9e52bcc4ca/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:5b02fb34468b6aaa40dfc198d813a641e3a63b98c2b05a16b9f80b7ec314185e", size = 23359 },
|
||||
{ url = "https://files.pythonhosted.org/packages/69/84/83439e16197337b8b14b6a5b9c2105fff81d42c2a7c5b58ac7b62ee2c3b1/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:0bff5e0ae4ef2e1ae4fdf2dfd5b76c75e5c2fa4132d05fc1b0dabcd20c7e28c4", size = 23306 },
|
||||
{ url = "https://files.pythonhosted.org/packages/9a/34/a15aa69f01e2181ed8d2b685c0d2f6655d5cca2c4db0ddea775e631918cd/MarkupSafe-3.0.2-cp311-cp311-win32.whl", hash = "sha256:6c89876f41da747c8d3677a2b540fb32ef5715f97b66eeb0c6b66f5e3ef6f59d", size = 15094 },
|
||||
{ url = "https://files.pythonhosted.org/packages/da/b8/3a3bd761922d416f3dc5d00bfbed11f66b1ab89a0c2b6e887240a30b0f6b/MarkupSafe-3.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:70a87b411535ccad5ef2f1df5136506a10775d267e197e4cf531ced10537bd6b", size = 15521 },
|
||||
{ url = "https://files.pythonhosted.org/packages/22/09/d1f21434c97fc42f09d290cbb6350d44eb12f09cc62c9476effdb33a18aa/MarkupSafe-3.0.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:9778bd8ab0a994ebf6f84c2b949e65736d5575320a17ae8984a77fab08db94cf", size = 14274 },
|
||||
{ url = "https://files.pythonhosted.org/packages/6b/b0/18f76bba336fa5aecf79d45dcd6c806c280ec44538b3c13671d49099fdd0/MarkupSafe-3.0.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:846ade7b71e3536c4e56b386c2a47adf5741d2d8b94ec9dc3e92e5e1ee1e2225", size = 12348 },
|
||||
{ url = "https://files.pythonhosted.org/packages/e0/25/dd5c0f6ac1311e9b40f4af06c78efde0f3b5cbf02502f8ef9501294c425b/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1c99d261bd2d5f6b59325c92c73df481e05e57f19837bdca8413b9eac4bd8028", size = 24149 },
|
||||
{ url = "https://files.pythonhosted.org/packages/f3/f0/89e7aadfb3749d0f52234a0c8c7867877876e0a20b60e2188e9850794c17/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e17c96c14e19278594aa4841ec148115f9c7615a47382ecb6b82bd8fea3ab0c8", size = 23118 },
|
||||
{ url = "https://files.pythonhosted.org/packages/d5/da/f2eeb64c723f5e3777bc081da884b414671982008c47dcc1873d81f625b6/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:88416bd1e65dcea10bc7569faacb2c20ce071dd1f87539ca2ab364bf6231393c", size = 22993 },
|
||||
{ url = "https://files.pythonhosted.org/packages/da/0e/1f32af846df486dce7c227fe0f2398dc7e2e51d4a370508281f3c1c5cddc/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:2181e67807fc2fa785d0592dc2d6206c019b9502410671cc905d132a92866557", size = 24178 },
|
||||
{ url = "https://files.pythonhosted.org/packages/c4/f6/bb3ca0532de8086cbff5f06d137064c8410d10779c4c127e0e47d17c0b71/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:52305740fe773d09cffb16f8ed0427942901f00adedac82ec8b67752f58a1b22", size = 23319 },
|
||||
{ url = "https://files.pythonhosted.org/packages/a2/82/8be4c96ffee03c5b4a034e60a31294daf481e12c7c43ab8e34a1453ee48b/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:ad10d3ded218f1039f11a75f8091880239651b52e9bb592ca27de44eed242a48", size = 23352 },
|
||||
{ url = "https://files.pythonhosted.org/packages/51/ae/97827349d3fcffee7e184bdf7f41cd6b88d9919c80f0263ba7acd1bbcb18/MarkupSafe-3.0.2-cp312-cp312-win32.whl", hash = "sha256:0f4ca02bea9a23221c0182836703cbf8930c5e9454bacce27e767509fa286a30", size = 15097 },
|
||||
{ url = "https://files.pythonhosted.org/packages/c1/80/a61f99dc3a936413c3ee4e1eecac96c0da5ed07ad56fd975f1a9da5bc630/MarkupSafe-3.0.2-cp312-cp312-win_amd64.whl", hash = "sha256:8e06879fc22a25ca47312fbe7c8264eb0b662f6db27cb2d3bbbc74b1df4b9b87", size = 15601 },
|
||||
{ url = "https://files.pythonhosted.org/packages/83/0e/67eb10a7ecc77a0c2bbe2b0235765b98d164d81600746914bebada795e97/MarkupSafe-3.0.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:ba9527cdd4c926ed0760bc301f6728ef34d841f405abf9d4f959c478421e4efd", size = 14274 },
|
||||
{ url = "https://files.pythonhosted.org/packages/2b/6d/9409f3684d3335375d04e5f05744dfe7e9f120062c9857df4ab490a1031a/MarkupSafe-3.0.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:f8b3d067f2e40fe93e1ccdd6b2e1d16c43140e76f02fb1319a05cf2b79d99430", size = 12352 },
|
||||
{ url = "https://files.pythonhosted.org/packages/d2/f5/6eadfcd3885ea85fe2a7c128315cc1bb7241e1987443d78c8fe712d03091/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:569511d3b58c8791ab4c2e1285575265991e6d8f8700c7be0e88f86cb0672094", size = 24122 },
|
||||
{ url = "https://files.pythonhosted.org/packages/0c/91/96cf928db8236f1bfab6ce15ad070dfdd02ed88261c2afafd4b43575e9e9/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15ab75ef81add55874e7ab7055e9c397312385bd9ced94920f2802310c930396", size = 23085 },
|
||||
{ url = "https://files.pythonhosted.org/packages/c2/cf/c9d56af24d56ea04daae7ac0940232d31d5a8354f2b457c6d856b2057d69/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f3818cb119498c0678015754eba762e0d61e5b52d34c8b13d770f0719f7b1d79", size = 22978 },
|
||||
{ url = "https://files.pythonhosted.org/packages/2a/9f/8619835cd6a711d6272d62abb78c033bda638fdc54c4e7f4272cf1c0962b/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:cdb82a876c47801bb54a690c5ae105a46b392ac6099881cdfb9f6e95e4014c6a", size = 24208 },
|
||||
{ url = "https://files.pythonhosted.org/packages/f9/bf/176950a1792b2cd2102b8ffeb5133e1ed984547b75db47c25a67d3359f77/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:cabc348d87e913db6ab4aa100f01b08f481097838bdddf7c7a84b7575b7309ca", size = 23357 },
|
||||
{ url = "https://files.pythonhosted.org/packages/ce/4f/9a02c1d335caabe5c4efb90e1b6e8ee944aa245c1aaaab8e8a618987d816/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:444dcda765c8a838eaae23112db52f1efaf750daddb2d9ca300bcae1039adc5c", size = 23344 },
|
||||
{ url = "https://files.pythonhosted.org/packages/ee/55/c271b57db36f748f0e04a759ace9f8f759ccf22b4960c270c78a394f58be/MarkupSafe-3.0.2-cp313-cp313-win32.whl", hash = "sha256:bcf3e58998965654fdaff38e58584d8937aa3096ab5354d493c77d1fdd66d7a1", size = 15101 },
|
||||
{ url = "https://files.pythonhosted.org/packages/29/88/07df22d2dd4df40aba9f3e402e6dc1b8ee86297dddbad4872bd5e7b0094f/MarkupSafe-3.0.2-cp313-cp313-win_amd64.whl", hash = "sha256:e6a2a455bd412959b57a172ce6328d2dd1f01cb2135efda2e4576e8a23fa3b0f", size = 15603 },
|
||||
{ url = "https://files.pythonhosted.org/packages/62/6a/8b89d24db2d32d433dffcd6a8779159da109842434f1dd2f6e71f32f738c/MarkupSafe-3.0.2-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:b5a6b3ada725cea8a5e634536b1b01c30bcdcd7f9c6fff4151548d5bf6b3a36c", size = 14510 },
|
||||
{ url = "https://files.pythonhosted.org/packages/7a/06/a10f955f70a2e5a9bf78d11a161029d278eeacbd35ef806c3fd17b13060d/MarkupSafe-3.0.2-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:a904af0a6162c73e3edcb969eeeb53a63ceeb5d8cf642fade7d39e7963a22ddb", size = 12486 },
|
||||
{ url = "https://files.pythonhosted.org/packages/34/cf/65d4a571869a1a9078198ca28f39fba5fbb910f952f9dbc5220afff9f5e6/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4aa4e5faecf353ed117801a068ebab7b7e09ffb6e1d5e412dc852e0da018126c", size = 25480 },
|
||||
{ url = "https://files.pythonhosted.org/packages/0c/e3/90e9651924c430b885468b56b3d597cabf6d72be4b24a0acd1fa0e12af67/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0ef13eaeee5b615fb07c9a7dadb38eac06a0608b41570d8ade51c56539e509d", size = 23914 },
|
||||
{ url = "https://files.pythonhosted.org/packages/66/8c/6c7cf61f95d63bb866db39085150df1f2a5bd3335298f14a66b48e92659c/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d16a81a06776313e817c951135cf7340a3e91e8c1ff2fac444cfd75fffa04afe", size = 23796 },
|
||||
{ url = "https://files.pythonhosted.org/packages/bb/35/cbe9238ec3f47ac9a7c8b3df7a808e7cb50fe149dc7039f5f454b3fba218/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:6381026f158fdb7c72a168278597a5e3a5222e83ea18f543112b2662a9b699c5", size = 25473 },
|
||||
{ url = "https://files.pythonhosted.org/packages/e6/32/7621a4382488aa283cc05e8984a9c219abad3bca087be9ec77e89939ded9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:3d79d162e7be8f996986c064d1c7c817f6df3a77fe3d6859f6f9e7be4b8c213a", size = 24114 },
|
||||
{ url = "https://files.pythonhosted.org/packages/0d/80/0985960e4b89922cb5a0bac0ed39c5b96cbc1a536a99f30e8c220a996ed9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:131a3c7689c85f5ad20f9f6fb1b866f402c445b220c19fe4308c0b147ccd2ad9", size = 24098 },
|
||||
{ url = "https://files.pythonhosted.org/packages/82/78/fedb03c7d5380df2427038ec8d973587e90561b2d90cd472ce9254cf348b/MarkupSafe-3.0.2-cp313-cp313t-win32.whl", hash = "sha256:ba8062ed2cf21c07a9e295d5b8a2a5ce678b913b45fdf68c32d95d6c1291e0b6", size = 15208 },
|
||||
{ url = "https://files.pythonhosted.org/packages/4f/65/6079a46068dfceaeabb5dcad6d674f5f5c61a6fa5673746f42a9f4c233b3/MarkupSafe-3.0.2-cp313-cp313t-win_amd64.whl", hash = "sha256:e444a31f8db13eb18ada366ab3cf45fd4b31e4db1236a4448f68778c1d1a5a2f", size = 15739 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "werkzeug"
|
||||
version = "3.1.3"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "markupsafe" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/9f/69/83029f1f6300c5fb2471d621ab06f6ec6b3324685a2ce0f9777fd4a8b71e/werkzeug-3.1.3.tar.gz", hash = "sha256:60723ce945c19328679790e3282cc758aa4a6040e4bb330f53d30fa546d44746", size = 806925 }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/52/24/ab44c871b0f07f491e5d2ad12c9bd7358e527510618cb1b803a88e986db1/werkzeug-3.1.3-py3-none-any.whl", hash = "sha256:54b78bf3716d19a65be4fceccc0d1d7b89e608834989dfae50ea87564639213e", size = 224498 },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "wxocr"
|
||||
version = "0.1.0"
|
||||
source = { virtual = "." }
|
||||
dependencies = [
|
||||
{ name = "flask" },
|
||||
]
|
||||
|
||||
[package.metadata]
|
||||
requires-dist = [{ name = "flask", specifier = ">=3.1.0" }]
|
||||
Reference in New Issue
Block a user