gpt4 book ai didi

python - 如何在 Python 中的 Google Cloud Function 中运行子进程

转载 作者:行者123 更新时间:2023-12-03 21:05:13 24 4
gpt4 key购买 nike

我正在尝试在 GCP 函数中运行 bash 脚本,但不知何故它不起作用。
这是我的函数,它基本上将文件(代理)导出到 Google Apigee:

def test2(request):
cmd = "python ./my-proxy/tools/deploy.py -n myProxy -u userName:!password -o myOrg -e test -d ./my-proxy -p /"
# no block, it start a sub process.
p = subprocess.Popen(cmd , shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)

# and you can block util the cmd execute finish
p.wait()
# or stdout, stderr = p.communicate()
return "Proxy deployed to Apigee"

这是我的 deploy.py文件看起来像:
!/usr/bin/env python

import base64
import getopt
import httplib
import json
import re
import os
import sys
import StringIO
import urlparse
import xml.dom.minidom
import zipfile


def httpCall(verb, uri, headers, body):
if httpScheme == 'https':
conn = httplib.HTTPSConnection(httpHost)
else:
conn = httplib.HTTPConnection(httpHost)

if headers == None:
hdrs = dict()
else:
hdrs = headers

hdrs['Authorization'] = 'Basic %s' % base64.b64encode(UserPW)
conn.request(verb, uri, body, hdrs)

return conn.getresponse()


def getElementText(n):
c = n.firstChild
str = StringIO.StringIO()

while c != None:
if c.nodeType == xml.dom.Node.TEXT_NODE:
str.write(c.data)
c = c.nextSibling

return str.getvalue().strip()


def getElementVal(n, name):
c = n.firstChild

while c != None:
if c.nodeName == name:
return getElementText(c)
c = c.nextSibling

return None


# Return TRUE if any component of the file path contains a directory name that
# starts with a "." like '.svn', but not '.' or '..'
def pathContainsDot(p):
c = re.compile('\.\w+')

for pc in p.split('/'):
if c.match(pc) != None:
return True

return False


def getDeployments():
# Print info on deployments
hdrs = {'Accept': 'application/xml'}
resp = httpCall('GET',
'/v1/organizations/%s/apis/%s/deployments' \
% (Organization, Name),
hdrs, None)

if resp.status != 200:
return None

ret = list()
deployments = xml.dom.minidom.parse(resp)
environments = deployments.getElementsByTagName('Environment')

for env in environments:
envName = env.getAttribute('name')
revisions = env.getElementsByTagName('Revision')
for rev in revisions:
revNum = int(rev.getAttribute('name'))
error = None
state = getElementVal(rev, 'State')
basePaths = rev.getElementsByTagName('BasePath')

if len(basePaths) > 0:
basePath = getElementText(basePaths[0])
else:
basePath = 'unknown'

# svrs = rev.getElementsByTagName('Server')
status = {'environment': envName,
'revision': revNum,
'basePath': basePath,
'state': state}

if error != None:
status['error'] = error

ret.append(status)

return ret


def printDeployments(dep):
for d in dep:
print 'Environment: %s' % d['environment']
print ' Revision: %i BasePath = %s' % (d['revision'], d['basePath'])
print ' State: %s' % d['state']
if 'error' in d:
print ' Error: %s' % d['error']

ApigeeHost = 'https://api.enterprise.apigee.com'
UserPW = None
Directory = None
Organization = None
Environment = None
Name = None
BasePath = '/'
ShouldDeploy = True

Options = 'h:u:d:e:n:p:o:i:z:'

opts = getopt.getopt(sys.argv[1:], Options)[0]

for o in opts:
if o[0] == '-n':
Name = o[1]
elif o[0] == '-o':
Organization = o[1]
elif o[0] == '-h':
ApigeeHost = o[1]
elif o[0] == '-d':
Directory = o[1]
elif o[0] == '-e':
Environment = o[1]
elif o[0] == '-p':
BasePath = o[1]
elif o[0] == '-u':
UserPW = o[1]
elif o[0] == '-i':
ShouldDeploy = False
elif o[0] == '-z':
ZipFile = o[1]

if UserPW == None or \
(Directory == None and ZipFile == None) or \
Environment == None or \
Name == None or \
Organization == None:
print """Usage: deploy -n [name] (-d [directory name] | -z [zipfile])
-e [environment] -u [username:password] -o [organization]
[-p [base path] -h [apigee API url] -i]
base path defaults to "/"
Apigee URL defaults to "https://api.enterprise.apigee.com"
-i denotes to import only and not actually deploy
"""
sys.exit(1)

url = urlparse.urlparse(ApigeeHost)
httpScheme = url[0]
httpHost = url[1]

body = None

if Directory != None:
# Construct a ZIPped copy of the bundle in memory
tf = StringIO.StringIO()
zipout = zipfile.ZipFile(tf, 'w')

