require 'net/http' require 'cgi' require 'openssl' require 'rexml/document' module GoogleApps class Transport attr_reader :request, :response, :domain, :feeds attr_accessor :auth, :user, :group, :nickname, :export BOUNDARY = "=AaB03xDFHT8xgg" PAGE_SIZE = { user: 100, group: 200 } def initialize(domain, targets = {}) @auth = targets[:auth] || "https://www.google.com/accounts/ClientLogin" @user = targets[:user] || "https://apps-apis.google.com/a/feeds/#{domain}/user/2.0" @pubkey = targets[:pubkey] || "https://apps-apis.google.com/a/feeds/compliance/audit/publickey/#{domain}" @migration = targets[:migration] || "https://apps-apis.google.com/a/feeds/migration/2.0/#{domain}" @group = targets[:group] || "https://apps-apis.google.com/a/feeds/group/2.0/#{domain}" @nickname = targets[:nickname] || "https://apps-apis.google.com/a/feeds/#{domain}/nickname/2.0" @export = targets[:export] || "https://apps-apis.google.com/a/feeds/compliance/audit/mail/export/#{domain}" @domain = domain @token = nil @response = nil @request = nil @feeds = [] end # authenticate will take the provided account and # password and attempt to authenticate them with # Google # # authenticate 'username@domain', 'password' # # authenticate returns the HTTP response received # from Google def authenticate(account, pass) uri = URI(@auth) @request = Net::HTTP::Post.new(uri.path) @request.body = auth_body(account, pass) set_headers :auth @response = request uri set_auth_token @response end # request_export performs the GoogleApps API call to # generate a mailbox export. It takes the username # and an GoogleApps::Atom::Export instance as # arguments # # request_export 'username', document # # request_export returns the HTTP response received # from Google. def request_export(username, document) add(@export + "/#{username}", document) end # export_status checks the status of a mailbox export # request. It takes the username and the request_id # as arguments # # export_status 'username', 847576 # # export_status will return the status of the HTTP # response from Google def export_status(username, req_id) get(@export + "/#{username}", req_id) end def fetch_export(username, req_id, filename) # :nodoc: # TODO: Shouldn't rely on export_status being run first. Self, this is lazy and stupid. export_status(username, req_id) doc = REXML::Document.new(@response.body) urls = [] doc.elements.each('entry/apps:property') do |property| urls << property.attributes['value'] if property.attributes['name'].match 'fileUrl' end urls.each do |url| download(url, filename + "#{urls.index(url)}") end end # download makes a get request of the provided url # and writes the body to the provided filename. # # download 'url', 'save_file' def download(url, filename) uri = URI(url) @request = Net::HTTP::Get.new uri.path set_headers :user File.open(filename, "w") do |file| file.puts request(uri).body end end # get is a generic target for method_missing. It is # intended to handle the general case of retrieving a # record from the Google Apps Domain. It takes an API # endpoint and an id as arguments. # # get 'endpoint', 'username' # # get returns the HTTP response received from Google. def get(endpoint, id = nil) # TODO: Need to handle uri.scheme == 'https') do |http| http.request(@request) end end def set_headers(request_type) case request_type when :auth @request['content-type'] = "application/x-www-form-urlencoded" when :migrate @request['content-type'] = "multipart/related; boundary=\"#{BOUNDARY}\"" @request['authorization'] = "GoogleLogin auth=#{@token}" else @request['content-type'] = "application/atom+xml" @request['authorization'] = "GoogleLogin auth=#{@token}" end end def multi_part(properties, message) post_body = [] post_body << "--#{BOUNDARY}\n" post_body << "Content-Type: application/atom+xml\n\n" post_body << properties.to_s post_body << "\n--#{BOUNDARY}\n" post_body << "Content-Type: message/rfc822\n\n" post_body << message.to_s post_body << "\n--#{BOUNDARY}--}" post_body.join end end end