首页 > 解决方案 > 通过Node调用golang函数时无法打开文件

问题描述

我按照https://medium.com/learning-the-go-programming-language/calling-go-functions-from-other-languages-4c7d8bcc69bf中的教程使我的节点应用程序能够调用 golang 函数。提供的示例就像一个魅力。
但是,我确实无法在另一种情况下实施。在这里我想通过只提供文件的绝对路径来打开文件并调用Go的函数,但它总是告诉我由于没有这样的文件而找不到文件。我正在尝试直接在 Go 中运行它并且它可以工作!
我做错了还是它是一个实际的错误/未完成的功能?

这是我为 c-style lib 构建的 golang 源代码:

package main

import "C"

import (
    "bufio"
    "fmt"
    "log"
    "os"
)

func main() {}

//export ReadSomething
func ReadSomething(filePath string) {
    file, err := os.Open(filePath)
    if err != nil {
        log.Fatal(err)
    }

    defer file.Close()

    scanner := bufio.NewScanner(file)
    for scanner.Scan() {
        fmt.Println(scanner.Text())
    }

    if err := scanner.Err(); err != nil {
        log.Fatal(err)
    }

}

我用这个命令构建了它:

go build -buildmode=c-shared -o simpleread.so main.go

如果您想知道标题输出是什么:

/* Created by "go tool cgo" - DO NOT EDIT. */

/* package command-line-arguments */

#line 1 "cgo-builtin-prolog"

#include <stddef.h> /* for ptrdiff_t below */

#ifndef GO_CGO_EXPORT_PROLOGUE_H
#define GO_CGO_EXPORT_PROLOGUE_H

typedef struct { const char *p; ptrdiff_t n; } _GoString_;

#endif

/* Start of preamble from import "C" comments.  */
/* End of preamble from import "C" comments.  */


/* Start of boilerplate cgo prologue.  */
#line 1 "cgo-gcc-export-header-prolog"

#ifndef GO_CGO_PROLOGUE_H
#define GO_CGO_PROLOGUE_H

typedef signed char GoInt8;
typedef unsigned char GoUint8;
typedef short GoInt16;
typedef unsigned short GoUint16;
typedef int GoInt32;
typedef unsigned int GoUint32;
typedef long long GoInt64;
typedef unsigned long long GoUint64;
typedef GoInt64 GoInt;
typedef GoUint64 GoUint;
typedef __SIZE_TYPE__ GoUintptr;
typedef float GoFloat32;
typedef double GoFloat64;
typedef float _Complex GoComplex64;
typedef double _Complex GoComplex128;

/*
  static assertion to make sure the file is being used on architecture
  at least with matching size of GoInt.
*/
typedef char _check_for_64_bit_pointer_matching_GoInt[sizeof(void*)==64/8 ? 1:-1];

typedef _GoString_ GoString;
typedef void *GoMap;
typedef void *GoChan;
typedef struct { void *t; void *v; } GoInterface;
typedef struct { void *data; GoInt len; GoInt cap; } GoSlice;

#endif

/* End of boilerplate cgo prologue.  */

#ifdef __cplusplus
extern "C" {
#endif


extern void ReadSomething(GoString p0);

#ifdef __cplusplus
}
#endif

然后,下面是我如何通过 Node.js 调用它。我在产生错误的行上给出评论:

var ref = require("ref")
var ffi = require("ffi-napi")
var Struct = require("ref-struct")
var ArrayType = require("ref-array")
var LongArray = ArrayType(ref.types.longlong);

  var GoString = Struct({
    p: "string",
    n: "longlong"
  });

  var simpleRead = ffi.Library("./simpleread.so", {
    ReadSomething: ["void", [GoString]]
  });

  // error here, can't open the specified file
  simpleRead.ReadSomething("/home/ivan/Documents/crashsite/node-go-crossfire/simpletext.txt")

我在 Ubuntu 18.04 64 位上运行它。

标签: node.jsgo

解决方案


请记住,Go 中的字符串就像切片。它们由指向支持数据的指针和长度组成。这就是为什么在您的代码中,GoString定义为:

var GoString = Struct({
    p: "string",  // pointer
    n: "longlong" // length
});

我建议你定义一个函数来创建一个GoString例如

function NewGoString(str) {
    return new GoString({p: str, n: str.length})
}

您可以在代码中使用它,例如:

var simpleRead = ffi.Library("./simpleread.so", {
    ReadSomething: ["void", [GoString]]
});

simpleRead.ReadSomething(NewGoString("/path/to/your/file"))

推荐阅读