2024-08-06 13:03:23 +12:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2024-08-06 14:29:12 +12:00
|
|
|
"log"
|
2024-09-04 18:54:10 +12:00
|
|
|
"strconv"
|
2024-08-06 14:29:12 +12:00
|
|
|
"strings"
|
2024-08-06 13:03:23 +12:00
|
|
|
)
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
var (
|
|
|
|
ErrTooComplex = errors.New("Type declaration is too complex to parse")
|
|
|
|
ErrNoContent = errors.New("There's no content to include")
|
|
|
|
)
|
|
|
|
|
|
|
|
// parseHeader parses a whole C++ header into our CppParsedHeader intermediate format.
|
2024-08-27 19:12:08 +12:00
|
|
|
func parseHeader(topLevel []interface{}, addNamePrefix string) (*CppParsedHeader, error) {
|
2024-08-06 13:03:23 +12:00
|
|
|
|
2024-08-07 18:56:14 +12:00
|
|
|
var ret CppParsedHeader
|
2024-08-06 13:03:23 +12:00
|
|
|
|
2024-10-08 17:36:25 +13:00
|
|
|
nextTopLevel:
|
2024-08-15 19:51:13 +12:00
|
|
|
for _, node := range topLevel {
|
2024-08-06 13:03:23 +12:00
|
|
|
|
|
|
|
node, ok := node.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("inner[] element not an object")
|
|
|
|
}
|
|
|
|
|
|
|
|
kind, ok := node["kind"].(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("node has no kind")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch kind {
|
|
|
|
|
|
|
|
case "CXXRecordDecl":
|
2024-08-17 14:09:51 +12:00
|
|
|
|
2024-08-10 12:53:26 +12:00
|
|
|
// Process the inner class definition
|
2024-08-27 19:12:08 +12:00
|
|
|
obj, err := processClassType(node, addNamePrefix)
|
2024-08-10 12:53:26 +12:00
|
|
|
if err != nil {
|
2024-08-25 19:08:28 +12:00
|
|
|
if errors.Is(err, ErrNoContent) {
|
|
|
|
log.Printf("-> Skipping (%v)\n", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// A real error (shouldn't happen)
|
2024-08-10 12:53:26 +12:00
|
|
|
panic(err)
|
2024-08-06 13:03:23 +12:00
|
|
|
}
|
|
|
|
|
2024-08-10 12:53:26 +12:00
|
|
|
ret.Classes = append(ret.Classes, obj)
|
|
|
|
|
2024-08-06 13:03:23 +12:00
|
|
|
case "StaticAssertDecl":
|
|
|
|
// ignore
|
|
|
|
|
2024-08-19 19:14:10 +12:00
|
|
|
case "ClassTemplateDecl",
|
|
|
|
"ClassTemplateSpecializationDecl",
|
|
|
|
"ClassTemplatePartialSpecializationDecl",
|
|
|
|
"FunctionTemplateDecl",
|
2024-10-08 17:36:40 +13:00
|
|
|
"VarTemplatePartialSpecializationDecl", // e.g. Qt6 qcontainerinfo.h
|
|
|
|
"VarTemplateSpecializationDecl", // e.g. qhashfunctions.h
|
|
|
|
"TypeAliasTemplateDecl", // e.g. qendian.h
|
|
|
|
"VarTemplateDecl": // e.g. qglobal.h
|
2024-08-19 19:14:10 +12:00
|
|
|
// Template stuff probably can't be supported in the binding since
|
|
|
|
// we would need to link a concrete instantiation for each type in
|
|
|
|
// the CABI
|
|
|
|
// Ignore this node
|
2024-08-10 13:21:30 +12:00
|
|
|
|
|
|
|
case "FileScopeAsmDecl":
|
|
|
|
// ignore
|
|
|
|
|
|
|
|
case "NamespaceDecl":
|
2024-08-27 19:12:08 +12:00
|
|
|
// Parse everything inside the namespace with prefix, as if it is
|
|
|
|
// a whole separate file
|
|
|
|
// Then copy the parsed elements back into our own file
|
|
|
|
namespace, ok := node["name"].(string)
|
|
|
|
if !ok {
|
2024-10-08 17:36:25 +13:00
|
|
|
// Qt 5 has none of these
|
|
|
|
// Qt 6 has some e.g. qloggingcategory.h
|
|
|
|
// Treat it as not having existed
|
|
|
|
continue nextTopLevel
|
2024-08-27 19:12:08 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
namespaceInner, ok := node["inner"].([]interface{})
|
|
|
|
if !ok {
|
|
|
|
// A namespace declaration with no inner content means that, for
|
|
|
|
// the rest of this whole file, we are in this namespace
|
|
|
|
// Update our own `addNamePrefix` accordingly
|
|
|
|
addNamePrefix += namespace + "::"
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
contents, err := parseHeader(namespaceInner, namespace+"::")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.AddContentFrom(contents)
|
|
|
|
}
|
2024-08-10 13:21:30 +12:00
|
|
|
|
2024-08-10 14:04:52 +12:00
|
|
|
case "FunctionDecl":
|
|
|
|
// TODO
|
|
|
|
|
2024-08-19 19:14:10 +12:00
|
|
|
case "EnumDecl":
|
2024-09-04 18:54:10 +12:00
|
|
|
// Child class enum
|
|
|
|
en, err := processEnum(node, addNamePrefix)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Errorf("processEnum: %w", err)) // A real problem
|
|
|
|
}
|
2024-09-16 19:03:45 +12:00
|
|
|
|
|
|
|
// n.b. In some cases we may produce multiple "copies" of an enum
|
|
|
|
// (e.g. qcborcommon and qmetatype both define QCborSimpleType)
|
|
|
|
// Allow, but use a transform pass to avoid multiple definitions of
|
|
|
|
// it
|
|
|
|
ret.Enums = append(ret.Enums, en)
|
2024-08-19 19:14:10 +12:00
|
|
|
|
|
|
|
case "VarDecl":
|
|
|
|
// TODO e.g. qmath.h
|
|
|
|
// We could probably generate setter/getter for this in the CABI
|
2024-08-14 18:34:46 +12:00
|
|
|
|
|
|
|
case "CXXConstructorDecl":
|
|
|
|
// TODO (why is this at the top level? e.g qobject.h)
|
|
|
|
|
|
|
|
case "CXXDestructorDecl":
|
|
|
|
// ignore
|
|
|
|
|
|
|
|
case "CXXConversionDecl":
|
|
|
|
// TODO (e.g. qbytearray.h)
|
|
|
|
|
2024-08-19 19:14:10 +12:00
|
|
|
case "LinkageSpecDecl":
|
|
|
|
// TODO e.g. qfuturewatcher.h
|
|
|
|
// Probably can't be supported in the Go binding
|
|
|
|
|
2024-10-08 18:21:38 +13:00
|
|
|
case "UsingDirectiveDecl", // qtextstream.h
|
|
|
|
"UsingDecl", // qglobal.h
|
|
|
|
"UsingShadowDecl": // global.h
|
2024-08-27 19:12:08 +12:00
|
|
|
// TODO e.g.
|
2024-08-19 19:14:10 +12:00
|
|
|
// Should be treated like a typedef
|
|
|
|
|
2024-10-08 18:21:38 +13:00
|
|
|
case "TypeAliasDecl", "TypedefDecl":
|
2024-08-27 19:12:08 +12:00
|
|
|
td, err := processTypedef(node, addNamePrefix)
|
2024-08-26 22:45:52 +12:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("processTypedef: %w", err)
|
2024-08-14 18:26:42 +12:00
|
|
|
}
|
2024-08-26 22:45:52 +12:00
|
|
|
ret.Typedefs = append(ret.Typedefs, td)
|
2024-08-14 18:26:42 +12:00
|
|
|
|
2024-08-10 14:04:52 +12:00
|
|
|
case "CXXMethodDecl":
|
|
|
|
// A C++ class method implementation directly in the header
|
|
|
|
// Skip over these
|
|
|
|
|
2024-08-27 19:12:08 +12:00
|
|
|
case "FullComment":
|
|
|
|
// Safe to skip
|
|
|
|
|
2024-08-06 13:03:23 +12:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("missing handling for clang ast node type %q", kind)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &ret, nil // done
|
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// processTypedef parses a single C++ typedef into our intermediate format.
|
2024-08-26 22:45:52 +12:00
|
|
|
func processTypedef(node map[string]interface{}, addNamePrefix string) (CppTypedef, error) {
|
|
|
|
// Must have a name
|
|
|
|
nodename, ok := node["name"].(string)
|
|
|
|
if !ok {
|
|
|
|
return CppTypedef{}, errors.New("node has no name")
|
|
|
|
}
|
|
|
|
|
|
|
|
if typ, ok := node["type"].(map[string]interface{}); ok {
|
|
|
|
if qualType, ok := typ["qualType"].(string); ok {
|
|
|
|
return CppTypedef{
|
|
|
|
Alias: addNamePrefix + nodename,
|
|
|
|
UnderlyingType: parseSingleTypeString(qualType),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CppTypedef{}, errors.New("processTypedef: ???")
|
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// processClassType parses a single C++ class definition into our intermediate format.
|
2024-08-25 19:08:28 +12:00
|
|
|
func processClassType(node map[string]interface{}, addNamePrefix string) (CppClass, error) {
|
2024-08-07 18:56:14 +12:00
|
|
|
var ret CppClass
|
2024-08-20 20:10:57 +12:00
|
|
|
ret.CanDelete = true
|
2024-08-06 13:03:23 +12:00
|
|
|
|
2024-08-25 19:08:28 +12:00
|
|
|
// Must have a name
|
|
|
|
nodename, ok := node["name"].(string)
|
|
|
|
if !ok {
|
|
|
|
// This can happen for some nested class definitions e.g. qbytearraymatcher.h::Data
|
|
|
|
return CppClass{}, ErrNoContent // errors.New("node has no name")
|
|
|
|
}
|
|
|
|
nodename = addNamePrefix + nodename
|
2024-08-28 18:47:44 +12:00
|
|
|
|
|
|
|
// Hacks:
|
|
|
|
if nodename == "FromBase64Result" {
|
|
|
|
nodename = "QByteArray::FromBase64Result"
|
|
|
|
}
|
2024-08-29 17:17:35 +12:00
|
|
|
if nodename == "Connection" {
|
|
|
|
// qobject.h requires this, defined in qobjectdefs.h
|
|
|
|
// We produce a type named 'Connection' instead of 'QMetaObject::Connection' as expected, not sure why
|
|
|
|
nodename = "QMetaObject::Connection"
|
|
|
|
}
|
2024-08-28 18:47:44 +12:00
|
|
|
|
2024-08-25 19:08:28 +12:00
|
|
|
ret.ClassName = nodename
|
|
|
|
|
|
|
|
log.Printf("-> Processing class %q...\n", nodename)
|
|
|
|
|
|
|
|
// Also skip over any custom exceptions
|
|
|
|
if !AllowClass(nodename) {
|
|
|
|
return CppClass{}, ErrNoContent
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip over forward class declarations
|
|
|
|
// This is determined in two ways:
|
|
|
|
// 1. If the class has no inner nodes
|
|
|
|
inner, ok := node["inner"].([]interface{})
|
|
|
|
if !ok {
|
|
|
|
return CppClass{}, ErrNoContent
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. If this class has only one `inner` entry that's a VisibilityAttr
|
|
|
|
if len(inner) == 1 {
|
|
|
|
if node, ok := inner[0].(map[string]interface{}); ok {
|
|
|
|
if kind, ok := node["kind"].(string); ok && kind == "VisibilityAttr" {
|
|
|
|
return CppClass{}, ErrNoContent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-08-10 12:53:26 +12:00
|
|
|
|
|
|
|
// Check if this was 'struct' (default visible) or 'class' (default invisible)
|
|
|
|
visibility := true
|
|
|
|
if tagUsed, ok := node["tagUsed"].(string); ok && tagUsed == "class" {
|
|
|
|
visibility = false
|
|
|
|
}
|
2024-08-10 12:53:48 +12:00
|
|
|
|
|
|
|
// Check if this is an abstract class
|
|
|
|
if definitionData, ok := node["definitionData"].(map[string]interface{}); ok {
|
|
|
|
if isAbstract, ok := definitionData["isAbstract"].(bool); ok && isAbstract {
|
|
|
|
ret.Abstract = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:54:26 +12:00
|
|
|
// Check if this (publicly) inherits another class
|
|
|
|
if bases, ok := node["bases"].([]interface{}); ok {
|
|
|
|
for _, base := range bases {
|
|
|
|
base, ok := base.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
access, ok := base["access"].(string)
|
|
|
|
if !(ok && access == "public") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if typ, ok := base["type"].(map[string]interface{}); ok {
|
|
|
|
if qualType, ok := typ["qualType"].(string); ok {
|
|
|
|
ret.Inherits = append(ret.Inherits, qualType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-08-10 12:53:48 +12:00
|
|
|
|
2024-08-18 15:56:48 +12:00
|
|
|
isSignal := false
|
|
|
|
|
2024-08-10 12:53:48 +12:00
|
|
|
// Parse all methods
|
|
|
|
|
2024-08-06 14:29:12 +12:00
|
|
|
nextMethod:
|
2024-08-06 13:03:23 +12:00
|
|
|
for _, node := range inner {
|
|
|
|
node, ok := node.(map[string]interface{})
|
|
|
|
if !ok {
|
2024-08-07 18:56:14 +12:00
|
|
|
return CppClass{}, errors.New("inner[] element not an object")
|
2024-08-06 13:03:23 +12:00
|
|
|
}
|
|
|
|
|
2024-08-15 19:51:13 +12:00
|
|
|
kind, ok := node["kind"].(string)
|
2024-08-06 13:03:23 +12:00
|
|
|
if !ok {
|
|
|
|
panic("inner element has no kind")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch kind {
|
2024-08-07 18:50:39 +12:00
|
|
|
case "AccessSpecDecl":
|
|
|
|
// Swap between visible/invisible
|
|
|
|
access, ok := node["access"].(string)
|
|
|
|
if !ok {
|
|
|
|
panic("AccessSpecDecl missing `access` field")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch access {
|
|
|
|
case "public":
|
|
|
|
visibility = true
|
|
|
|
case "private", "protected":
|
|
|
|
visibility = false
|
|
|
|
default:
|
|
|
|
panic("unexpected access visibility '" + access + "'")
|
|
|
|
}
|
|
|
|
|
2024-08-18 15:56:48 +12:00
|
|
|
// Clang sees Q_SIGNALS/signals as being a macro for `public`
|
|
|
|
// If this AccessSpecDecl was imported from a macro, assume it's signals
|
|
|
|
isSignal = false
|
|
|
|
if loc, ok := node["loc"].(map[string]interface{}); ok {
|
|
|
|
if _, ok := loc["expansionLoc"].(map[string]interface{}); ok {
|
|
|
|
isSignal = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
case "FriendDecl":
|
|
|
|
// Safe to ignore
|
|
|
|
|
2024-08-10 12:53:26 +12:00
|
|
|
case "VisibilityAttr":
|
|
|
|
// These seem to have no useful content
|
|
|
|
|
2024-08-25 19:08:28 +12:00
|
|
|
case "CXXRecordDecl":
|
|
|
|
// Child class type definition e.g. QAbstractEventDispatcher::TimerInfo
|
|
|
|
// Parse as a whole child class
|
|
|
|
|
|
|
|
if !visibility {
|
|
|
|
continue // Skip private/protected
|
|
|
|
}
|
|
|
|
|
|
|
|
child, err := processClassType(node, nodename+"::")
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, ErrNoContent) {
|
2024-08-28 17:59:56 +12:00
|
|
|
log.Printf("-> Skipping inner class because: %v", err)
|
2024-08-25 19:08:28 +12:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
panic(err) // A real problem
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.ChildClassdefs = append(ret.ChildClassdefs, child)
|
|
|
|
|
2024-10-08 18:21:38 +13:00
|
|
|
case "TypeAliasDecl", "TypedefDecl":
|
2024-08-26 22:45:52 +12:00
|
|
|
// Child class typedef
|
|
|
|
td, err := processTypedef(node, nodename+"::")
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Errorf("processTypedef: %w", err)) // A real problem
|
|
|
|
}
|
|
|
|
ret.ChildTypedefs = append(ret.ChildTypedefs, td)
|
|
|
|
|
2024-09-04 18:54:10 +12:00
|
|
|
case "EnumDecl":
|
|
|
|
// Child class enum
|
|
|
|
|
|
|
|
if !visibility {
|
|
|
|
continue // Skip private/protected
|
|
|
|
}
|
|
|
|
|
|
|
|
en, err := processEnum(node, nodename+"::")
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Errorf("processEnum: %w", err)) // A real problem
|
|
|
|
}
|
|
|
|
if len(en.Entries) > 0 { // e.g. qmetatype's version of QCborSimpleType (the real one is in qcborcommon)
|
|
|
|
ret.ChildEnums = append(ret.ChildEnums, en)
|
|
|
|
}
|
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
case "CXXConstructorDecl":
|
2024-08-25 12:48:19 +12:00
|
|
|
|
|
|
|
if isImplicit, ok := node["isImplicit"].(bool); ok && isImplicit {
|
|
|
|
// This is an implicit ctor. Therefore the class is constructable
|
|
|
|
// even if we're currently in a `private:` block.
|
|
|
|
|
|
|
|
} else if !visibility {
|
2024-08-14 18:39:04 +12:00
|
|
|
continue // Skip private/protected
|
|
|
|
}
|
2024-08-10 12:52:45 +12:00
|
|
|
|
2024-08-10 12:53:48 +12:00
|
|
|
if ret.Abstract {
|
|
|
|
continue // The bindings can't construct an abstract class
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:54:00 +12:00
|
|
|
// Check if this is `= delete`
|
2024-08-20 20:10:57 +12:00
|
|
|
if isExplicitlyDeleted(node) {
|
2024-08-10 12:54:00 +12:00
|
|
|
continue
|
|
|
|
}
|
2024-08-10 12:52:45 +12:00
|
|
|
|
|
|
|
var mm CppMethod
|
|
|
|
err := parseMethod(node, &mm)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, ErrTooComplex) {
|
2024-08-20 20:19:53 +12:00
|
|
|
log.Printf("Skipping ctor with complex type")
|
2024-08-10 12:52:45 +12:00
|
|
|
continue nextMethod
|
|
|
|
}
|
|
|
|
|
|
|
|
// Real error
|
|
|
|
return CppClass{}, err
|
|
|
|
}
|
|
|
|
|
2024-08-24 11:46:26 +12:00
|
|
|
// Always set IsStatic for constructors, since they can be called without
|
|
|
|
// an existing class instance
|
|
|
|
mm.IsStatic = true
|
|
|
|
|
2024-08-10 12:52:45 +12:00
|
|
|
ret.Ctors = append(ret.Ctors, mm)
|
2024-08-08 17:51:10 +12:00
|
|
|
|
|
|
|
case "CXXDestructorDecl":
|
2024-08-10 12:52:45 +12:00
|
|
|
// We don't need to expose destructors in the binding beyond offering
|
|
|
|
// a regular delete function
|
2024-08-20 20:10:57 +12:00
|
|
|
// However if this destructor is private or deleted, we should
|
|
|
|
// not bind it
|
|
|
|
|
2024-08-25 12:48:19 +12:00
|
|
|
if isImplicit, ok := node["isImplicit"].(bool); ok && isImplicit {
|
|
|
|
// This is an implicit dtor. Therefore the class is deleteable
|
|
|
|
// even if we're currently in a `private:` block.
|
|
|
|
ret.CanDelete = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-08-20 20:10:57 +12:00
|
|
|
if !visibility {
|
|
|
|
ret.CanDelete = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if this is `= delete`
|
|
|
|
if isExplicitlyDeleted(node) {
|
|
|
|
ret.CanDelete = false
|
|
|
|
continue
|
|
|
|
}
|
2024-08-07 18:51:51 +12:00
|
|
|
|
2024-08-06 14:29:12 +12:00
|
|
|
case "CXXMethodDecl":
|
2024-08-07 18:50:39 +12:00
|
|
|
if !visibility {
|
|
|
|
continue // Skip private/protected
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:54:00 +12:00
|
|
|
// Check if this is `= delete`
|
2024-08-20 20:10:57 +12:00
|
|
|
if isExplicitlyDeleted(node) {
|
2024-08-10 12:54:00 +12:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-08-06 14:29:12 +12:00
|
|
|
// Method
|
|
|
|
methodName, ok := node["name"].(string)
|
|
|
|
if !ok {
|
2024-08-07 18:56:14 +12:00
|
|
|
return CppClass{}, errors.New("method has no name")
|
2024-08-06 14:29:12 +12:00
|
|
|
}
|
|
|
|
|
2024-08-07 18:56:14 +12:00
|
|
|
var mm CppMethod
|
|
|
|
mm.MethodName = methodName
|
2024-08-06 14:29:12 +12:00
|
|
|
|
2024-08-18 17:47:51 +12:00
|
|
|
if strings.Contains(methodName, `QGADGET`) {
|
|
|
|
log.Printf("Skipping method %q with weird QGADGET behaviour\n", mm.MethodName)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:52:45 +12:00
|
|
|
err := parseMethod(node, &mm)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, ErrTooComplex) {
|
|
|
|
log.Printf("Skipping method %q with complex type", mm.MethodName)
|
|
|
|
continue nextMethod
|
2024-08-06 14:29:12 +12:00
|
|
|
}
|
|
|
|
|
2024-08-10 12:52:45 +12:00
|
|
|
// Real error
|
|
|
|
return CppClass{}, err
|
2024-08-06 14:29:12 +12:00
|
|
|
}
|
|
|
|
|
2024-09-21 10:29:20 +12:00
|
|
|
mm.IsSignal = isSignal && !mm.IsStatic && AllowSignal(mm)
|
|
|
|
|
|
|
|
// Once all processing is complete, pass to exceptions for final decision
|
|
|
|
|
|
|
|
if err := AllowMethod(mm); err != nil {
|
|
|
|
if errors.Is(err, ErrTooComplex) {
|
|
|
|
log.Printf("Skipping method %q with complex type", mm.MethodName)
|
2024-08-18 19:02:43 +12:00
|
|
|
continue nextMethod
|
|
|
|
}
|
2024-08-18 15:56:48 +12:00
|
|
|
|
2024-09-21 10:29:20 +12:00
|
|
|
// Real error
|
|
|
|
return CppClass{}, err
|
2024-08-18 15:57:40 +12:00
|
|
|
}
|
|
|
|
|
2024-08-07 18:56:14 +12:00
|
|
|
ret.Methods = append(ret.Methods, mm)
|
2024-08-06 14:29:12 +12:00
|
|
|
|
2024-08-06 13:03:23 +12:00
|
|
|
default:
|
2024-08-18 15:55:23 +12:00
|
|
|
log.Printf("==> NOT IMPLEMENTED %q\n", kind)
|
2024-08-06 13:03:23 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil // done
|
|
|
|
}
|
2024-08-08 17:51:10 +12:00
|
|
|
|
2024-08-20 20:10:57 +12:00
|
|
|
// isExplicitlyDeleted checks if this node is marked `= delete`.
|
|
|
|
func isExplicitlyDeleted(node map[string]interface{}) bool {
|
|
|
|
|
|
|
|
if explicitlyDeleted, ok := node["explicitlyDeleted"].(bool); ok && explicitlyDeleted {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if explicitlyDefaulted, ok := node["explicitlyDefaulted"].(string); ok && explicitlyDefaulted == "deleted" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// processEnum parses a Clang enum into our CppEnum intermediate format.
|
2024-09-04 18:54:10 +12:00
|
|
|
func processEnum(node map[string]interface{}, addNamePrefix string) (CppEnum, error) {
|
|
|
|
var ret CppEnum
|
|
|
|
|
2024-09-16 19:03:02 +12:00
|
|
|
// Underlying type
|
2024-09-18 09:35:53 +12:00
|
|
|
ret.UnderlyingType = parseSingleTypeString("int")
|
2024-09-16 19:03:02 +12:00
|
|
|
if nodefut, ok := node["fixedUnderlyingType"].(map[string]interface{}); ok {
|
|
|
|
if nodequal, ok := nodefut["qualType"].(string); ok {
|
2024-09-18 09:35:53 +12:00
|
|
|
ret.UnderlyingType = parseSingleTypeString(nodequal)
|
2024-09-16 19:03:02 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Name
|
2024-09-04 18:54:10 +12:00
|
|
|
nodename, ok := node["name"].(string)
|
|
|
|
if !ok {
|
|
|
|
// An unnamed enum is possible (e.g. qcalendar.h)
|
|
|
|
// It defines integer constants just in the current scope
|
|
|
|
ret.EnumName = addNamePrefix
|
|
|
|
|
|
|
|
} else {
|
|
|
|
ret.EnumName = addNamePrefix + nodename
|
|
|
|
}
|
|
|
|
|
2024-09-16 19:03:02 +12:00
|
|
|
// Entries
|
2024-09-04 18:54:10 +12:00
|
|
|
inner, ok := node["inner"].([]interface{})
|
|
|
|
if !ok {
|
|
|
|
// An enum with no entries? We're done
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var lastImplicitValue int64 = -1
|
|
|
|
|
2024-10-08 17:36:02 +13:00
|
|
|
nextEnumEntry:
|
2024-09-04 18:54:10 +12:00
|
|
|
for _, entry := range inner {
|
|
|
|
entry, ok := entry.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return ret, errors.New("bad inner type")
|
|
|
|
}
|
|
|
|
|
|
|
|
kind, ok := entry["kind"].(string)
|
2024-10-08 17:36:02 +13:00
|
|
|
if kind == "DeprecatedAttr" {
|
|
|
|
continue nextEnumEntry // skip
|
|
|
|
} else if kind == "EnumConstantDecl" {
|
|
|
|
// allow
|
|
|
|
} else {
|
|
|
|
// unknown kind, or maybe !ok
|
2024-09-04 18:54:10 +12:00
|
|
|
return ret, fmt.Errorf("unexpected kind %q", kind)
|
|
|
|
}
|
|
|
|
|
|
|
|
var cee CppEnumEntry
|
|
|
|
|
|
|
|
entryname, ok := entry["name"].(string)
|
|
|
|
if !ok {
|
|
|
|
return ret, errors.New("entry without name")
|
|
|
|
}
|
|
|
|
|
|
|
|
cee.EntryName = entryname
|
|
|
|
|
|
|
|
// Try to find the enum value
|
|
|
|
ei1, ok := entry["inner"].([]interface{})
|
|
|
|
if !ok || len(ei1) == 0 {
|
|
|
|
// Enum case without definition e.g. QCalendar::Gregorian
|
|
|
|
// This means one more than the last value
|
|
|
|
cee.EntryValue = fmt.Sprintf("%d", lastImplicitValue+1)
|
|
|
|
|
2024-10-07 18:31:32 +13:00
|
|
|
} else if len(ei1) >= 1 {
|
|
|
|
|
|
|
|
// There may be more than one RHS `inner` expression if one of them
|
|
|
|
// is a comment
|
|
|
|
// Iterate through each of the ei1 entries and see if any of them
|
|
|
|
// work for the purposes of enum constant value parsing
|
|
|
|
for _, ei1_0 := range ei1 {
|
|
|
|
|
|
|
|
ei1_0 := ei1_0.(map[string]interface{})
|
|
|
|
|
|
|
|
// Best case: .inner -> kind=ConstantExpr value=xx
|
|
|
|
// e.g. qabstractitemmodel
|
|
|
|
if ei1Kind, ok := ei1_0["kind"].(string); ok && ei1Kind == "ConstantExpr" {
|
|
|
|
log.Printf("Got ConstantExpr OK")
|
|
|
|
if ei1Value, ok := ei1_0["value"].(string); ok {
|
|
|
|
cee.EntryValue = ei1Value
|
|
|
|
goto afterParse
|
|
|
|
}
|
2024-09-04 18:54:10 +12:00
|
|
|
}
|
|
|
|
|
2024-10-07 18:31:32 +13:00
|
|
|
// Best case: .inner -> kind=ImplicitCastExpr .inner -> kind=ConstantExpr value=xx
|
|
|
|
// e.g. QCalendar (when there is a int typecast)
|
|
|
|
if ei1Kind, ok := ei1_0["kind"].(string); ok && ei1Kind == "ImplicitCastExpr" {
|
|
|
|
log.Printf("Got ImplicitCastExpr OK")
|
|
|
|
if ei2, ok := ei1_0["inner"].([]interface{}); ok && len(ei2) > 0 {
|
|
|
|
ei2_0 := ei2[0].(map[string]interface{})
|
|
|
|
if ei2Kind, ok := ei2_0["kind"].(string); ok && ei2Kind == "ConstantExpr" {
|
|
|
|
log.Printf("Got ConstantExpr OK")
|
|
|
|
if ei2Value, ok := ei2_0["value"].(string); ok {
|
|
|
|
cee.EntryValue = ei2Value
|
|
|
|
goto afterParse
|
|
|
|
}
|
2024-09-04 18:54:10 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-10-07 18:31:32 +13:00
|
|
|
|
2024-10-08 17:36:02 +13:00
|
|
|
if ei1Kind, ok := ei1_0["kind"].(string); ok && ei1Kind == "DeprecatedAttr" {
|
|
|
|
log.Printf("Enum entry %q is deprecated, skipping", ret.EnumName+"::"+entryname)
|
|
|
|
continue nextEnumEntry
|
|
|
|
}
|
|
|
|
|
2024-09-04 18:54:10 +12:00
|
|
|
}
|
2024-10-07 18:31:32 +13:00
|
|
|
// If we made it here, we did not hit any of the `goto afterParse` cases
|
2024-09-04 18:54:10 +12:00
|
|
|
|
|
|
|
}
|
|
|
|
afterParse:
|
|
|
|
if cee.EntryValue == "" {
|
|
|
|
return ret, fmt.Errorf("Complex enum %q entry %q", ret.EnumName, entryname)
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
if cee.EntryValue == "true" || cee.EntryValue == "false" {
|
2024-09-18 09:35:53 +12:00
|
|
|
ret.UnderlyingType = parseSingleTypeString("bool")
|
2024-09-04 18:54:10 +12:00
|
|
|
|
|
|
|
} else {
|
|
|
|
lastImplicitValue, err = strconv.ParseInt(cee.EntryValue, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, fmt.Errorf("Enum %q entry %q has non-parseable value %q: %w", ret.EnumName, entryname, cee.EntryValue, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.Entries = append(ret.Entries, cee)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// parseMethod parses a Clang method into our CppMethod intermediate format.
|
2024-08-10 12:52:45 +12:00
|
|
|
func parseMethod(node map[string]interface{}, mm *CppMethod) error {
|
|
|
|
|
|
|
|
if typobj, ok := node["type"].(map[string]interface{}); ok {
|
|
|
|
if qualType, ok := typobj["qualType"].(string); ok {
|
|
|
|
// The qualType is the whole type of the method, including its parameter types
|
|
|
|
// If anything here is too complicated, skip the whole method
|
|
|
|
|
|
|
|
var err error = nil
|
2024-08-28 18:47:23 +12:00
|
|
|
mm.ReturnType, mm.Parameters, mm.IsConst, err = parseTypeString(qualType)
|
2024-08-10 12:52:45 +12:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-18 15:24:04 +12:00
|
|
|
if storageClass, ok := node["storageClass"].(string); ok && storageClass == "static" {
|
|
|
|
mm.IsStatic = true
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:52:45 +12:00
|
|
|
if methodInner, ok := node["inner"].([]interface{}); ok {
|
|
|
|
paramCounter := 0
|
|
|
|
for _, methodObj := range methodInner {
|
|
|
|
methodObj, ok := methodObj.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return errors.New("inner[] element not an object")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch methodObj["kind"] {
|
|
|
|
case "ParmVarDecl":
|
|
|
|
// Parameter variable
|
|
|
|
parmName, _ := methodObj["name"].(string) // n.b. may be unnamed
|
|
|
|
if parmName == "" {
|
|
|
|
|
|
|
|
// Generate a default parameter name
|
|
|
|
// Super nice autogen names if this is a Q_PROPERTY setter:
|
|
|
|
if len(mm.Parameters) == 1 && strings.HasPrefix(mm.MethodName, "set") {
|
|
|
|
parmName = strings.ToLower(string(mm.MethodName[3])) + mm.MethodName[4:]
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Otherwise - default
|
|
|
|
parmName = fmt.Sprintf("param%d", paramCounter+1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Block reserved Go words, replace with generic parameters
|
2024-08-15 19:49:38 +12:00
|
|
|
if goReservedWord(parmName) {
|
2024-08-10 12:52:45 +12:00
|
|
|
parmName += "Val"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the name for the existing nth parameter
|
|
|
|
mm.Parameters[paramCounter].ParameterName = parmName
|
|
|
|
|
|
|
|
// If this parameter has any internal AST nodes of its
|
|
|
|
// own, assume it means it's an optional parameter
|
|
|
|
if _, ok := methodObj["inner"]; ok {
|
|
|
|
mm.Parameters[paramCounter].Optional = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next
|
|
|
|
paramCounter++
|
|
|
|
|
|
|
|
default:
|
|
|
|
// Something else inside a declaration??
|
2024-08-18 15:55:23 +12:00
|
|
|
log.Printf("==> NOT IMPLEMENTED CXXMethodDecl->%q\n", methodObj["kind"])
|
2024-08-10 12:52:45 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-15 19:53:01 +12:00
|
|
|
// Fixups
|
|
|
|
// QDataStream.operator<< return a reference to QDataStream, but have a private
|
|
|
|
// copy constructor
|
|
|
|
// The bindings don't know it's the same object. This is just a trick for a more
|
|
|
|
// ergonomic C++ API but has no real effect
|
|
|
|
// Wipe out
|
|
|
|
if (mm.MethodName == "operator<<" || mm.MethodName == "operator>>" || mm.MethodName == "writeBytes") && mm.ReturnType.ParameterType == "QDataStream" && mm.ReturnType.ByRef {
|
|
|
|
mm.ReturnType = CppParameter{ParameterType: "void"}
|
|
|
|
}
|
|
|
|
|
2024-08-25 12:48:33 +12:00
|
|
|
// Change operator= (assign) to always return void. By default it returns *self which
|
|
|
|
// is a trick for more ergnonomic C++ that has no real effect
|
2024-08-25 15:32:16 +12:00
|
|
|
if mm.MethodName == "operator=" ||
|
|
|
|
mm.MethodName == "operator&=" || // qbitarray.h
|
|
|
|
mm.MethodName == "operator|=" || // qbitarray.h
|
|
|
|
mm.MethodName == "operator^=" { // qbitarray.h
|
2024-08-25 12:48:33 +12:00
|
|
|
mm.ReturnType = CppParameter{ParameterType: "void"}
|
|
|
|
}
|
|
|
|
|
2024-08-10 12:52:45 +12:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-08-26 22:49:33 +12:00
|
|
|
// parseTypeString converts a function/method type string such as
|
2024-08-28 18:47:23 +12:00
|
|
|
// - `QString (const char *, const char *, int) const`
|
2024-08-08 17:51:10 +12:00
|
|
|
// - `void (const QKeySequence \u0026)`
|
|
|
|
// into its (A) return type and (B) separate parameter types.
|
|
|
|
// These clang strings never contain the parameter's name, so the names here are
|
|
|
|
// not filled in.
|
2024-08-28 18:47:23 +12:00
|
|
|
func parseTypeString(typeString string) (CppParameter, []CppParameter, bool, error) {
|
2024-08-08 17:51:10 +12:00
|
|
|
|
2024-08-10 14:04:52 +12:00
|
|
|
if strings.Contains(typeString, `&&`) { // TODO Rvalue references
|
2024-08-28 18:47:23 +12:00
|
|
|
return CppParameter{}, nil, false, ErrTooComplex
|
2024-08-10 14:04:52 +12:00
|
|
|
}
|
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
// Cut to exterior-most (, ) pair
|
|
|
|
opos := strings.Index(typeString, `(`)
|
|
|
|
epos := strings.LastIndex(typeString, `)`)
|
|
|
|
|
|
|
|
if opos == -1 || epos == -1 {
|
2024-08-28 18:47:23 +12:00
|
|
|
return CppParameter{}, nil, false, fmt.Errorf("Type string %q missing brackets", typeString)
|
|
|
|
}
|
|
|
|
|
|
|
|
isConst := false
|
|
|
|
if strings.Contains(typeString[epos:], `const`) {
|
|
|
|
isConst = true
|
2024-08-08 17:51:10 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
returnType := parseSingleTypeString(strings.TrimSpace(typeString[0:opos]))
|
|
|
|
|
2024-08-15 19:53:01 +12:00
|
|
|
// Skip functions that return ints-by-reference since the ergonomics don't
|
|
|
|
// go through the binding
|
|
|
|
if returnType.IntType() && returnType.ByRef {
|
2024-08-28 18:47:23 +12:00
|
|
|
return CppParameter{}, nil, false, ErrTooComplex // e.g. QSize::rheight()
|
2024-08-15 19:53:01 +12:00
|
|
|
}
|
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
inner := typeString[opos+1 : epos]
|
|
|
|
|
|
|
|
// Should be no more brackets
|
|
|
|
if strings.ContainsAny(inner, `()`) {
|
2024-08-28 18:47:23 +12:00
|
|
|
return CppParameter{}, nil, false, ErrTooComplex
|
2024-08-08 17:51:10 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parameters are separated by commas and nesting can not be possible
|
2024-08-19 19:11:19 +12:00
|
|
|
params := tokenizeMultipleParameters(inner) // strings.Split(inner, `,`)
|
2024-08-08 17:51:10 +12:00
|
|
|
|
|
|
|
ret := make([]CppParameter, 0, len(params))
|
|
|
|
for _, p := range params {
|
|
|
|
|
|
|
|
insert := parseSingleTypeString(p)
|
|
|
|
|
|
|
|
if insert.ParameterType != "" {
|
|
|
|
ret = append(ret, insert)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-28 18:47:23 +12:00
|
|
|
return returnType, ret, isConst, nil
|
2024-08-08 17:51:10 +12:00
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// tokenizeMultipleParameters is like strings.Split by comma, except it does not
|
|
|
|
// split if a comma is used for an interior template type e.g. QMap<K,V>.
|
|
|
|
// It is expected to be used with a Clang type representing a function's parameter
|
|
|
|
// list.
|
2024-08-19 19:11:19 +12:00
|
|
|
func tokenizeMultipleParameters(p string) []string {
|
|
|
|
// Tokenize into top-level strings
|
|
|
|
templateDepth := 0
|
|
|
|
tokens := []string{}
|
|
|
|
wip := ""
|
|
|
|
p = strings.TrimSpace(p)
|
|
|
|
for _, c := range p {
|
|
|
|
if c == '<' {
|
|
|
|
wip += string(c)
|
|
|
|
templateDepth++
|
|
|
|
} else if c == '>' {
|
|
|
|
wip += string(c)
|
|
|
|
templateDepth--
|
|
|
|
} else if c == ',' && templateDepth == 0 {
|
|
|
|
tokens = append(tokens, wip)
|
|
|
|
wip = ""
|
|
|
|
} else {
|
|
|
|
wip += string(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tokens = append(tokens, wip)
|
|
|
|
return tokens
|
|
|
|
}
|
|
|
|
|
2024-09-20 19:13:00 +12:00
|
|
|
// tokenizeSingleParameter tokenizes a Clang qualType into separate tokens.
|
|
|
|
// Interior templates or brackets are tokenized together as a single token.
|
2024-08-28 17:59:21 +12:00
|
|
|
func tokenizeSingleParameter(p string) []string {
|
|
|
|
// Tokenize into top-level strings
|
|
|
|
templateDepth := 0
|
|
|
|
tokens := []string{}
|
|
|
|
wip := ""
|
|
|
|
p = strings.TrimSpace(p)
|
|
|
|
for _, c := range p {
|
|
|
|
if c == '<' || c == '(' {
|
|
|
|
wip += string(c)
|
|
|
|
templateDepth++
|
|
|
|
} else if c == '>' || c == ')' {
|
|
|
|
wip += string(c)
|
|
|
|
templateDepth--
|
|
|
|
} else if (c == '*' || c == '&') && templateDepth == 0 {
|
|
|
|
if len(wip) > 0 {
|
|
|
|
tokens = append(tokens, wip)
|
|
|
|
}
|
|
|
|
tokens = append(tokens, string(c))
|
|
|
|
wip = ""
|
|
|
|
} else if c == ' ' && templateDepth == 0 {
|
|
|
|
if len(wip) > 0 {
|
|
|
|
tokens = append(tokens, wip)
|
|
|
|
}
|
|
|
|
wip = ""
|
|
|
|
} else {
|
|
|
|
wip += string(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(wip) > 0 {
|
|
|
|
tokens = append(tokens, wip)
|
|
|
|
}
|
|
|
|
|
|
|
|
return tokens
|
|
|
|
}
|
2024-09-20 19:13:00 +12:00
|
|
|
|
|
|
|
// parseSingleTypeString parses the Clang qualType for a single type into our
|
|
|
|
// CppParameter intermediate format.
|
2024-08-08 17:51:10 +12:00
|
|
|
func parseSingleTypeString(p string) CppParameter {
|
|
|
|
|
2024-08-29 17:15:41 +12:00
|
|
|
isSigned := false
|
|
|
|
|
2024-08-29 18:07:39 +12:00
|
|
|
tokens := tokenizeSingleParameter(p)
|
2024-08-08 17:51:10 +12:00
|
|
|
insert := CppParameter{}
|
|
|
|
for _, tok := range tokens {
|
2024-08-19 19:42:34 +12:00
|
|
|
|
2024-08-15 19:53:01 +12:00
|
|
|
if tok == "" {
|
|
|
|
continue // extra space
|
2024-08-19 19:42:34 +12:00
|
|
|
|
2024-08-29 18:07:39 +12:00
|
|
|
} else if tok == "const" {
|
2024-08-08 17:51:10 +12:00
|
|
|
insert.Const = true
|
2024-08-15 19:53:01 +12:00
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
} else if tok == "&" { // U+0026
|
|
|
|
insert.ByRef = true
|
2024-08-19 19:42:34 +12:00
|
|
|
|
2024-08-26 22:51:21 +12:00
|
|
|
} else if tok == "signed" {
|
2024-08-29 17:15:41 +12:00
|
|
|
// We don't need this - UNLESS it's 'signed char'
|
|
|
|
isSigned = true
|
2024-08-26 22:51:21 +12:00
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
} else if tok == "*" {
|
|
|
|
insert.Pointer = true
|
2024-08-28 18:22:05 +12:00
|
|
|
insert.PointerCount++
|
2024-08-19 19:42:34 +12:00
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
} else {
|
|
|
|
// Valid part of the type name
|
2024-08-29 17:15:41 +12:00
|
|
|
if tok == "char" && isSigned {
|
|
|
|
tok = "signed char"
|
|
|
|
isSigned = false
|
|
|
|
}
|
2024-08-08 17:51:10 +12:00
|
|
|
insert.ParameterType += " " + tok
|
|
|
|
}
|
|
|
|
}
|
|
|
|
insert.ParameterType = strings.TrimSpace(insert.ParameterType)
|
|
|
|
|
2024-08-29 17:17:35 +12:00
|
|
|
if strings.HasPrefix(insert.ParameterType, `::`) {
|
|
|
|
insert.ParameterType = insert.ParameterType[2:]
|
|
|
|
}
|
|
|
|
|
2024-08-08 17:51:10 +12:00
|
|
|
return insert
|
|
|
|
}
|