0

我试图在 BigQuery 上创建一个表 - 我有一个数据集,需要使用 api 添加一个表并从云存储导入数据 (json.tar.gz)。我需要能够使用 ruby​​ 客户端来自动化整个过程。我有两个问题:

  1. 我已经阅读了文档并试图让它上传(下面的代码)并且没有成功并且完全不知道我做错了什么。有人可以启发我或指出我正确的方向吗?

  2. 一旦我提出请求,我怎么知道工作何时真正完成?从 API 来看,我想我打算使用 jobs.get 请求?由于没有完成第一部分,我一直无法看到这方面。

这是我下面的代码。

config= {
  'configuration'=> {
      'load'=> {
        'sourceUris'=> ["gs://person-bucket/person_json.tar.gz"],
        'schema'=> {
          'fields'=> [
            { 'name'=>'person_id', 'type'=>'integer' },
            { 'name'=> 'person_name', 'type'=>'string' },
            { 'name'=> 'logged_in_at', 'type'=>'timestamp' },
          ]
        },
        'destinationTable'=> {
          'projectId'=> "XXXXXXXXX",
          'datasetId'=> "personDataset",
          'tableId'=> "person"
        },
        'createDisposition' => 'CREATE_IF_NEEDED',
        'maxBadRecords'=> 10,
      }
    },
    'jobReference'=>{'projectId'=>XXXXXXXXX}
  }

multipart_boundary="xxx"
body = "--#{multipart_boundary}\n"
body += "Content-Type: application/json; charset=UTF-8\n\n"
body += "#{config.to_json}\n"
body += "--#{multipart_boundary}\n"
body +="Content-Type: application/octet-stream\n\n"
body += "--#{multipart_boundary}--\n"

param_hash = {:api_method=> bigquery.jobs.insert }
param_hash[:parameters] = {'projectId' => 'XXXXXXXX'}
param_hash[:body] = body
param_hash[:headers] = {'Content-Type' => "multipart/related; boundary=#{multipart_boundary}"}

result = @client.execute(param_hash)
puts JSON.parse(result.response.header)

我收到以下错误:{"error"=>{"errors"=>[{"domain"=>"global", "reason"=>"wrongUrlForUpload", "message"=>"上传必须发送到上传网址。将此请求重新发送到https://www.googleapis.com/upload/bigquery/v2/projects/XXXXXXXX/jobs "}], "code"=>400, "message"=>"上传必须是发送到上传 URL。将此请求重新发送到https://www.googleapis.com/upload/bigquery/v2/projects/XXXXXXXX/jobs "}}

从请求标头来看,它似乎要转到错误所说的应该转到的同一个 URI,我不知道如何继续。任何帮助将非常感激。

谢谢你,祝你有美好的一天!

4

2 回答 2

0

谢谢你。答案已解决。请参阅此处: 如何将 json 从云存储上的文件导入 Bigquery

我认为可恢复上传部分(http://rubydoc.info/github/google/google-api-ruby-client/file/README.md#Media_Upload)的文档中的代码行应为:

result = client.execute(:api_method => drive.files.insert,

否则,此行将抛出一个错误,'result' undefined:

upload = result.resumable_upload
于 2013-11-14T23:37:06.163 回答
0
  1. 由于这是一个“媒体上传”请求,因此发出请求的协议略有不同。此处的 ruby​​ 文档http://rubydoc.info/github/google/google-api-ruby-client/file/README.md#Media_Upload对其进行了更详细的描述。我会使用可恢复上传而不是分段上传,因为它更简单。

  2. 是的,正如您所怀疑的,知道何时完成的方法是执行 jobs.get() 来查找正在运行的作业的状态。作业 ID 将在 jobs.insert() 的响应中返回。如果您想要更多控制,您可以传递您自己的作业 ID,以便在 jobs.insert() 调用返回错误的情况下,您可以查明作业是否实际开始。

于 2013-11-14T15:52:47.603 回答