3

当我的 DeltaSynWorker 运行时,我收到 NoMethodErrors。这发生在作为当前工作应用程序的修订而构建的应用程序中。我不是最初的程序员,我是从 Java 背景开始的(我提到这一点是因为我可能遗漏了一些对其他人来说非常明显的东西)。NoMethodeError当代码与当前在不同 Web 应用程序中正常工作的代码非常相似时,我无法弄清楚为什么会被抛出。

错误:

NoMethodError: undefined method `client' for #<ApiRequestBuilder:0x0000000705a8f0>

delta_sync_worker.rb

class DeltaSyncWorker < SyncWorker

  include Sidekiq::Worker
  sidekiq_options queue: "delta_syncs"

  def perform(subscriber_id, client_id)

      sleep(10)
      current_subscriber = ApiSubscriberDecorator.decorate(Subscriber.find(subscriber_id))
      Time.zone = current_subscriber.time_zone
      client = ApiClientDecorator.decorate(Client.find(client_id))
      arb = ApiRequestBuilder.new(URI.parse(SR_HOST + '/servlet/sync/process'))

      if arb.client(:subscriber => current_subscriber, :client => client)

        arb.transmit

        if arb.success?
          current_subscriber.touch(:sync_updated_at)
          decorated_client = ClientDecorator.decorate(client.model)        
          ConfirmationsSyncWorker.perform_in(1.hours, current_subscriber.id)
        else
          error_params = {:subscriber => current_subscriber.id, :response_body =>            arb.response.body, :request_body => arb.request.body, :sync_type => "DeltaSyncWorker"}
          Airbrake.notify(:error_class => "sync_error", :error_message => "Sync Error: #{arb.response.message}", :params => error_params)
        end
      end
    end
  end

api_request_builder.rb

require 'nokogiri'
class ApiRequestBuilder < AbstractController::Base
  include AbstractController::Rendering
  include AbstractController::Layouts
  include AbstractController::Helpers
  include AbstractController::Translation
  include AbstractController::AssetPaths

  self.view_paths = "app/api"

  attr_accessor :request_body, :request, :response, :append_request_headers, :request_method, :url

  def initialize(url, *args)
    @url = url
    if args
      args.each do |arg|
        arg.each_pair{ |k,v| instance_variable_set("@#{k.to_s}", v) }
      end
    end
  end

  # this will search for an api request template in api/requests, render that template and set any instance variables
  def method_missing(meth, *args, &block)

    if lookup_context.template_exists?("requests/#{meth.to_s}")

      if args
        args.each do |arg|
          arg.each_pair{|k,v| instance_variable_set("@#{k.to_s}", v) }
        end
      end

      @request_body = (render :template => "requests/#{meth.to_s}")
    else
      super
    end

  end

  def transmit
    @request_method ||= "Post"
    @request = "Net::HTTP::#{@request_method}".constantize.new(@url.path)
    @request['x-ss-user'] = @subscriber.sr_user if @subscriber &&        @subscriber.sr_user.present?
    @request['x-ss-pwd'] =  @subscriber.sr_password if @subscriber && @subscriber.sr_password.present?

    unless @append_request_headers.nil?
      @append_request_headers.each_pair{ |k,v| @request[k] = v }
    end
    @request.body = @request_body if request_body? && @request.request_body_permitted?

    @http = Net::HTTP.new(@url.host, @url.port)
    @http.use_ssl = true
    @http.verify_mode = OpenSSL::SSL::VERIFY_NONE
    @response = @http.request(@request)

  end

  def success?
    if @response.code == 200.to_s
      return true
    else
      return false
    end
  end

  def request_body?
    unless @request_body.nil?
      return true
    else
      return false
    end
  end
end

我一直在NoMethodError这里查看其他问题,但找不到适合我情况的答案。任何帮助是极大的赞赏。谢谢!

4

2 回答 2

1

method_missing将捕获未定义方法的已发送消息,最后的调用super会将其传递给 Ruby 的标准method_missing行为,这就是您所看到的 ( NoMethodError)。但是,这只发生在if不满足条件的情况下,这就是我怀疑这里发生的情况,并且可以解释为什么它在某些情况下有效但在其他情况下无效。对 的调用:client,在继承链中没有找到匹配的方法,将寻找一个名为“requests/client”的模板 - 尝试添加这个模板,看看是否能解决问题。

于 2013-05-22T18:56:57.697 回答
1

我知道我以前见过这个,我觉得它看起来不像,但基本上方法丢失只是拦截方法调用,当你调用 arb.client 时,它被方法丢失捕获,因此尝试渲染api/client.xml.arb 或 api 无论文件类型是什么。-- 请注意,在初始化程序目录中应该有一个名为 somethig 的文件,例如 api_template_handler.rb 或 arb_temmplate_handler.rb,它允许 rails 在视图目录中查看该模板类型 - 确保首先存在。另外旁注,_client.xml.api 是该目录中其他 api 请求使用的部分(完全同步),

要调试 Id,请在以下行上方

if lookup_context.template_exists?("requests/#{meth.to_s}")

添加日志语句

Rails.logger.debug "Can I see View?" + lookup_context.template_exists?("requests/#{meth.to_s}")

如果为真,那么问题是由于缺少方法而没有正确捕获错误。如果为 false,则 sidekiq 工作人员未正确加载 rails,或者未将视图路径添加到 rails 视图路径。

如果为真,我猜它可能与未加载客户端模型或客户端模型上的属性不存在有关,构建器正在尝试调用,并且错误以某种方式冒泡到 api 请求构建器类.

哦,也只是一般的东西,但要确保 redis 和 sidekiq 正在运行,如果不是本地环境,请重新启动乘客。

让我知道事情的后续。

于 2013-05-23T04:26:44.643 回答