// Gets all required enum fields that must be part of the pyi file because they're // default values in some methods. Dictionary <EnumType, HashSet <EnumValue> > GetRequiredEnumFields(List <PyClass> classes) { var reqEnumFields = new Dictionary <EnumType, HashSet <EnumValue> >(); var argToEnumType = new Dictionary <string, EnumType>(StringComparer.Ordinal); foreach (var pyClass in classes) { foreach (var method in pyClass.Methods) { DocComments docComments; if (method.Attributes.Any(AttributeKind.New)) { docComments = pyClass.DocComments; } else { docComments = method.DocComments; } var docs = docComments.Sections.OfType <ArgsDocCommentSection>().FirstOrDefault(); if (docs is null) { continue; } int hasThis = method.Arguments.Count != 0 && method.Arguments[0].IsSelf ? 1 : 0; if (docs.Args.Length != (method.Arguments.Count - hasThis)) { throw new InvalidOperationException(); } argToEnumType.Clear(); for (int i = 0; i < docs.Args.Length; i++) { var docArg = docs.Args[i]; if (docArg.Name != method.Arguments[hasThis + i].Name) { throw new InvalidOperationException(); } if (!ParseUtils.TryConvertSphinxTypeToTypeName(docArg.SphinxType, out var typeName)) { continue; } if (!exportedPythonTypes.TryFindByName(typeName, out var enumType)) { continue; } argToEnumType.Add(docArg.Name, enumType); } var argsAttr = method.Attributes.Attributes.FirstOrDefault(a => a.Kind == AttributeKind.Args); if (argsAttr is null) { continue; } foreach (var(name, value) in ParseUtils.GetArgsNameValues(argsAttr.Text)) { if (!argToEnumType.TryGetValue(name, out var enumType)) { continue; } if (!uint.TryParse(value, out var rawValue)) { throw new InvalidOperationException($"Couldn't parse {value} as an integer"); } var enumValue = enumType.Values.FirstOrDefault(a => a.Value == rawValue); if (enumValue is null) { throw new InvalidOperationException($"Couldn't find an enum value in {enumType.RawName} with a value equal to {value}"); } if (!reqEnumFields.TryGetValue(enumType, out var hash)) { reqEnumFields.Add(enumType, hash = new HashSet <EnumValue>()); } hash.Add(enumValue); } } } return(reqEnumFields); }
static void Write(FileWriter writer, PyiDocGen docGen, IdentifierConverter idConverter, PyClass pyClass, PyMethod method, DocComments docComments, Dictionary <string, EnumType> toEnumType) { if (method.Attributes.Any(AttributeKind.ClassMethod)) { writer.WriteLine("@classmethod"); } if (method.Attributes.Any(AttributeKind.StaticMethod)) { writer.WriteLine("@staticmethod"); } bool isGetter = method.Attributes.Any(AttributeKind.Getter); bool isSetter = method.Attributes.Any(AttributeKind.Setter); if (isGetter) { writer.WriteLine("@property"); } if (isSetter) { writer.WriteLine($"@{method.Name}.setter"); } string sphinxReturnType = string.Empty; if (isGetter || isSetter) { if (docComments.Sections.FirstOrDefault() is not TextDocCommentSection textDocs || textDocs.Lines.Length == 0) { throw new InvalidOperationException(); } if (!ParseUtils.TryParseTypeAndDocs(textDocs.Lines[0], out _, out var typeInfo)) { throw new InvalidOperationException(); } sphinxReturnType = typeInfo.SphinxType; } else { var returns = docComments.Sections.OfType <ReturnsDocCommentSection>().FirstOrDefault(); if (returns is not null) { sphinxReturnType = returns.Returns.SphinxType; } } bool isCtor = method.Attributes.Any(AttributeKind.New); writer.Write("def "); writer.Write(isCtor ? "__init__" : method.Name); writer.Write("("); int argCount = 0; if (isCtor) { writer.Write("self"); argCount++; } var argsDocs = docComments.Sections.OfType <ArgsDocCommentSection>().FirstOrDefault(); int hasThis = method.Arguments.Count != 0 && method.Arguments[0].IsSelf ? 1 : 0; Dictionary <string, string> toDefaultValue; var argsAttr = method.Attributes.Attributes.FirstOrDefault(a => a.Kind == AttributeKind.Args); if (argsAttr is null) { toDefaultValue = new Dictionary <string, string>(StringComparer.Ordinal); } else { toDefaultValue = ParseUtils.GetArgsNameValues(argsAttr.Text).ToDictionary(a => a.name, a => a.value, StringComparer.Ordinal); } for (int i = 0; i < method.Arguments.Count; i++) { if (argsDocs is not null && argsDocs.Args.Length != method.Arguments.Count - hasThis) { throw new InvalidOperationException(); } var methodArg = method.Arguments[i]; if (argCount > 0) { writer.Write(", "); } argCount++; if (methodArg.IsSelf) { writer.Write("self"); } else { writer.Write(methodArg.Name); string docsSphinxType; if (argsDocs is not null) { var docsArg = argsDocs.Args[i - hasThis]; if (docsArg.Name != methodArg.Name) { throw new InvalidOperationException(); } docsSphinxType = docsArg.SphinxType; } else { docsSphinxType = string.Empty; } if (i == 1 && isSetter) { docsSphinxType = sphinxReturnType; } writer.Write(": "); var type = GetType(pyClass, method.Name, methodArg.RustType, docsSphinxType); writer.Write(type); if (toDefaultValue.TryGetValue(methodArg.Name, out var defaultValueStr)) { writer.Write(" = "); if (!TryGetValueStr(idConverter, type, defaultValueStr, toEnumType, out var valueStr)) { throw new InvalidOperationException($"method {pyClass.Name}.{method.Name}(): Couldn't convert default value `{defaultValueStr}` to a Python value"); } writer.Write(valueStr); } } } writer.Write(") -> "); if (method.HasReturnType && !isCtor) { writer.Write(GetReturnType(pyClass, method.Name, method.RustReturnType, sphinxReturnType)); } else { writer.Write("None"); } if (method.DocComments.Sections.Count == 0) { writer.WriteLine(": ..."); } else { writer.WriteLine(":"); using (writer.Indent()) { WriteDocs(writer, docGen.Convert(method.DocComments)); writer.WriteLine("..."); } } }