dirList = os.walk(Directory)
for dirEntry in dirList:
if not pathContainsDot(dirEntry[0]):
for fileEntry in dirEntry[2]:
if not fileEntry.endswith('~'):
fn = os.path.join(dirEntry[0], fileEntry)
en = os.path.join(
os.path.relpath(dirEntry[0], Directory),
fileEntry)
print 'Writing %s to %s' % (fn, en)
zipout.write(fn, en)

zipout.close()
body = tf.getvalue()
elif ZipFile != None:
f = open(ZipFile, 'r')
body = f.read()
f.close()

# Upload the bundle to the API
hdrs = {'Content-Type': 'application/octet-stream',
'Accept': 'application/json'}
uri = '/v1/organizations/%s/apis?action=import&name=%s' % \
(Organization, Name)
resp = httpCall('POST', uri, hdrs, body)

if resp.status != 200 and resp.status != 201:
print 'Import failed to %s with status %i:\n%s' % \
(uri, resp.status, resp.read())
sys.exit(2)

deployment = json.load(resp)
revision = int(deployment['revision'])

print 'Imported new proxy version %i' % revision

if ShouldDeploy:
# Undeploy duplicates
deps = getDeployments()
for d in deps:
if d['environment'] == Environment and \
d['basePath'] == BasePath and \
d['revision'] != revision:
print 'Undeploying revision %i in same environment and path:' % \
d['revision']
conn = httplib.HTTPSConnection(httpHost)
resp = httpCall('POST',
('/v1/organizations/%s/apis/%s/deployments' +
'?action=undeploy' +
'&env=%s' +
'&revision=%i') % \
(Organization, Name, Environment, d['revision']),
None, None)
if resp.status != 200 and resp.status != 204:
print 'Error %i on undeployment:\n%s' % \
(resp.status, resp.read())

# Deploy the bundle
hdrs = {'Accept': 'application/json'}
resp = httpCall('POST',
('/v1/organizations/%s/apis/%s/deployments' +
'?action=deploy' +
'&env=%s' +
'&revision=%i' +
'&basepath=%s') % \
(Organization, Name, Environment, revision, BasePath),
hdrs, None)

if resp.status != 200 and resp.status != 201:
print 'Deploy failed with status %i:\n%s' % (resp.status, resp.read())
sys.exit(2)

deps = getDeployments()
printDeployments(deps)

当我在我的机器上本地运行而不是在 GCP 上运行时,这有效。不知道是否与我使用此功能连接到 Google Apigee 的事实有关。奇怪的是 GCP 上的日志没有显示任何错误,但是我没有将我的代理导出到 Apigee。

谢谢帮助!

更新:
尝试使用 subprocess.check_output()正如这里的一些人所鼓励的那样:
def test(request):
output = None
try:
output = subprocess.check_output([
"./my-proxy/tools/deploy.py",
'-n', 'myProxy',
'-u', 'myUserName:myPassword',
'-o', 'myOrgName',
'-e', 'test',
'-d', './my-proxy',
'-p', '/'])

except:
print(output)

return output

仍然没有在 GCP 上工作。就像我之前提到的,它在我的机器中就像一个魅力(上面的两个解决方案),但在 GCP 中却没有。
如下图所示,执行 deploy.py 后得到 200来自 GCP,但我的文件没有转到 Apigee:

enter image description here

GCP 日志也没有显示任何错误:

enter image description here

最佳答案

这个有可能!python可执行文件未安装或链接到 Cloud Function 运行时中,而是 python3是。因此,有几种方法可以解决这个问题:

  • 指定 python3作为要运行的程序:"python3 ./my-proxy/tools/deploy.py ..." ;
  • 添加 #! deploy.py 中的运算符脚本:#!/usr/bin/env python3 ;
  • 将 python 解释器指定为 Popen .您可以使用 sys.executable引用当前使用的可执行文件:
     process = subprocess.Popen(
    [
    sys.executable,
    "./deploy.py",
    "-n",
    "myProxy",
    "-u",
    "myUserName:myPassword",
    "-o",
    "myOrgName",
    "-e",
    "test",
    "-d",
    "./my-proxy",
    "-p",
    "/",
    ],
    stdout=subprocess.PIPE,
    stderr=subprocess.PIPE,
    universal_newlines=True,
    )

  • 您没有看到错误,因为它是在子进程中生成的,打印到其 stderr,随后被您的程序用 process.communicate() 捕获。或 process.check_output(...) ,但不打印。要查看您遇到的错误,您可以打印出 stdout 和 stderr 的内容:
        out, err = process.communicate()
    log.debug("returncode = %s", process.returncode)
    log.debug("stdout = %s", out)
    log.debug("stderr = %s", err)
    查看我们用于分析、重现和解决您的问题的源代码 github

    关于python - 如何在 Python 中的 Google Cloud Function 中运行子进程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54952129/

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com