add uv & page & testcase & docker

This commit is contained in:
yi_t
2025-03-25 19:17:14 +08:00
parent 6109b9834f
commit 118891e187
10 changed files with 670 additions and 13 deletions
+3
View File
@@ -0,0 +1,3 @@
.venv
__pycache__
temp/*
+1
View File
@@ -0,0 +1 @@
3.12
+1
View File
@@ -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
+15
View File
@@ -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
BIN
View File
Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

+55 -13
View File
@@ -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)
+9
View File
@@ -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",
]
+441
View File
@@ -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>
+4
View File
File diff suppressed because one or more lines are too long
Generated
+141
View File
@@ -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" }]