require 'cgi' module GoogleCells class Worksheet < GoogleCells::GoogleObject include Fetcher include Util @permanent_attributes = [ :etag, :title, :updated_at, :cells_uri, :lists_uri, :spreadsheet, :row_count, :col_count ] define_accessors def rows GoogleCells::CellSelector::RowSelector.new(self) end class UpdateError < StandardError ; end def save! return if @changed_cells.nil? || @changed_cells.empty? batch_url = concat_url(cells_uri, "/batch") response = request(:post, batch_url, body: to_xml, headers:{ "Content-Type" => "application/atom+xml", "If-Match" => "*"}) doc = Nokogiri.parse(response.body) for entry in doc.css("atom|entry") interrupted = entry.css("batch|interrupted")[0] if interrupted raise(UpdateError, "Update failed: %s" % interrupted["reason"]) end if !(entry.css("batch|status").first["code"] =~ /^2/) raise(UpdateError, "Update failed for cell %s: %s" % [entry.css("atom|id").text, entry.css("batch|status")[0]["reason"]]) end end @changed_cells = {} true end def track_changes(cell) @changed_cells ||= {} @changed_cells[cell.title] = cell # track only most recent change nil end private def to_xml xml = <<-EOS #{e(self.cells_uri)} EOS @changed_cells.each do |title, cell| xml << cell.to_xml end xml << <<-"EOS" EOS end end end