# Copyright 2010 Google Inc. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require 'time' require 'json' require 'base64' require 'autoparse' require 'addressable/uri' require 'addressable/template' require 'google/inflection' require 'google/api_client/errors' module Google class APIClient module Schema def self.parse(api, schema_data) # This method is super-long, but hard to break up due to the # unavoidable dependence on closures and execution context. schema_name = schema_data['id'] # Due to an oversight, schema IDs may not be URI references. # TODO(bobaman): Remove this code once this has been resolved. schema_uri = ( api.document_base + (schema_name[0..0] != '#' ? '#' + schema_name : schema_name) ) # puts schema_uri # Due to an oversight, schema IDs may not be URI references. # TODO(bobaman): Remove this whole lambda once this has been resolved. reformat_references = lambda do |data| # This code is not particularly efficient due to recursive traversal # and excess object creation, but this hopefully shouldn't be an # issue since it should only be called only once per schema per # process. if data.kind_of?(Hash) && data['$ref'] reference = data['$ref'] reference = '#' + reference if reference[0..0] != '#' data.merge({ '$ref' => reference }) elsif data.kind_of?(Hash) data.inject({}) do |accu, (key, value)| if value.kind_of?(Hash) accu[key] = reformat_references.call(value) else accu[key] = value end accu end else data end end schema_data = reformat_references.call(schema_data) # puts schema_data.inspect if schema_name api_name_string = Google::INFLECTOR.camelize(api.name) api_version_string = Google::INFLECTOR.camelize(api.version).gsub('.', '_') if Google::APIClient::Schema.const_defined?(api_name_string) api_name = Google::APIClient::Schema.const_get(api_name_string) else api_name = Google::APIClient::Schema.const_set( api_name_string, Module.new ) end if api_name.const_defined?(api_version_string) api_version = api_name.const_get(api_version_string) else api_version = api_name.const_set(api_version_string, Module.new) end if api_version.const_defined?(schema_name) schema_class = api_version.const_get(schema_name) end end # It's possible the schema has already been defined. If so, don't # redefine it. This means that reloading a schema which has already # been loaded into memory is not possible. unless schema_class schema_class = AutoParse.generate(schema_data, schema_uri) if schema_name api_version.const_set(schema_name, schema_class) end end return schema_class end end end end