gpt4 book ai didi

javascript - 上传到 Google Cloud Storage 时获取 403 SignatureDoesNotMatch?

转载 作者:行者123 更新时间:2023-11-30 20:58:31 26 4
gpt4 key购买 nike

我正在尝试将客户端的音频文件直接上传到我的 Google Cloud Storage 存储桶,以避免服务器端上传(有文件大小限制)。

我的问题:我在上传时收到 403 SignatureDoesNotMatch 错误。

这是响应中的错误:

<Error>
<Code>SignatureDoesNotMatch</Code>
<Message> The request signature we calculated does not match the signature you provided. Check your Google secret key and signing method</Message>
<StringToSign>
PUT

audio/mp3
1511112552
/bucketname/pathtofile/019%20-%20top%20cntndr%20V1.mp3
</StringToSign>
</Error>

我已经创建了一个签名 url。它看起来像这样:

https://storage.googleapis.com/google-testbucket/testdata.txt?GoogleAccessId=
1234567890123@developer.gserviceaccount.com&Expires=1331155464&Signature=BCl
z9e4UA2MRRDX62TPd8sNpUCxVsqUDG3YGPWvPcwN%2BmWBPqwgUYcOSszCPlgWREeF7oPGowkeKk
7J4WApzkzxERdOQmAdrvshKSzUHg8Jqp1lw9tbiJfE2ExdOOIoJVmGLoDeAGnfzCd4fTsWcLbal9
sFpqXsQI8IQi1493mw%3D

签名的 url 是按照在此处 https://cloud.google.com/storage/docs/access-control/create-signed-urls-program 的 Google 文档中找到的指南构建的

但是,处理此签名 url 的客户端 javascript 部分在文档中非常不清楚。

这是我创建和返回签名 url 的 python 代码。

GOOGLE_SERVICE_CREDENTIALS = 'google-service-credentials.json'

def get_signed_url(request):
filename = request.GET.get('filename')
expiration = request.GET.get('expiration')
type = request.GET.get('type')
signed_url = CloudStorageSignedURL(
method='PUT',
file_name=filename,
expiration_m=expiration,
content_type=type
)
signed_url = signed_url.sign_url()


return JsonResponse({ 'signed_url': signed_url })






class CloudStorageSignedURL(object):


def __init__(self, method, file_name, expiration_m, content_type):
self.HTTP_method = method
self.content_type = 'content-type: ' + content_type
self.expiration = int(expiration_m)
self.file_name = file_name


def sign_url(self):


expiration_dt = datetime.utcnow() + timedelta(minutes=self.expiration)
expiration = int(time.mktime( expiration_dt.timetuple() ))
bucket_path = '/' + settings.CLOUD_STORAGE_BUCKET + '/dev/tests/' + self.file_name
signature_string = self.HTTP_method + '\n' + '\n' + self.content_type + "\n" + str(expiration) + '\n' + bucket_path
print(signature_string)
creds = ServiceAccountCredentials.from_json_keyfile_name(GOOGLE_SERVICE_CREDENTIALS)
client_email = creds.service_account_email
signature = creds.sign_blob(signature_string)[1]
encoded_signature = base64.urlsafe_b64encode(signature).decode('utf-8')
base_url = settings.CLOUD_STORAGE_ROOT + 'dev/tests/' + self.file_name


return base_url + '?GoogleAccessId=' + client_email + '&Expires=' + str(expiration) + '&Signature=' + encoded_signature

用于上传文件的客户端 Javascript

import $ from 'jquery';
import axios from 'axios';



$("document").ready( () => {
console.log('window loaded');


$("#id_audio_file").change(function() {

const file = this.files[0]

const url = window.location.href.replace('submit/', 'upload/');
$.get(url + `?filename=${file.name}&expiration=10&type=${file.type}`, (data) => {
upload(data.signed_url, file);
})

});
});



function upload(url, file) {

const config = {
headers: {
'Content-Type': file.type,
}
}

axios.put(url, file, config)
.then(function (res) {
console.log(res);
})
.catch(function (err) {
console.log(err);
});
}

我真的觉得我涵盖了这里的所有基础,但我显然遗漏了一些细节。任何帮助将不胜感激!

最佳答案

首先,确保您使用的服务帐户具有将文件上传到存储桶的适当权限:通过 IAM settings或通过 access control .

如果设置了,那就是代码的问题了。我已经尝试过您的方法,而不是手动 创建签名的 url,但它对我来说也失败了,出现了同样的错误。但是,我已经使用 the Google Cloud python client library 中的 generate_signed_url 方法通过 signedURL 上传了一个文件。 .少说话,多写代码:

from google.oauth2 import service_account
import base64
import json
from datetime import datetime, timedelta
import time
import requests
from google.cloud import storage

# client = storage.Client()
# bucket = client.get_bucket('a-test-bucket')
# blob = bucket.blob('/test/mpthreetest.mp3')

GOOGLE_SERVICE_CREDENTIALS = 'signed_url_account.json'
FILENAME = 'mpthreetest.mp3'
EXPIRATION = 1511826970 # epoch time
TYPE = 'audio/mp3'

creds = service_account.Credentials.from_service_account_file(GOOGLE_SERVICE_CREDENTIALS)
bucket = storage.Client().get_bucket('a-test-bucket')
blob = bucket.blob('test/mpthreetest.mp3')

signed_url = blob.generate_signed_url(method='PUT', expiration=EXPIRATION, content_type=TYPE, credentials=creds)

print (signed_url)

req = requests.put(signed_url, open(FILENAME), headers={'Content-Type': TYPE})
print(req.content)

关于javascript - 上传到 Google Cloud Storage 时获取 403 SignatureDoesNotMatch?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47381667/

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