如何从NodeJs调用python脚本

时间:2015-06-07 02:34:35

标签: javascript python node.js python-2.7 raspberry-pi

我需要在NodeJs中调用这个python脚本。

Read.py

#!/usr/bin/env python
# -*- coding: utf8 -*-

import RPi.GPIO as GPIO
import MFRC522
import signal

continue_reading = True

# Capture SIGINT for cleanup when the script is aborted
def end_read(signal,frame):
    global continue_reading
    print "Ctrl+C captured, ending read."
    continue_reading = False
    GPIO.cleanup()

# Hook the SIGINT
signal.signal(signal.SIGINT, end_read)

# Create an object of the class MFRC522
MIFAREReader = MFRC522.MFRC522()

# Welcome message
print "Welcome to the MFRC522 data read example"
print "Press Ctrl-C to stop."

# This loop keeps checking for chips. If one is near it will get the UID and authenticate
while continue_reading:

    # Scan for cards    
    (status,TagType) = MIFAREReader.MFRC522_Request(MIFAREReader.PICC_REQIDL)

    # If a card is found
    if status == MIFAREReader.MI_OK:
    # Get the UID of the card
    (status,uid) = MIFAREReader.MFRC522_Anticoll()

    # If we have the UID, continue
    if status == MIFAREReader.MI_OK:

        # Print UID
        print "Card read UID: "+str(uid[0])+","+str(uid[1])+","+str(uid[2])+","+str(uid[3])

        # This is the default key for authentication
        key = [0xFF,0xFF,0xFF,0xFF,0xFF,0xFF]

        # Select the scanned tag
        MIFAREReader.MFRC522_SelectTag(uid)

        # Authenticate
        status = MIFAREReader.MFRC522_Auth(MIFAREReader.PICC_AUTHENT1A, 8, key, uid)

        # Check if authenticated
        if status == MIFAREReader.MI_OK:
            MIFAREReader.MFRC522_Read(8)
            MIFAREReader.MFRC522_StopCrypto1()
        else:
            print "Authentication error"

我使用了python-shell,这是

的NodeJs代码

Test.js

var PythonShell = require('python-shell');

var options = {
scriptPath: '/home/pi/gpio-admin/MFRC522-python/'
};
var pyshell = new PythonShell('Read.py',options);


pyshell.on('message', function (message) {

    console.log(message);
});

但是当我运行这段代码时,我在Node端没有看到任何内容。我认为当python脚本达到这个级别时会出现问题。

   (status,TagType) = MIFAREReader.MFRC522_Request(MIFAREReader.PICC_REQIDL)

因为我刚刚运行了只有print语句的while循环,所以它可以工作。之后我尝试了另一种方法来实现这一目标。但是我遇到了与上面相同的问题。这是另一种方法

AltTest.js

var python = require('child_process').spawn(
 'python',
 // second argument is array of parameters, e.g.:
 ["/home/pi/gpio-admin/MFRC522-python/Read.py"]
 );
 var output = "";
 python.stdout.on('data', function(){ 

    output += data ;
    console.log(data);
});
 python.on('close', function(code){ 

   console.log("Here you are there...");
 });

任何帮助将不胜感激

6 个答案:

答案 0 :(得分:16)

有多种方法可以做到这一点。

  • 第一种方法是npm install python-shell

这是代码

