fix
This commit is contained in:
10
.gitignore
vendored
Normal file
10
.gitignore
vendored
Normal file
@@ -0,0 +1,10 @@
|
||||
# Python-generated files
|
||||
__pycache__/
|
||||
*.py[oc]
|
||||
build/
|
||||
dist/
|
||||
wheels/
|
||||
*.egg-info
|
||||
|
||||
# Virtual environments
|
||||
.venv
|
||||
1
.python-version
Normal file
1
.python-version
Normal file
@@ -0,0 +1 @@
|
||||
3.13
|
||||
88
README.md
Normal file
88
README.md
Normal file
@@ -0,0 +1,88 @@
|
||||
# Poker Task 3: EMD Distance Calculation on TURN
|
||||
|
||||
这是扑克Task 3的实现,专注于计算TURN阶段两手牌力分布的EMD (Earth Mover's Distance) 距离。
|
||||
|
||||
## 功能特点
|
||||
|
||||
- **TURN阶段扑克分析**: 基于已知的公共牌和底牌,计算每手牌在所有可能RIVER牌下的牌力分布
|
||||
- **EMD距离计算**: 使用scipy的wasserstein_distance函数计算两个牌力分布的EMD距离
|
||||
- **专业扑克评估**: 集成poker_task1的专业扑克牌评估系统,支持所有标准扑克手牌类型
|
||||
|
||||
## 安装和使用
|
||||
|
||||
### 安装依赖
|
||||
|
||||
```bash
|
||||
# 使用uv管理依赖(推荐)
|
||||
uv install
|
||||
|
||||
# 或使用pip
|
||||
pip install scipy
|
||||
```
|
||||
|
||||
### 运行程序
|
||||
|
||||
```bash
|
||||
# 默认示例运行
|
||||
python main.py
|
||||
|
||||
# 或使用uv
|
||||
uv run python main.py
|
||||
```
|
||||
|
||||
### 输入格式
|
||||
|
||||
程序接受以下格式的输入:
|
||||
```
|
||||
玩家1底牌(两张)
|
||||
玩家2底牌(两张)
|
||||
公共牌(TURN阶段,4张牌)
|
||||
```
|
||||
|
||||
示例:
|
||||
```
|
||||
As Ks
|
||||
7s 6s
|
||||
8h 9d 9c Qh
|
||||
```
|
||||
|
||||
## 运行测试
|
||||
|
||||
```bash
|
||||
# 运行所有测试
|
||||
uv run python -m pytest tests/ -v
|
||||
|
||||
# 或
|
||||
python -m pytest tests/ -v
|
||||
```
|
||||
|
||||
## 项目结构
|
||||
|
||||
```
|
||||
poker_task3/
|
||||
├── poker_task3/ # 主要代码包
|
||||
│ ├── card.py # 扑克牌类定义
|
||||
│ ├── hand_evaluator.py # 手牌评估器
|
||||
│ ├── hand_ranking.py # 手牌排名系统
|
||||
│ ├── emd.py # EMD距离计算
|
||||
│ ├── task3.py # Task 3主要实现
|
||||
│ └── __init__.py
|
||||
├── tests/ # 测试文件
|
||||
│ └── test_task3.py # Task 3测试
|
||||
├── main.py # 程序入口
|
||||
└── README.md # 项目文档
|
||||
```
|
||||
|
||||
## 算法原理
|
||||
|
||||
1. **牌力评估**: 对于每手底牌,遍历所有可能的RIVER牌(剩余47张牌)
|
||||
2. **分布构建**: 统计每种手牌类型(High Card到Royal Flush)的出现次数
|
||||
3. **EMD计算**: 使用scipy.stats.wasserstein_distance计算两个分布的EMD距离
|
||||
|
||||
## 测试覆盖
|
||||
|
||||
- 输入解析测试
|
||||
- 牌力分布计算测试
|
||||
- EMD距离计算测试
|
||||
- 错误处理测试
|
||||
- 不同牌力强度对比测试
|
||||
18
main.py
Normal file
18
main.py
Normal file
@@ -0,0 +1,18 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from poker_task3.task3 import runTask3
|
||||
|
||||
|
||||
def main():
|
||||
"""task3的主入口点"""
|
||||
print("=== task3:扑克牌TURN阶段EMD距离计算 ===")
|
||||
|
||||
distance = runTask3()
|
||||
if distance is not None:
|
||||
print(f"EMD距离: {distance}")
|
||||
|
||||
return distance
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
1
poker_task3/__init__.py
Normal file
1
poker_task3/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
"""Poker evaluation and EMD distance calculation package."""
|
||||
149
poker_task3/card.py
Normal file
149
poker_task3/card.py
Normal file
@@ -0,0 +1,149 @@
|
||||
"""
|
||||
Card module for poker game
|
||||
"""
|
||||
|
||||
from enum import Enum
|
||||
from typing import List, Tuple, Optional
|
||||
|
||||
|
||||
class Suit(Enum):
|
||||
SPADES = 's' # 黑桃
|
||||
HEARTS = 'h' # 红桃
|
||||
DIAMONDS = 'd' # 方块
|
||||
CLUBS = 'c' # 梅花
|
||||
|
||||
def __str__(self):
|
||||
return self.value
|
||||
|
||||
|
||||
class Rank(Enum):
|
||||
TWO = (2, '2')
|
||||
THREE = (3, '3')
|
||||
FOUR = (4, '4')
|
||||
FIVE = (5, '5')
|
||||
SIX = (6, '6')
|
||||
SEVEN = (7, '7')
|
||||
EIGHT = (8, '8')
|
||||
NINE = (9, '9')
|
||||
TEN = (10, 'T')
|
||||
JACK = (11, 'J')
|
||||
QUEEN = (12, 'Q')
|
||||
KING = (13, 'K')
|
||||
ACE = (14, 'A')
|
||||
|
||||
def __new__(cls, value, symbol):
|
||||
obj = object.__new__(cls)
|
||||
obj._value_ = value
|
||||
obj.numeric_value = value
|
||||
obj.symbol = symbol
|
||||
return obj
|
||||
|
||||
def __str__(self):
|
||||
return self.symbol
|
||||
|
||||
def __lt__(self, other):
|
||||
if not isinstance(other, Rank):
|
||||
return NotImplemented
|
||||
return self.numeric_value < other.numeric_value
|
||||
|
||||
def __le__(self, other):
|
||||
if not isinstance(other, Rank):
|
||||
return NotImplemented
|
||||
return self.numeric_value <= other.numeric_value
|
||||
|
||||
def __gt__(self, other):
|
||||
if not isinstance(other, Rank):
|
||||
return NotImplemented
|
||||
return self.numeric_value > other.numeric_value
|
||||
|
||||
def __ge__(self, other):
|
||||
if not isinstance(other, Rank):
|
||||
return NotImplemented
|
||||
return self.numeric_value >= other.numeric_value
|
||||
|
||||
|
||||
class Card:
|
||||
def __init__(self, rank: Rank, suit: Suit):
|
||||
self.rank = rank
|
||||
self.suit = suit
|
||||
|
||||
def __str__(self):
|
||||
return f"{self.rank}{self.suit}"
|
||||
|
||||
def __repr__(self):
|
||||
return f"Card({self.rank}, {self.suit})"
|
||||
|
||||
def __eq__(self, other):
|
||||
if not isinstance(other, Card):
|
||||
return False
|
||||
return self.rank == other.rank and self.suit == other.suit
|
||||
|
||||
def __hash__(self):
|
||||
return hash((self.rank, self.suit))
|
||||
|
||||
def __lt__(self, other):
|
||||
if not isinstance(other, Card):
|
||||
return NotImplemented
|
||||
return self.rank.numeric_value < other.rank.numeric_value
|
||||
|
||||
@classmethod
|
||||
def create_card(cls, card_str: str) -> 'Card':
|
||||
"""
|
||||
从字符串创建Card对象,例如 "As", "Kh", "2c"
|
||||
"""
|
||||
if len(card_str) != 2:
|
||||
raise ValueError(f"Invalid card string: {card_str}")
|
||||
|
||||
rank_char = card_str[0]
|
||||
suit_char = card_str[1].lower()
|
||||
|
||||
# 查找rank
|
||||
rank = None
|
||||
for r in Rank:
|
||||
if r.symbol == rank_char:
|
||||
rank = r
|
||||
break
|
||||
|
||||
if rank is None:
|
||||
raise ValueError(f"Invalid rank: {rank_char}")
|
||||
|
||||
# 查找suit
|
||||
suit = None
|
||||
for s in Suit:
|
||||
if s.value == suit_char:
|
||||
suit = s
|
||||
break
|
||||
|
||||
if suit is None:
|
||||
raise ValueError(f"Invalid suit: {suit_char}")
|
||||
|
||||
return cls(rank, suit)
|
||||
|
||||
@classmethod
|
||||
def parse_cards(cls, cards_str: str) -> List['Card']:
|
||||
"""
|
||||
从字符串解析多张牌,例如 "AsKs AhAdAc6s7s"
|
||||
"""
|
||||
cards_str = cards_str.strip()
|
||||
if cards_str == "":
|
||||
return []
|
||||
|
||||
# 将字符串分割成单独的牌
|
||||
card_strings = []
|
||||
i = 0
|
||||
while i < len(cards_str):
|
||||
if cards_str[i].isspace():
|
||||
i += 1
|
||||
continue
|
||||
|
||||
if i + 1 < len(cards_str):
|
||||
card_str = cards_str[i:i+2]
|
||||
card_strings.append(card_str)
|
||||
i += 2
|
||||
else:
|
||||
raise ValueError(f"Invalid card format at position {i}")
|
||||
result = []
|
||||
for card_str in card_strings:
|
||||
card_tmp = cls.create_card(card_str)
|
||||
result.append(card_tmp)
|
||||
return result
|
||||
132
poker_task3/hand_evaluator.py
Normal file
132
poker_task3/hand_evaluator.py
Normal file
@@ -0,0 +1,132 @@
|
||||
"""
|
||||
Hand evaluator module for poker game
|
||||
"""
|
||||
|
||||
from typing import List, Tuple, Dict
|
||||
from collections import Counter
|
||||
from itertools import combinations
|
||||
from .card import Card, Rank, Suit
|
||||
from .hand_ranking import HandRanking, HandType
|
||||
|
||||
|
||||
class HandEvaluator:
|
||||
"""
|
||||
手牌评估器类,用于评估扑克手牌
|
||||
"""
|
||||
|
||||
@staticmethod
|
||||
def evaluate_hand(cards) -> HandRanking:
|
||||
"""
|
||||
从7张牌中评估出最好的5张牌组合
|
||||
"""
|
||||
if len(cards) != 7:
|
||||
raise ValueError(f"Expected 7 cards, got {len(cards)}")
|
||||
|
||||
best_ranking = None
|
||||
best_cards = None
|
||||
|
||||
# 尝试所有可能的5张牌组合
|
||||
for five_cards in combinations(cards, 5):
|
||||
ranking = HandEvaluator._evaluate_five_cards(list(five_cards))
|
||||
|
||||
if best_ranking is None or ranking > best_ranking:
|
||||
best_ranking = ranking
|
||||
best_cards = list(five_cards)
|
||||
|
||||
# 更新最佳ranking的cards
|
||||
best_ranking.cards = best_cards
|
||||
return best_ranking
|
||||
|
||||
@staticmethod
|
||||
def _evaluate_five_cards(cards: List[Card]) -> HandRanking:
|
||||
"""
|
||||
评估5张牌的手牌类型
|
||||
"""
|
||||
if len(cards) != 5:
|
||||
raise ValueError(f"Expected 5 cards, got {len(cards)}")
|
||||
|
||||
# 按点数排序(降序)
|
||||
sorted_cards = sorted(cards, key=lambda c: c.rank.numeric_value, reverse=True)
|
||||
ranks = [card.rank for card in sorted_cards]
|
||||
suits = [card.suit for card in sorted_cards]
|
||||
|
||||
# 统计点数出现次数
|
||||
rank_counts = Counter(ranks)
|
||||
count_values = sorted(rank_counts.values(), reverse=True)
|
||||
|
||||
# 检查是否是同花
|
||||
is_flush = len(set(suits)) == 1
|
||||
|
||||
# 检查是否是顺子
|
||||
is_straight, straight_high = HandEvaluator._is_straight(ranks)
|
||||
|
||||
# 根据牌型返回相应的HandRanking
|
||||
if is_straight and is_flush:
|
||||
if straight_high == Rank.ACE and ranks == [Rank.ACE, Rank.KING, Rank.QUEEN, Rank.JACK, Rank.TEN]:
|
||||
return HandRanking(HandType.ROYAL_FLUSH, [Rank.ACE], sorted_cards) # 皇家同花顺
|
||||
else:
|
||||
return HandRanking(HandType.STRAIGHT_FLUSH, [straight_high], sorted_cards) # 同花顺
|
||||
|
||||
elif count_values == [4, 1]: # 四条
|
||||
quad_rank = [rank for rank, count in rank_counts.items() if count == 4][0]
|
||||
kicker = [rank for rank, count in rank_counts.items() if count == 1][0]
|
||||
return HandRanking(HandType.FOUR_OF_A_KIND, [quad_rank, kicker], sorted_cards)
|
||||
|
||||
elif count_values == [3, 2]: # 三条+一对
|
||||
trips_rank = [rank for rank, count in rank_counts.items() if count == 3][0]
|
||||
pair_rank = [rank for rank, count in rank_counts.items() if count == 2][0]
|
||||
return HandRanking(HandType.FULL_HOUSE, [trips_rank, pair_rank], sorted_cards)
|
||||
|
||||
elif is_flush: # 同花
|
||||
return HandRanking(HandType.FLUSH, ranks, sorted_cards)
|
||||
|
||||
elif is_straight: # 顺子
|
||||
return HandRanking(HandType.STRAIGHT, [straight_high], sorted_cards)
|
||||
|
||||
elif count_values == [3, 1, 1]: # 三条
|
||||
trips_rank = [rank for rank, count in rank_counts.items() if count == 3][0]
|
||||
kickers = sorted([rank for rank, count in rank_counts.items() if count == 1], reverse=True)
|
||||
return HandRanking(HandType.THREE_OF_A_KIND, [trips_rank] + kickers, sorted_cards)
|
||||
|
||||
elif count_values == [2, 2, 1]: # 两对
|
||||
pairs = sorted([rank for rank, count in rank_counts.items() if count == 2], reverse=True)
|
||||
kicker = [rank for rank, count in rank_counts.items() if count == 1][0]
|
||||
return HandRanking(HandType.TWO_PAIR, pairs + [kicker], sorted_cards)
|
||||
|
||||
elif count_values == [2, 1, 1, 1]: # 一对
|
||||
pair_rank = [rank for rank, count in rank_counts.items() if count == 2][0]
|
||||
kickers = sorted([rank for rank, count in rank_counts.items() if count == 1], reverse=True)
|
||||
return HandRanking(HandType.ONE_PAIR, [pair_rank] + kickers, sorted_cards)
|
||||
|
||||
else: # 高牌
|
||||
return HandRanking(HandType.HIGH_CARD, ranks, sorted_cards)
|
||||
|
||||
@staticmethod
|
||||
def _is_straight(ranks: List[Rank]) -> Tuple[bool, Rank]:
|
||||
"""
|
||||
检查是否是顺子,返回(是否是顺子, 最高牌)
|
||||
"""
|
||||
# 排序点数值
|
||||
values = sorted([rank.numeric_value for rank in ranks], reverse=True)
|
||||
|
||||
# 检查常规顺子
|
||||
is_regular_straight = True
|
||||
for i in range(1, len(values)):
|
||||
if values[i-1] - values[i] != 1:
|
||||
is_regular_straight = False
|
||||
break
|
||||
|
||||
if is_regular_straight:
|
||||
# 返回最高牌
|
||||
highest_rank = None
|
||||
for rank in ranks:
|
||||
if rank.numeric_value == values[0]:
|
||||
highest_rank = rank
|
||||
break
|
||||
return True, highest_rank
|
||||
|
||||
# 检查A-2-3-4-5的特殊顺子(轮子)
|
||||
if values == [14, 5, 4, 3, 2]: # A, 5, 4, 3, 2
|
||||
return True, Rank.FIVE # 在轮子中,5是最高牌
|
||||
|
||||
return False, None
|
||||
125
poker_task3/hand_ranking.py
Normal file
125
poker_task3/hand_ranking.py
Normal file
@@ -0,0 +1,125 @@
|
||||
"""
|
||||
Hand ranking module for poker game
|
||||
"""
|
||||
|
||||
from enum import Enum
|
||||
from typing import List, Tuple
|
||||
from .card import Card, Rank
|
||||
|
||||
|
||||
class HandType(Enum):
|
||||
"""
|
||||
手牌类型枚举,按强度排序
|
||||
"""
|
||||
HIGH_CARD = (1, "High Card")
|
||||
ONE_PAIR = (2, "Pair")
|
||||
TWO_PAIR = (3, "Two Pair")
|
||||
THREE_OF_A_KIND = (4, "Three of a Kind")
|
||||
STRAIGHT = (5, "Straight")
|
||||
FLUSH = (6, "Flush")
|
||||
FULL_HOUSE = (7, "Full House")
|
||||
FOUR_OF_A_KIND = (8, "Four of a Kind")
|
||||
STRAIGHT_FLUSH = (9, "Straight Flush")
|
||||
ROYAL_FLUSH = (10, "Royal Flush")
|
||||
|
||||
def __new__(cls, strength, name):
|
||||
obj = object.__new__(cls)
|
||||
obj._value_ = strength
|
||||
obj.strength = strength
|
||||
obj.type_name = name
|
||||
return obj
|
||||
|
||||
def __str__(self):
|
||||
return self.type_name
|
||||
|
||||
def __lt__(self, other):
|
||||
if not isinstance(other, HandType):
|
||||
return NotImplemented
|
||||
return self.strength < other.strength
|
||||
|
||||
def __le__(self, other):
|
||||
if not isinstance(other, HandType):
|
||||
return NotImplemented
|
||||
return self.strength <= other.strength
|
||||
|
||||
def __gt__(self, other):
|
||||
if not isinstance(other, HandType):
|
||||
return NotImplemented
|
||||
return self.strength > other.strength
|
||||
|
||||
def __ge__(self, other):
|
||||
if not isinstance(other, HandType):
|
||||
return NotImplemented
|
||||
return self.strength >= other.strength
|
||||
|
||||
|
||||
class HandRanking:
|
||||
"""
|
||||
手牌排名类,包含手牌类型和关键牌
|
||||
"""
|
||||
|
||||
def __init__(self, hand_type: HandType, key_ranks: List[Rank], cards: List[Card]):
|
||||
self.hand_type = hand_type
|
||||
self.key_ranks = key_ranks # 用于比较的关键点数
|
||||
self.cards = cards # 组成这个排名的5张牌
|
||||
|
||||
def __str__(self):
|
||||
"""
|
||||
返回手牌排名的字符串表示
|
||||
"""
|
||||
if self.hand_type == HandType.FOUR_OF_A_KIND:
|
||||
return f"Quad({self.key_ranks[0].symbol})"
|
||||
elif self.hand_type == HandType.FULL_HOUSE:
|
||||
return f"Full House({self.key_ranks[0].symbol} over {self.key_ranks[1].symbol})"
|
||||
elif self.hand_type == HandType.FLUSH:
|
||||
return f"Flush({self.key_ranks[0].symbol} high)"
|
||||
elif self.hand_type == HandType.STRAIGHT:
|
||||
return f"Straight({self.key_ranks[0].symbol} high)"
|
||||
elif self.hand_type == HandType.STRAIGHT_FLUSH:
|
||||
if self.key_ranks[0] == Rank.ACE:
|
||||
return "Royal Flush"
|
||||
else:
|
||||
return f"Straight Flush({self.key_ranks[0].symbol} high)"
|
||||
elif self.hand_type == HandType.ROYAL_FLUSH:
|
||||
return "Royal Flush"
|
||||
elif self.hand_type == HandType.THREE_OF_A_KIND:
|
||||
return f"Three of a Kind({self.key_ranks[0].symbol})"
|
||||
elif self.hand_type == HandType.TWO_PAIR:
|
||||
return f"Two Pair({self.key_ranks[0].symbol} and {self.key_ranks[1].symbol})"
|
||||
elif self.hand_type == HandType.ONE_PAIR:
|
||||
return f"Pair({self.key_ranks[0].symbol})"
|
||||
else: # HIGH_CARD
|
||||
return f"High Card({self.key_ranks[0].symbol})"
|
||||
|
||||
def __repr__(self):
|
||||
return f"HandRanking({self.hand_type}, {[r.symbol for r in self.key_ranks]})"
|
||||
|
||||
def __eq__(self, other):
|
||||
if not isinstance(other, HandRanking):
|
||||
return False
|
||||
return (self.hand_type == other.hand_type and
|
||||
self.key_ranks == other.key_ranks)
|
||||
|
||||
def __lt__(self, other):
|
||||
if not isinstance(other, HandRanking):
|
||||
return NotImplemented
|
||||
|
||||
# 首先比较手牌类型
|
||||
if self.hand_type != other.hand_type:
|
||||
return self.hand_type < other.hand_type
|
||||
|
||||
# 如果手牌类型相同,比较关键点数
|
||||
for self_rank, other_rank in zip(self.key_ranks, other.key_ranks):
|
||||
if self_rank != other_rank:
|
||||
return self_rank < other_rank
|
||||
|
||||
return False # 完全相等
|
||||
|
||||
def __le__(self, other):
|
||||
return self == other or self < other
|
||||
|
||||
def __gt__(self, other):
|
||||
return not self <= other
|
||||
|
||||
def __ge__(self, other):
|
||||
return not self < other
|
||||
139
poker_task3/task3.py
Normal file
139
poker_task3/task3.py
Normal file
@@ -0,0 +1,139 @@
|
||||
"""
|
||||
任务3:扑克牌TURN阶段EMD距离计算器
|
||||
"""
|
||||
from typing import List, Tuple, Union
|
||||
from scipy.stats import wasserstein_distance
|
||||
from .card import Card, Rank, Suit
|
||||
from .hand_evaluator import HandEvaluator
|
||||
from .hand_ranking import HandRanking, HandType
|
||||
|
||||
|
||||
def calEmd(Hist1: List[Union[int, float]],
|
||||
Hist2: List[Union[int, float]]) -> float:
|
||||
return wasserstein_distance(Hist1, Hist2)
|
||||
|
||||
|
||||
def parseTurnInput(input_text: str) -> Tuple[Union[List[Card], None], Union[List[Card], None], Union[List[Card], None]]:
|
||||
"""
|
||||
解析TURN阶段输入
|
||||
格式: "As Ks\n7s 6s\n8h 9d 9c Qh"
|
||||
返回: (手牌1, 手牌2, 公共牌)
|
||||
"""
|
||||
lines = [line.strip() for line in input_text.strip().split('\n') if line.strip()]
|
||||
|
||||
if len(lines) != 3:
|
||||
print("输入必须包含恰好3行:手牌1,手牌2,公共牌")
|
||||
return None, None, None
|
||||
|
||||
hand1_cards = Card.parse_cards(lines[0])
|
||||
hand2_cards = Card.parse_cards(lines[1])
|
||||
board_cards = Card.parse_cards(lines[2])
|
||||
|
||||
if len(hand1_cards) != 2:
|
||||
print("手牌1必须包含恰好2张牌")
|
||||
return None, None, None
|
||||
if len(hand2_cards) != 2:
|
||||
print("手牌2必须包含恰好2张牌")
|
||||
return None, None, None
|
||||
if len(board_cards) != 4:
|
||||
print("公共牌必须包含恰好4张牌(TURN阶段)")
|
||||
return None, None, None
|
||||
|
||||
return hand1_cards, hand2_cards, board_cards
|
||||
|
||||
|
||||
def calHandEquityHist(hole_cards, board_cards) -> List[float]:
|
||||
"""
|
||||
计算TURN阶段手牌的胜率分布
|
||||
基于每个可能河牌下的胜率创建分布,而不是绝对强度排名
|
||||
"""
|
||||
all_cards = hole_cards + board_cards # 总共6张牌
|
||||
|
||||
# 创建剩余牌组
|
||||
used_cards = set(all_cards)
|
||||
remaining_deck = []
|
||||
|
||||
for rank in Rank:
|
||||
for suit in Suit:
|
||||
card = Card(rank, suit)
|
||||
if card not in used_cards:
|
||||
remaining_deck.append(card)
|
||||
|
||||
# 为每个可能的河牌计算该手牌的胜率
|
||||
winrates = []
|
||||
|
||||
for river_card in remaining_deck:
|
||||
# 当前手牌在这个河牌下的最终七张牌
|
||||
current_seven_cards = all_cards + [river_card]
|
||||
current_hand_ranking = HandEvaluator.evaluate_hand(current_seven_cards)
|
||||
|
||||
# 计算对抗所有可能对手牌的胜率
|
||||
wins = 0
|
||||
total_opponents = 0
|
||||
|
||||
# 生成所有可能的对手双牌组合
|
||||
used_cards_with_river = used_cards | {river_card}
|
||||
available_cards = [card for card in remaining_deck if card != river_card]
|
||||
|
||||
for i in range(len(available_cards)):
|
||||
for j in range(i + 1, len(available_cards)):
|
||||
opponent_cards = [available_cards[i], available_cards[j]]
|
||||
opponent_seven_cards = opponent_cards + board_cards + [river_card]
|
||||
opponent_hand_ranking = HandEvaluator.evaluate_hand(opponent_seven_cards)
|
||||
|
||||
total_opponents += 1
|
||||
if current_hand_ranking > opponent_hand_ranking:
|
||||
wins += 1
|
||||
elif current_hand_ranking == opponent_hand_ranking:
|
||||
wins += 0.5 # 平局
|
||||
|
||||
winrate = wins / total_opponents if total_opponents > 0 else 0.0
|
||||
winrates.append(winrate)
|
||||
|
||||
num_bins = 30
|
||||
hist = [0.0] * num_bins
|
||||
|
||||
for winrate in winrates:
|
||||
bin_index = min(int(winrate * num_bins), num_bins - 1)
|
||||
hist[bin_index] += 1.0
|
||||
|
||||
total = sum(hist)
|
||||
if total > 0:
|
||||
hist = [x / total for x in hist]
|
||||
|
||||
return hist
|
||||
|
||||
|
||||
def calPokerEmdTurn(input_text: str) -> Union[float, None]:
|
||||
"""
|
||||
计算TURN阶段两手牌的EMD距离
|
||||
"""
|
||||
hand1_cards, hand2_cards, board_cards = parseTurnInput(input_text)
|
||||
|
||||
# 检查解析是否成功
|
||||
if hand1_cards is None or hand2_cards is None or board_cards is None:
|
||||
return None
|
||||
|
||||
# 计算胜率分布
|
||||
dist1 = calHandEquityHist(hand1_cards, board_cards)
|
||||
dist2 = calHandEquityHist(hand2_cards, board_cards)
|
||||
|
||||
# 计算EMD距离
|
||||
emd_distance = calEmd(dist1, dist2)
|
||||
|
||||
return emd_distance
|
||||
|
||||
|
||||
def runTask3():
|
||||
"""运行任务3的示例输入"""
|
||||
example_input = """
|
||||
As Ks
|
||||
7s 6s
|
||||
8h 9d 9c Qh"""
|
||||
|
||||
try:
|
||||
distance = calPokerEmdTurn(example_input)
|
||||
return distance
|
||||
except Exception as e:
|
||||
print(f"错误: {e}")
|
||||
return None
|
||||
15
pyproject.toml
Normal file
15
pyproject.toml
Normal file
@@ -0,0 +1,15 @@
|
||||
[project]
|
||||
name = "poker-task3"
|
||||
version = "0.1.0"
|
||||
description = "Add your description here"
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.13"
|
||||
dependencies = [
|
||||
"pytest>=8.4.2",
|
||||
"scipy>=1.16.2",
|
||||
]
|
||||
|
||||
[dependency-groups]
|
||||
dev = [
|
||||
"pytest>=8.4.2",
|
||||
]
|
||||
100
tests/test_task3.py
Normal file
100
tests/test_task3.py
Normal file
@@ -0,0 +1,100 @@
|
||||
"""测试任务3 - 使用poker_task1进行扑克牌EMD距离计算"""
|
||||
import sys
|
||||
import os
|
||||
sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))
|
||||
|
||||
# 从poker_task3.task3模块导入函数
|
||||
from poker_task3.task3 import parseTurnInput, calHandEquityHist, calPokerEmdTurn
|
||||
from poker_task3.card import Card
|
||||
|
||||
|
||||
def test_parse_turn_input():
|
||||
"""测试TURN阶段输入解析"""
|
||||
input_text = """
|
||||
As Ks
|
||||
7s 6s
|
||||
8h 9d 9c Qh
|
||||
"""
|
||||
|
||||
hand1, hand2, board = parseTurnInput(input_text)
|
||||
|
||||
assert len(hand1) == 2
|
||||
assert len(hand2) == 2
|
||||
assert len(board) == 4
|
||||
|
||||
assert str(hand1[0]) == "As"
|
||||
assert str(hand1[1]) == "Ks"
|
||||
assert str(hand2[0]) == "7s"
|
||||
assert str(hand2[1]) == "6s"
|
||||
|
||||
print("✓ TURN输入解析测试通过")
|
||||
|
||||
|
||||
def test_hand_equity_Hist():
|
||||
"""测试手牌胜率分布计算"""
|
||||
hole_cards = Card.parse_cards("As Ks")
|
||||
board_cards = Card.parse_cards("8h 9d 9c Qh")
|
||||
|
||||
Hist = calHandEquityHist(hole_cards, board_cards)
|
||||
|
||||
assert len(Hist) == 30 # 胜率分布的区间数量(改进为30个区间)
|
||||
assert all(isinstance(x, (int, float)) for x in Hist)
|
||||
assert all(x >= 0 for x in Hist) # 所有值都应该是非负的
|
||||
assert abs(sum(Hist) - 1.0) < 1e-10 # 分布总和应该为1(标准化后)
|
||||
|
||||
print("✓ 手牌胜率分布测试通过")
|
||||
|
||||
|
||||
def test_poker_emd_calculation():
|
||||
"""测试主要的EMD计算函数"""
|
||||
input_text = """As Ks
|
||||
7s 6s
|
||||
8h 9d 9c Qh"""
|
||||
|
||||
distance = calPokerEmdTurn(input_text)
|
||||
|
||||
assert isinstance(distance, (int, float))
|
||||
assert distance >= 0 # EMD总是非负的
|
||||
|
||||
print(f"✓ 扑克牌EMD计算测试通过 (距离: {distance:.3f})")
|
||||
|
||||
|
||||
def test_different_hand_strengths():
|
||||
"""测试不同强度手牌的EMD"""
|
||||
# 强牌 vs 弱牌
|
||||
input_text = """As Ks
|
||||
2c 3d
|
||||
8h 9d 9c Qh"""
|
||||
|
||||
distance = calPokerEmdTurn(input_text)
|
||||
|
||||
assert isinstance(distance, (int, float))
|
||||
assert distance >= 0 # 应该有非负距离
|
||||
|
||||
print(f"✓ 不同手牌强度测试通过 (距离: {distance:.3f})")
|
||||
|
||||
|
||||
def test_error_handling():
|
||||
"""测试无效输入的错误处理"""
|
||||
# 错误的行数
|
||||
result = parseTurnInput("As Ks\n7s 6s")
|
||||
assert result == (None, None, None), "应该返回(None, None, None)"
|
||||
|
||||
# 手牌中牌数错误
|
||||
result = parseTurnInput("As Ks Qs\n7s 6s\n8h 9d 9c Qh")
|
||||
assert result == (None, None, None), "应该返回(None, None, None)"
|
||||
|
||||
# 公共牌数错误
|
||||
result = parseTurnInput("As Ks\n7s 6s\n8h 9d 9c")
|
||||
assert result == (None, None, None), "应该返回(None, None, None)"
|
||||
|
||||
print("✓ 错误处理测试通过")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
test_parse_turn_input()
|
||||
test_hand_equity_Hist()
|
||||
test_poker_emd_calculation()
|
||||
test_different_hand_strengths()
|
||||
test_error_handling()
|
||||
print("所有任务3测试通过! ✓")
|
||||
190
uv.lock
generated
Normal file
190
uv.lock
generated
Normal file
@@ -0,0 +1,190 @@
|
||||
version = 1
|
||||
revision = 3
|
||||
requires-python = ">=3.13"
|
||||
|
||||
[[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, upload-time = "2022-10-25T02:36:22.414Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "iniconfig"
|
||||
version = "2.1.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/f2/97/ebf4da567aa6827c909642694d71c9fcf53e5b504f2d96afea02718862f3/iniconfig-2.1.0.tar.gz", hash = "sha256:3abbd2e30b36733fee78f9c7f7308f2d0050e88f0087fd25c2645f63c773e1c7", size = 4793, upload-time = "2025-03-19T20:09:59.721Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/2c/e1/e6716421ea10d38022b952c159d5161ca1193197fb744506875fbb87ea7b/iniconfig-2.1.0-py3-none-any.whl", hash = "sha256:9deba5723312380e77435581c6bf4935c94cbfab9b1ed33ef8d238ea168eb760", size = 6050, upload-time = "2025-03-19T20:10:01.071Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "numpy"
|
||||
version = "2.3.3"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/d0/19/95b3d357407220ed24c139018d2518fab0a61a948e68286a25f1a4d049ff/numpy-2.3.3.tar.gz", hash = "sha256:ddc7c39727ba62b80dfdbedf400d1c10ddfa8eefbd7ec8dcb118be8b56d31029", size = 20576648, upload-time = "2025-09-09T16:54:12.543Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/7d/b9/984c2b1ee61a8b803bf63582b4ac4242cf76e2dbd663efeafcb620cc0ccb/numpy-2.3.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f5415fb78995644253370985342cd03572ef8620b934da27d77377a2285955bf", size = 20949588, upload-time = "2025-09-09T15:56:59.087Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a6/e4/07970e3bed0b1384d22af1e9912527ecbeb47d3b26e9b6a3bced068b3bea/numpy-2.3.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:d00de139a3324e26ed5b95870ce63be7ec7352171bc69a4cf1f157a48e3eb6b7", size = 14177802, upload-time = "2025-09-09T15:57:01.73Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/35/c7/477a83887f9de61f1203bad89cf208b7c19cc9fef0cebef65d5a1a0619f2/numpy-2.3.3-cp313-cp313-macosx_14_0_arm64.whl", hash = "sha256:9dc13c6a5829610cc07422bc74d3ac083bd8323f14e2827d992f9e52e22cd6a6", size = 5106537, upload-time = "2025-09-09T15:57:03.765Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/52/47/93b953bd5866a6f6986344d045a207d3f1cfbad99db29f534ea9cee5108c/numpy-2.3.3-cp313-cp313-macosx_14_0_x86_64.whl", hash = "sha256:d79715d95f1894771eb4e60fb23f065663b2298f7d22945d66877aadf33d00c7", size = 6640743, upload-time = "2025-09-09T15:57:07.921Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/23/83/377f84aaeb800b64c0ef4de58b08769e782edcefa4fea712910b6f0afd3c/numpy-2.3.3-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:952cfd0748514ea7c3afc729a0fc639e61655ce4c55ab9acfab14bda4f402b4c", size = 14278881, upload-time = "2025-09-09T15:57:11.349Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9a/a5/bf3db6e66c4b160d6ea10b534c381a1955dfab34cb1017ea93aa33c70ed3/numpy-2.3.3-cp313-cp313-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5b83648633d46f77039c29078751f80da65aa64d5622a3cd62aaef9d835b6c93", size = 16636301, upload-time = "2025-09-09T15:57:14.245Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a2/59/1287924242eb4fa3f9b3a2c30400f2e17eb2707020d1c5e3086fe7330717/numpy-2.3.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:b001bae8cea1c7dfdb2ae2b017ed0a6f2102d7a70059df1e338e307a4c78a8ae", size = 16053645, upload-time = "2025-09-09T15:57:16.534Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e6/93/b3d47ed882027c35e94ac2320c37e452a549f582a5e801f2d34b56973c97/numpy-2.3.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:8e9aced64054739037d42fb84c54dd38b81ee238816c948c8f3ed134665dcd86", size = 18578179, upload-time = "2025-09-09T15:57:18.883Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/20/d9/487a2bccbf7cc9d4bfc5f0f197761a5ef27ba870f1e3bbb9afc4bbe3fcc2/numpy-2.3.3-cp313-cp313-win32.whl", hash = "sha256:9591e1221db3f37751e6442850429b3aabf7026d3b05542d102944ca7f00c8a8", size = 6312250, upload-time = "2025-09-09T15:57:21.296Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1b/b5/263ebbbbcede85028f30047eab3d58028d7ebe389d6493fc95ae66c636ab/numpy-2.3.3-cp313-cp313-win_amd64.whl", hash = "sha256:f0dadeb302887f07431910f67a14d57209ed91130be0adea2f9793f1a4f817cf", size = 12783269, upload-time = "2025-09-09T15:57:23.034Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fa/75/67b8ca554bbeaaeb3fac2e8bce46967a5a06544c9108ec0cf5cece559b6c/numpy-2.3.3-cp313-cp313-win_arm64.whl", hash = "sha256:3c7cf302ac6e0b76a64c4aecf1a09e51abd9b01fc7feee80f6c43e3ab1b1dbc5", size = 10195314, upload-time = "2025-09-09T15:57:25.045Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/11/d0/0d1ddec56b162042ddfafeeb293bac672de9b0cfd688383590090963720a/numpy-2.3.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:eda59e44957d272846bb407aad19f89dc6f58fecf3504bd144f4c5cf81a7eacc", size = 21048025, upload-time = "2025-09-09T15:57:27.257Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/36/9e/1996ca6b6d00415b6acbdd3c42f7f03ea256e2c3f158f80bd7436a8a19f3/numpy-2.3.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:823d04112bc85ef5c4fda73ba24e6096c8f869931405a80aa8b0e604510a26bc", size = 14301053, upload-time = "2025-09-09T15:57:30.077Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/05/24/43da09aa764c68694b76e84b3d3f0c44cb7c18cdc1ba80e48b0ac1d2cd39/numpy-2.3.3-cp313-cp313t-macosx_14_0_arm64.whl", hash = "sha256:40051003e03db4041aa325da2a0971ba41cf65714e65d296397cc0e32de6018b", size = 5229444, upload-time = "2025-09-09T15:57:32.733Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bc/14/50ffb0f22f7218ef8af28dd089f79f68289a7a05a208db9a2c5dcbe123c1/numpy-2.3.3-cp313-cp313t-macosx_14_0_x86_64.whl", hash = "sha256:6ee9086235dd6ab7ae75aba5662f582a81ced49f0f1c6de4260a78d8f2d91a19", size = 6738039, upload-time = "2025-09-09T15:57:34.328Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/55/52/af46ac0795e09657d45a7f4db961917314377edecf66db0e39fa7ab5c3d3/numpy-2.3.3-cp313-cp313t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:94fcaa68757c3e2e668ddadeaa86ab05499a70725811e582b6a9858dd472fb30", size = 14352314, upload-time = "2025-09-09T15:57:36.255Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a7/b1/dc226b4c90eb9f07a3fff95c2f0db3268e2e54e5cce97c4ac91518aee71b/numpy-2.3.3-cp313-cp313t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:da1a74b90e7483d6ce5244053399a614b1d6b7bc30a60d2f570e5071f8959d3e", size = 16701722, upload-time = "2025-09-09T15:57:38.622Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9d/9d/9d8d358f2eb5eced14dba99f110d83b5cd9a4460895230f3b396ad19a323/numpy-2.3.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:2990adf06d1ecee3b3dcbb4977dfab6e9f09807598d647f04d385d29e7a3c3d3", size = 16132755, upload-time = "2025-09-09T15:57:41.16Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b6/27/b3922660c45513f9377b3fb42240bec63f203c71416093476ec9aa0719dc/numpy-2.3.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:ed635ff692483b8e3f0fcaa8e7eb8a75ee71aa6d975388224f70821421800cea", size = 18651560, upload-time = "2025-09-09T15:57:43.459Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/5b/8e/3ab61a730bdbbc201bb245a71102aa609f0008b9ed15255500a99cd7f780/numpy-2.3.3-cp313-cp313t-win32.whl", hash = "sha256:a333b4ed33d8dc2b373cc955ca57babc00cd6f9009991d9edc5ddbc1bac36bcd", size = 6442776, upload-time = "2025-09-09T15:57:45.793Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1c/3a/e22b766b11f6030dc2decdeff5c2fb1610768055603f9f3be88b6d192fb2/numpy-2.3.3-cp313-cp313t-win_amd64.whl", hash = "sha256:4384a169c4d8f97195980815d6fcad04933a7e1ab3b530921c3fef7a1c63426d", size = 12927281, upload-time = "2025-09-09T15:57:47.492Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/7b/42/c2e2bc48c5e9b2a83423f99733950fbefd86f165b468a3d85d52b30bf782/numpy-2.3.3-cp313-cp313t-win_arm64.whl", hash = "sha256:75370986cc0bc66f4ce5110ad35aae6d182cc4ce6433c40ad151f53690130bf1", size = 10265275, upload-time = "2025-09-09T15:57:49.647Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/6b/01/342ad585ad82419b99bcf7cebe99e61da6bedb89e213c5fd71acc467faee/numpy-2.3.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:cd052f1fa6a78dee696b58a914b7229ecfa41f0a6d96dc663c1220a55e137593", size = 20951527, upload-time = "2025-09-09T15:57:52.006Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ef/d8/204e0d73fc1b7a9ee80ab1fe1983dd33a4d64a4e30a05364b0208e9a241a/numpy-2.3.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:414a97499480067d305fcac9716c29cf4d0d76db6ebf0bf3cbce666677f12652", size = 14186159, upload-time = "2025-09-09T15:57:54.407Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/22/af/f11c916d08f3a18fb8ba81ab72b5b74a6e42ead4c2846d270eb19845bf74/numpy-2.3.3-cp314-cp314-macosx_14_0_arm64.whl", hash = "sha256:50a5fe69f135f88a2be9b6ca0481a68a136f6febe1916e4920e12f1a34e708a7", size = 5114624, upload-time = "2025-09-09T15:57:56.5Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fb/11/0ed919c8381ac9d2ffacd63fd1f0c34d27e99cab650f0eb6f110e6ae4858/numpy-2.3.3-cp314-cp314-macosx_14_0_x86_64.whl", hash = "sha256:b912f2ed2b67a129e6a601e9d93d4fa37bef67e54cac442a2f588a54afe5c67a", size = 6642627, upload-time = "2025-09-09T15:57:58.206Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ee/83/deb5f77cb0f7ba6cb52b91ed388b47f8f3c2e9930d4665c600408d9b90b9/numpy-2.3.3-cp314-cp314-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9e318ee0596d76d4cb3d78535dc005fa60e5ea348cd131a51e99d0bdbe0b54fe", size = 14296926, upload-time = "2025-09-09T15:58:00.035Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/77/cc/70e59dcb84f2b005d4f306310ff0a892518cc0c8000a33d0e6faf7ca8d80/numpy-2.3.3-cp314-cp314-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ce020080e4a52426202bdb6f7691c65bb55e49f261f31a8f506c9f6bc7450421", size = 16638958, upload-time = "2025-09-09T15:58:02.738Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b6/5a/b2ab6c18b4257e099587d5b7f903317bd7115333ad8d4ec4874278eafa61/numpy-2.3.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:e6687dc183aa55dae4a705b35f9c0f8cb178bcaa2f029b241ac5356221d5c021", size = 16071920, upload-time = "2025-09-09T15:58:05.029Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/b8/f1/8b3fdc44324a259298520dd82147ff648979bed085feeacc1250ef1656c0/numpy-2.3.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:d8f3b1080782469fdc1718c4ed1d22549b5fb12af0d57d35e992158a772a37cf", size = 18577076, upload-time = "2025-09-09T15:58:07.745Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/f0/a1/b87a284fb15a42e9274e7fcea0dad259d12ddbf07c1595b26883151ca3b4/numpy-2.3.3-cp314-cp314-win32.whl", hash = "sha256:cb248499b0bc3be66ebd6578b83e5acacf1d6cb2a77f2248ce0e40fbec5a76d0", size = 6366952, upload-time = "2025-09-09T15:58:10.096Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/70/5f/1816f4d08f3b8f66576d8433a66f8fa35a5acfb3bbd0bf6c31183b003f3d/numpy-2.3.3-cp314-cp314-win_amd64.whl", hash = "sha256:691808c2b26b0f002a032c73255d0bd89751425f379f7bcd22d140db593a96e8", size = 12919322, upload-time = "2025-09-09T15:58:12.138Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8c/de/072420342e46a8ea41c324a555fa90fcc11637583fb8df722936aed1736d/numpy-2.3.3-cp314-cp314-win_arm64.whl", hash = "sha256:9ad12e976ca7b10f1774b03615a2a4bab8addce37ecc77394d8e986927dc0dfe", size = 10478630, upload-time = "2025-09-09T15:58:14.64Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d5/df/ee2f1c0a9de7347f14da5dd3cd3c3b034d1b8607ccb6883d7dd5c035d631/numpy-2.3.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:9cc48e09feb11e1db00b320e9d30a4151f7369afb96bd0e48d942d09da3a0d00", size = 21047987, upload-time = "2025-09-09T15:58:16.889Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d6/92/9453bdc5a4e9e69cf4358463f25e8260e2ffc126d52e10038b9077815989/numpy-2.3.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:901bf6123879b7f251d3631967fd574690734236075082078e0571977c6a8e6a", size = 14301076, upload-time = "2025-09-09T15:58:20.343Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/13/77/1447b9eb500f028bb44253105bd67534af60499588a5149a94f18f2ca917/numpy-2.3.3-cp314-cp314t-macosx_14_0_arm64.whl", hash = "sha256:7f025652034199c301049296b59fa7d52c7e625017cae4c75d8662e377bf487d", size = 5229491, upload-time = "2025-09-09T15:58:22.481Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3d/f9/d72221b6ca205f9736cb4b2ce3b002f6e45cd67cd6a6d1c8af11a2f0b649/numpy-2.3.3-cp314-cp314t-macosx_14_0_x86_64.whl", hash = "sha256:533ca5f6d325c80b6007d4d7fb1984c303553534191024ec6a524a4c92a5935a", size = 6737913, upload-time = "2025-09-09T15:58:24.569Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3c/5f/d12834711962ad9c46af72f79bb31e73e416ee49d17f4c797f72c96b6ca5/numpy-2.3.3-cp314-cp314t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0edd58682a399824633b66885d699d7de982800053acf20be1eaa46d92009c54", size = 14352811, upload-time = "2025-09-09T15:58:26.416Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a1/0d/fdbec6629d97fd1bebed56cd742884e4eead593611bbe1abc3eb40d304b2/numpy-2.3.3-cp314-cp314t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:367ad5d8fbec5d9296d18478804a530f1191e24ab4d75ab408346ae88045d25e", size = 16702689, upload-time = "2025-09-09T15:58:28.831Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9b/09/0a35196dc5575adde1eb97ddfbc3e1687a814f905377621d18ca9bc2b7dd/numpy-2.3.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:8f6ac61a217437946a1fa48d24c47c91a0c4f725237871117dea264982128097", size = 16133855, upload-time = "2025-09-09T15:58:31.349Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/7a/ca/c9de3ea397d576f1b6753eaa906d4cdef1bf97589a6d9825a349b4729cc2/numpy-2.3.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:179a42101b845a816d464b6fe9a845dfaf308fdfc7925387195570789bb2c970", size = 18652520, upload-time = "2025-09-09T15:58:33.762Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fd/c2/e5ed830e08cd0196351db55db82f65bc0ab05da6ef2b72a836dcf1936d2f/numpy-2.3.3-cp314-cp314t-win32.whl", hash = "sha256:1250c5d3d2562ec4174bce2e3a1523041595f9b651065e4a4473f5f48a6bc8a5", size = 6515371, upload-time = "2025-09-09T15:58:36.04Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/47/c7/b0f6b5b67f6788a0725f744496badbb604d226bf233ba716683ebb47b570/numpy-2.3.3-cp314-cp314t-win_amd64.whl", hash = "sha256:b37a0b2e5935409daebe82c1e42274d30d9dd355852529eab91dab8dcca7419f", size = 13112576, upload-time = "2025-09-09T15:58:37.927Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/06/b9/33bba5ff6fb679aa0b1f8a07e853f002a6b04b9394db3069a1270a7784ca/numpy-2.3.3-cp314-cp314t-win_arm64.whl", hash = "sha256:78c9f6560dc7e6b3990e32df7ea1a50bbd0e2a111e05209963f5ddcab7073b0b", size = 10545953, upload-time = "2025-09-09T15:58:40.576Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "packaging"
|
||||
version = "25.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/a1/d4/1fc4078c65507b51b96ca8f8c3ba19e6a61c8253c72794544580a7b6c24d/packaging-25.0.tar.gz", hash = "sha256:d443872c98d677bf60f6a1f2f8c1cb748e8fe762d2bf9d3148b5599295b0fc4f", size = 165727, upload-time = "2025-04-19T11:48:59.673Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895e234f4d07f7545804097de4b666afd8/packaging-25.0-py3-none-any.whl", hash = "sha256:29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484", size = 66469, upload-time = "2025-04-19T11:48:57.875Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pluggy"
|
||||
version = "1.6.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "poker-task3"
|
||||
version = "0.1.0"
|
||||
source = { virtual = "." }
|
||||
dependencies = [
|
||||
{ name = "pytest" },
|
||||
{ name = "scipy" },
|
||||
]
|
||||
|
||||
[package.dev-dependencies]
|
||||
dev = [
|
||||
{ name = "pytest" },
|
||||
]
|
||||
|
||||
[package.metadata]
|
||||
requires-dist = [
|
||||
{ name = "pytest", specifier = ">=8.4.2" },
|
||||
{ name = "scipy", specifier = ">=1.16.2" },
|
||||
]
|
||||
|
||||
[package.metadata.requires-dev]
|
||||
dev = [{ name = "pytest", specifier = ">=8.4.2" }]
|
||||
|
||||
[[package]]
|
||||
name = "pygments"
|
||||
version = "2.19.2"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/b0/77/a5b8c569bf593b0140bde72ea885a803b82086995367bf2037de0159d924/pygments-2.19.2.tar.gz", hash = "sha256:636cb2477cec7f8952536970bc533bc43743542f70392ae026374600add5b887", size = 4968631, upload-time = "2025-06-21T13:39:12.283Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pytest"
|
||||
version = "8.4.2"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "colorama", marker = "sys_platform == 'win32'" },
|
||||
{ name = "iniconfig" },
|
||||
{ name = "packaging" },
|
||||
{ name = "pluggy" },
|
||||
{ name = "pygments" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/a3/5c/00a0e072241553e1a7496d638deababa67c5058571567b92a7eaa258397c/pytest-8.4.2.tar.gz", hash = "sha256:86c0d0b93306b961d58d62a4db4879f27fe25513d4b969df351abdddb3c30e01", size = 1519618, upload-time = "2025-09-04T14:34:22.711Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/a8/a4/20da314d277121d6534b3a980b29035dcd51e6744bd79075a6ce8fa4eb8d/pytest-8.4.2-py3-none-any.whl", hash = "sha256:872f880de3fc3a5bdc88a11b39c9710c3497a547cfa9320bc3c5e62fbf272e79", size = 365750, upload-time = "2025-09-04T14:34:20.226Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "scipy"
|
||||
version = "1.16.2"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "numpy" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/4c/3b/546a6f0bfe791bbb7f8d591613454d15097e53f906308ec6f7c1ce588e8e/scipy-1.16.2.tar.gz", hash = "sha256:af029b153d243a80afb6eabe40b0a07f8e35c9adc269c019f364ad747f826a6b", size = 30580599, upload-time = "2025-09-11T17:48:08.271Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/c1/27/c5b52f1ee81727a9fc457f5ac1e9bf3d6eab311805ea615c83c27ba06400/scipy-1.16.2-cp313-cp313-macosx_10_14_x86_64.whl", hash = "sha256:84f7bf944b43e20b8a894f5fe593976926744f6c185bacfcbdfbb62736b5cc70", size = 36604856, upload-time = "2025-09-11T17:41:47.695Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/32/a9/15c20d08e950b540184caa8ced675ba1128accb0e09c653780ba023a4110/scipy-1.16.2-cp313-cp313-macosx_12_0_arm64.whl", hash = "sha256:5c39026d12edc826a1ef2ad35ad1e6d7f087f934bb868fc43fa3049c8b8508f9", size = 28864626, upload-time = "2025-09-11T17:41:52.642Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4c/fc/ea36098df653cca26062a627c1a94b0de659e97127c8491e18713ca0e3b9/scipy-1.16.2-cp313-cp313-macosx_14_0_arm64.whl", hash = "sha256:e52729ffd45b68777c5319560014d6fd251294200625d9d70fd8626516fc49f5", size = 20855689, upload-time = "2025-09-11T17:41:57.886Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/dc/6f/d0b53be55727f3e6d7c72687ec18ea6d0047cf95f1f77488b99a2bafaee1/scipy-1.16.2-cp313-cp313-macosx_14_0_x86_64.whl", hash = "sha256:024dd4a118cccec09ca3209b7e8e614931a6ffb804b2a601839499cb88bdf925", size = 23512151, upload-time = "2025-09-11T17:42:02.303Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/11/85/bf7dab56e5c4b1d3d8eef92ca8ede788418ad38a7dc3ff50262f00808760/scipy-1.16.2-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:7a5dc7ee9c33019973a470556081b0fd3c9f4c44019191039f9769183141a4d9", size = 33329824, upload-time = "2025-09-11T17:42:07.549Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/da/6a/1a927b14ddc7714111ea51f4e568203b2bb6ed59bdd036d62127c1a360c8/scipy-1.16.2-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:c2275ff105e508942f99d4e3bc56b6ef5e4b3c0af970386ca56b777608ce95b7", size = 35681881, upload-time = "2025-09-11T17:42:13.255Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c1/5f/331148ea5780b4fcc7007a4a6a6ee0a0c1507a796365cc642d4d226e1c3a/scipy-1.16.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:af80196eaa84f033e48444d2e0786ec47d328ba00c71e4299b602235ffef9acb", size = 36006219, upload-time = "2025-09-11T17:42:18.765Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/46/3a/e991aa9d2aec723b4a8dcfbfc8365edec5d5e5f9f133888067f1cbb7dfc1/scipy-1.16.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:9fb1eb735fe3d6ed1f89918224e3385fbf6f9e23757cacc35f9c78d3b712dd6e", size = 38682147, upload-time = "2025-09-11T17:42:25.177Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a1/57/0f38e396ad19e41b4c5db66130167eef8ee620a49bc7d0512e3bb67e0cab/scipy-1.16.2-cp313-cp313-win_amd64.whl", hash = "sha256:fda714cf45ba43c9d3bae8f2585c777f64e3f89a2e073b668b32ede412d8f52c", size = 38520766, upload-time = "2025-09-11T17:43:25.342Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1b/a5/85d3e867b6822d331e26c862a91375bb7746a0b458db5effa093d34cdb89/scipy-1.16.2-cp313-cp313-win_arm64.whl", hash = "sha256:2f5350da923ccfd0b00e07c3e5cfb316c1c0d6c1d864c07a72d092e9f20db104", size = 25451169, upload-time = "2025-09-11T17:43:30.198Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/09/d9/60679189bcebda55992d1a45498de6d080dcaf21ce0c8f24f888117e0c2d/scipy-1.16.2-cp313-cp313t-macosx_10_14_x86_64.whl", hash = "sha256:53d8d2ee29b925344c13bda64ab51785f016b1b9617849dac10897f0701b20c1", size = 37012682, upload-time = "2025-09-11T17:42:30.677Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/83/be/a99d13ee4d3b7887a96f8c71361b9659ba4ef34da0338f14891e102a127f/scipy-1.16.2-cp313-cp313t-macosx_12_0_arm64.whl", hash = "sha256:9e05e33657efb4c6a9d23bd8300101536abd99c85cca82da0bffff8d8764d08a", size = 29389926, upload-time = "2025-09-11T17:42:35.845Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bf/0a/130164a4881cec6ca8c00faf3b57926f28ed429cd6001a673f83c7c2a579/scipy-1.16.2-cp313-cp313t-macosx_14_0_arm64.whl", hash = "sha256:7fe65b36036357003b3ef9d37547abeefaa353b237e989c21027b8ed62b12d4f", size = 21381152, upload-time = "2025-09-11T17:42:40.07Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/47/a6/503ffb0310ae77fba874e10cddfc4a1280bdcca1d13c3751b8c3c2996cf8/scipy-1.16.2-cp313-cp313t-macosx_14_0_x86_64.whl", hash = "sha256:6406d2ac6d40b861cccf57f49592f9779071655e9f75cd4f977fa0bdd09cb2e4", size = 23914410, upload-time = "2025-09-11T17:42:44.313Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fa/c7/1147774bcea50d00c02600aadaa919facbd8537997a62496270133536ed6/scipy-1.16.2-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:ff4dc42bd321991fbf611c23fc35912d690f731c9914bf3af8f417e64aca0f21", size = 33481880, upload-time = "2025-09-11T17:42:49.325Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/6a/74/99d5415e4c3e46b2586f30cdbecb95e101c7192628a484a40dd0d163811a/scipy-1.16.2-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:654324826654d4d9133e10675325708fb954bc84dae6e9ad0a52e75c6b1a01d7", size = 35791425, upload-time = "2025-09-11T17:42:54.711Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1b/ee/a6559de7c1cc710e938c0355d9d4fbcd732dac4d0d131959d1f3b63eb29c/scipy-1.16.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:63870a84cd15c44e65220eaed2dac0e8f8b26bbb991456a033c1d9abfe8a94f8", size = 36178622, upload-time = "2025-09-11T17:43:00.375Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4e/7b/f127a5795d5ba8ece4e0dce7d4a9fb7cb9e4f4757137757d7a69ab7d4f1a/scipy-1.16.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:fa01f0f6a3050fa6a9771a95d5faccc8e2f5a92b4a2e5440a0fa7264a2398472", size = 38783985, upload-time = "2025-09-11T17:43:06.661Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3e/9f/bc81c1d1e033951eb5912cd3750cc005943afa3e65a725d2443a3b3c4347/scipy-1.16.2-cp313-cp313t-win_amd64.whl", hash = "sha256:116296e89fba96f76353a8579820c2512f6e55835d3fad7780fece04367de351", size = 38631367, upload-time = "2025-09-11T17:43:14.44Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d6/5e/2cc7555fd81d01814271412a1d59a289d25f8b63208a0a16c21069d55d3e/scipy-1.16.2-cp313-cp313t-win_arm64.whl", hash = "sha256:98e22834650be81d42982360382b43b17f7ba95e0e6993e2a4f5b9ad9283a94d", size = 25787992, upload-time = "2025-09-11T17:43:19.745Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8b/ac/ad8951250516db71619f0bd3b2eb2448db04b720a003dd98619b78b692c0/scipy-1.16.2-cp314-cp314-macosx_10_14_x86_64.whl", hash = "sha256:567e77755019bb7461513c87f02bb73fb65b11f049aaaa8ca17cfaa5a5c45d77", size = 36595109, upload-time = "2025-09-11T17:43:35.713Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ff/f6/5779049ed119c5b503b0f3dc6d6f3f68eefc3a9190d4ad4c276f854f051b/scipy-1.16.2-cp314-cp314-macosx_12_0_arm64.whl", hash = "sha256:17d9bb346194e8967296621208fcdfd39b55498ef7d2f376884d5ac47cec1a70", size = 28859110, upload-time = "2025-09-11T17:43:40.814Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/82/09/9986e410ae38bf0a0c737ff8189ac81a93b8e42349aac009891c054403d7/scipy-1.16.2-cp314-cp314-macosx_14_0_arm64.whl", hash = "sha256:0a17541827a9b78b777d33b623a6dcfe2ef4a25806204d08ead0768f4e529a88", size = 20850110, upload-time = "2025-09-11T17:43:44.981Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/0d/ad/485cdef2d9215e2a7df6d61b81d2ac073dfacf6ae24b9ae87274c4e936ae/scipy-1.16.2-cp314-cp314-macosx_14_0_x86_64.whl", hash = "sha256:d7d4c6ba016ffc0f9568d012f5f1eb77ddd99412aea121e6fa8b4c3b7cbad91f", size = 23497014, upload-time = "2025-09-11T17:43:49.074Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a7/74/f6a852e5d581122b8f0f831f1d1e32fb8987776ed3658e95c377d308ed86/scipy-1.16.2-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:9702c4c023227785c779cba2e1d6f7635dbb5b2e0936cdd3a4ecb98d78fd41eb", size = 33401155, upload-time = "2025-09-11T17:43:54.661Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d9/f5/61d243bbc7c6e5e4e13dde9887e84a5cbe9e0f75fd09843044af1590844e/scipy-1.16.2-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:d1cdf0ac28948d225decdefcc45ad7dd91716c29ab56ef32f8e0d50657dffcc7", size = 35691174, upload-time = "2025-09-11T17:44:00.101Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/03/99/59933956331f8cc57e406cdb7a483906c74706b156998f322913e789c7e1/scipy-1.16.2-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:70327d6aa572a17c2941cdfb20673f82e536e91850a2e4cb0c5b858b690e1548", size = 36070752, upload-time = "2025-09-11T17:44:05.619Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c6/7d/00f825cfb47ee19ef74ecf01244b43e95eae74e7e0ff796026ea7cd98456/scipy-1.16.2-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:5221c0b2a4b58aa7c4ed0387d360fd90ee9086d383bb34d9f2789fafddc8a936", size = 38701010, upload-time = "2025-09-11T17:44:11.322Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e4/9f/b62587029980378304ba5a8563d376c96f40b1e133daacee76efdcae32de/scipy-1.16.2-cp314-cp314-win_amd64.whl", hash = "sha256:f5a85d7b2b708025af08f060a496dd261055b617d776fc05a1a1cc69e09fe9ff", size = 39360061, upload-time = "2025-09-11T17:45:09.814Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/82/04/7a2f1609921352c7fbee0815811b5050582f67f19983096c4769867ca45f/scipy-1.16.2-cp314-cp314-win_arm64.whl", hash = "sha256:2cc73a33305b4b24556957d5857d6253ce1e2dcd67fa0ff46d87d1670b3e1e1d", size = 26126914, upload-time = "2025-09-11T17:45:14.73Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/51/b9/60929ce350c16b221928725d2d1d7f86cf96b8bc07415547057d1196dc92/scipy-1.16.2-cp314-cp314t-macosx_10_14_x86_64.whl", hash = "sha256:9ea2a3fed83065d77367775d689401a703d0f697420719ee10c0780bcab594d8", size = 37013193, upload-time = "2025-09-11T17:44:16.757Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/2a/41/ed80e67782d4bc5fc85a966bc356c601afddd175856ba7c7bb6d9490607e/scipy-1.16.2-cp314-cp314t-macosx_12_0_arm64.whl", hash = "sha256:7280d926f11ca945c3ef92ba960fa924e1465f8d07ce3a9923080363390624c4", size = 29390172, upload-time = "2025-09-11T17:44:21.783Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c4/a3/2f673ace4090452696ccded5f5f8efffb353b8f3628f823a110e0170b605/scipy-1.16.2-cp314-cp314t-macosx_14_0_arm64.whl", hash = "sha256:8afae1756f6a1fe04636407ef7dbece33d826a5d462b74f3d0eb82deabefd831", size = 21381326, upload-time = "2025-09-11T17:44:25.982Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/42/bf/59df61c5d51395066c35836b78136accf506197617c8662e60ea209881e1/scipy-1.16.2-cp314-cp314t-macosx_14_0_x86_64.whl", hash = "sha256:5c66511f29aa8d233388e7416a3f20d5cae7a2744d5cee2ecd38c081f4e861b3", size = 23915036, upload-time = "2025-09-11T17:44:30.527Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/91/c3/edc7b300dc16847ad3672f1a6f3f7c5d13522b21b84b81c265f4f2760d4a/scipy-1.16.2-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:efe6305aeaa0e96b0ccca5ff647a43737d9a092064a3894e46c414db84bc54ac", size = 33484341, upload-time = "2025-09-11T17:44:35.981Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/26/c7/24d1524e72f06ff141e8d04b833c20db3021020563272ccb1b83860082a9/scipy-1.16.2-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:7f3a337d9ae06a1e8d655ee9d8ecb835ea5ddcdcbd8d23012afa055ab014f374", size = 35790840, upload-time = "2025-09-11T17:44:41.76Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/aa/b7/5aaad984eeedd56858dc33d75efa59e8ce798d918e1033ef62d2708f2c3d/scipy-1.16.2-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:bab3605795d269067d8ce78a910220262711b753de8913d3deeaedb5dded3bb6", size = 36174716, upload-time = "2025-09-11T17:44:47.316Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/fd/c2/e276a237acb09824822b0ada11b028ed4067fdc367a946730979feacb870/scipy-1.16.2-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:b0348d8ddb55be2a844c518cd8cc8deeeb8aeba707cf834db5758fc89b476a2c", size = 38790088, upload-time = "2025-09-11T17:44:53.011Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c6/b4/5c18a766e8353015439f3780f5fc473f36f9762edc1a2e45da3ff5a31b21/scipy-1.16.2-cp314-cp314t-win_amd64.whl", hash = "sha256:26284797e38b8a75e14ea6631d29bda11e76ceaa6ddb6fdebbfe4c4d90faf2f9", size = 39457455, upload-time = "2025-09-11T17:44:58.899Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/97/30/2f9a5243008f76dfc5dee9a53dfb939d9b31e16ce4bd4f2e628bfc5d89d2/scipy-1.16.2-cp314-cp314t-win_arm64.whl", hash = "sha256:d2a4472c231328d4de38d5f1f68fdd6d28a615138f842580a8a321b5845cf779", size = 26448374, upload-time = "2025-09-11T17:45:03.45Z" },
|
||||
]
|
||||
Reference in New Issue
Block a user