Eu odeio escrever as coisas duas vezes, então eu inventei uma maneira decente de não ter que escrever as coisas duas vezes. No entanto, isso parece quebrar meu type-hinting:
from enum import Enum
from dataclasses import make_dataclass, field, dataclass
class DatasetNames(Enum):
test1 = "test1_string"
test2 = "test2_string"
test3 = "test3_string"
def get_path(s: str) -> str:
return s + "_path"
# the normal way to do this, but I have to type every new dataset name twice
# and there's a lot of duplicate code
@dataclass(frozen=True)
class StaticDatasetPaths:
test1 = get_path("test1_string")
test2 = get_path("test2_string")
test3 = get_path("test3_string")
# mypy recognizes that `StaticDatasetPaths` is a class
# mypy recognizes that `StaticDatasetPaths.test2` is a string
print(StaticDatasetPaths.test2) # 'test2_string_path'
# this is my way of doing it, without having to type every new dataset name twice and no duplicate code
DynamicDatasetPaths = make_dataclass(
'DynamicDatasetPaths',
[
(
name.name,
str,
field(default=get_path(name.value))
)
for name in DatasetNames
],
frozen=True
)
# mypy thinks `DynamicDatasetPaths` is a `variable` of type `type`
# mypy thinks that `DynamicDatasetPaths.test2` is an `function` of type `Unknown`
print(DynamicDatasetPaths.test2) # 'test2_string_path'
Como posso informar ao mypy que DynamicDatasetPaths é uma classe de dados congelada cujos atributos são strings?
Normalmente, quando me deparo com casos como esse, consigo simplesmente usar a cast
e dizer ao mypy qual é o tipo correto, mas não sei o tipo correto para "classe de dados congelada cujos atributos são strings".
(Além disso, se houver uma maneira melhor de não ter o código duplicado, ficaria feliz em saber sobre isso também.)
Uma classe de dados serve para criar instâncias. Como você não está instanciando a classe de dados, mas acessando
test1
,test2
, etc. como atributos de classe, você realmente não precisa de uma classe de dados, mas pode simplesmente criarpath
uma propriedade daEnum
classe. E como todos os membros da suaEnum
classe têm valores de string, você pode criar umaStrEnum
classe para operações de string mais fáceis:Se a
get_path
função for cara no seu caso de uso real, considere fazerpath
umacached_property
.