var PythonShell = require('python-shell');
//you can use error handling to see if there are any errors
PythonShell.run('my_script.py', options, function (err, results) { 
//your code

你可以使用发送消息给python shell pyshell.send('hello');

您可以在此处找到API参考资料 - https://github.com/extrabacon/python-shell

更多参考资料 - https://www.npmjs.com/package/python

如果你想使用面向服务的架构 - http://ianhinsdale.com/code/2013/12/08/communicating-between-nodejs-and-python/

答案 1 :(得分:2)

  • 安装python-shell:-npm安装python-shell

    Index.js

    let {PythonShell} = require('python-shell')
    
    function runPy(){
        return new Promise(async function(resolve, reject){
              let options = {
              mode: 'text',
              pythonOptions: ['-u'],
              scriptPath: './test.py',//Path to your script
              args: [JSON.stringify({"name": ["xyz", "abc"], "age": ["28","26"]})]//Approach to send JSON as when I tried 'json' in mode I was getting error.
             };
    
              await PythonShell.run('test.py', options, function (err, results) {
              //On 'results' we get list of strings of all print done in your py scripts sequentially. 
              if (err) throw err;
              console.log('results: ');
              for(let i of results){
                    console.log(i, "---->", typeof i)
              }
          resolve(results[1])//I returned only JSON(Stringified) out of all string I got from py script
         });
       })
     } 
    
    function runMain(){
        return new Promise(async function(resolve, reject){
            let r =  await runPy()
            console.log(JSON.parse(JSON.stringify(r.toString())), "Done...!@")//Approach to parse string to JSON.
        })
     }
    
    runMain() //run main function
    

test.py

    import sys #You will get input from node in sys.argv(list)
    import json
    import pandas as pd #Import just to check if you dont have pandas module you can comment it or install pandas using pip install pandas

    def add_two(a, b):
        sum = 0
        for i in range(a, b):
            sum += i
        print(sum)  

    if __name__ == "__main__":
        print("Here...!")
        # print(sys.argv)
        j = json.loads(sys.argv[1]) #sys.argv[0] is filename
        print(j)
        add_two(20000, 5000000) #I make this function just to check 
    # So for all print done here you will get a list for all print in node, here-> console.log(i, "---->", typeof i)

答案 2 :(得分:0)

采用微服务方法。将Python脚本托管为HTTP REST API服务。使用来自node.js的API-您无需集成技术;它不可扩展。

答案 3 :(得分:0)

您可以按以下方式从 node.js 中运行 python 脚本。 使用 child_process.spawn 这是在node.js库中构建的

router.get('/', (req, res) => {
    const {spawn} = require('child_process');
    const path = require('path');
    function runScript(){
        return spawn('python', [
              path.join(__dirname, '../../scripts/myscript.py'),
              '-some_arg',
              '--another_arg',
        ]);
    }
    const subprocess = runScript();
    // print output of script
    subprocess.stdout.on('data', (data) => {
            console.log(`data:${data}`);
    });
    subprocess.stderr.on('data', (data) => {
           console.log(`error:${data}`);
    });
    subprocess.stderr.on('close', () => {
               console.log("Closed");
    });
    // const subprocess = runScript()
    res.set('Content-Type', 'text/plain');
    subprocess.stdout.pipe(res);
    subprocess.stderr.pipe(res);
});

答案 4 :(得分:0)

如果您想避免软件包管理器/膨胀,请考虑编写一个shell脚本函数来运行python,将结果本地传输到txt文件,然后收集并发送。使用http:

    const { exec } = require("child_process");
    
    function runShellScript(script, callback) {
        exec(script, (error, stdOut, stderr) => {
            
            var result = {status: true};
            
            if (error) {
                result.status = false;
                result.error = error.message;
            }
            if (stderr) {
                result.status = false;
                result.stderr = stderr;
            }
    
            if(stdOut){
                result.result = stdOut;
            }
            
    
            callback(result);
        });
    }
    
    runShellScript("python3 myscript.py >> output.txt", function(res) {
        console.log(res);
fs.readFileSync('output.txt');
    });

答案 5 :(得分:0)

我的节点js代码

const { spawn }=require('child_process')

const child_python=spawn('python',['hello.py']);

child_python.stdout.on('data',(data)=> {
    console.log(`stdout :${data}`);
})

child_python.stderr.on('data',(data)=>{
    console.log(`stderr : ${data}`);
})

child_python.on('close',(code)=>{
    console.log(`child process exited with code ${code}`)
})

您可以查看此视频以从 Nodejs 运行您的 Python 脚本 它也告诉了如何传递参数

链接 https://www.youtube.com/watch?v=eN0nMuS8ur8