require 'rexml/document'
require 'facebooker/session'

begin
    require 'nokogiri'
rescue Exception
end

module Facebooker
  class Parser

    module REXMLElementExtensions # :nodoc:
      def content
        self.text || ''
      end

      def [] key
        attributes[key]
      end

      def text?
        false
      end
    end

    module REXMLTextExtensions # :nodoc:
      def text?
        true
      end
    end

    ::REXML::Element.__send__(:include, REXMLElementExtensions)
    ::REXML::Text.__send__(:include, REXMLTextExtensions)

    def self.parse(method, data)
      Errors.process(data)
      parser = Parser::PARSERS[method]
      raise "Can't find a parser for '#{method}'" unless parser
      parser.process(data)
    end

    def self.array_of(response_element, element_name)
      values_to_return = []
      response_element.children.each do |element|
        next if element.text?
        next unless element.name == element_name
        values_to_return << yield(element)
      end
      values_to_return
    end

    def self.array_of_text_values(response_element, element_name)
      array_of(response_element, element_name) do |element|
        element.content.strip
      end
    end

    def self.array_of_hashes(response_element, element_name)
      array_of(response_element, element_name) do |element|
        hashinate(element)
      end
    end

    def self.element(name, data)
      data = data.body rescue data # either data or an HTTP response
      if Object.const_defined?(:Nokogiri)
        xml = Nokogiri::XML(data.strip)
        if node = xml.at(name)
          return node
        end
        if xml.root.name == name
          return xml.root
        end
      else
        doc = REXML::Document.new(data)
        doc.elements.each(name) do |element|
          return element
        end
      end
      raise "Element #{name} not found in #{data}"
    end
    
    def self.hash_or_value_for(element)
      if element.children.size == 1 && element.children.first.text?
        element.content.strip
      else
        # We can have lists in not list item
        if element['list'] == 'true'
          element.children.reject{|c| c.text? }.map { |subchild| hash_or_value_for(subchild)}
        else
          hashinate(element)
        end
      end
    end

    def self.hashinate(response_element)
      response_element.children.reject{|c| c.text? }.inject({}) do |hash, child|
        # If the node hasn't any child, and is not a list, we want empty strings, not empty hashes,
        #   except if attributes['nil'] == true
        hash[child.name] =
        if (child['nil'] == 'true')
          nil
        elsif (child.children.size == 1 && child.children.first.text?) || (child.children.size == 0 && child['list'] != 'true')
          anonymous_field_from(child, hash) || child.content.strip
        elsif child['list'] == 'true'
          child.children.reject{|c| c.text? }.map { |subchild| hash_or_value_for(subchild)}
        else
          child.children.reject{|c| c.text? }.inject({}) do |subhash, subchild|
            subhash[subchild.name] = hash_or_value_for(subchild)
            subhash
          end
        end #if (child.attributes)
        hash
      end #do |hash, child|
    end
    
    def self.hash_by_key_or_value_for(element, convert_1_to_true=false)
      if element.children.size == 0
        { element['key'] => nil }
      elsif element.children.size == 1 && element.children.first.text?
        { element['key'] => (convert_1_to_true ? element.content.strip == '1' : element.content.strip) }
      else
        hashinate_by_key(element, convert_1_to_true)
      end
    end

    # A modification to hashinate. The new dashboard API returns XML in a different format than
    # the other calls.  What used to be the element name has become an attribute called "key".
    def self.hashinate_by_key(response_element, convert_1_to_true=false)
      response_element.children.reject{|c| c.text? }.inject({}) do |hash, child|
        
        # If the node hasn't any child, and is not a list, we want empty strings, not empty hashes,
        #   except if attributes['nil'] == true
        hash[child['key']] =
        if (child['nil'] == 'true')
          nil
        elsif (child.children.size == 1 && child.children.first.text?) || (child.children.size == 0 && child['list'] != 'true')
          anonymous_field_from(child, hash) || (convert_1_to_true ? child.content.strip == '1' : child.content.strip)
        elsif child['list'] == 'true' && child.children.reject {|subchild| subchild.text?}.all? { |subchild| !subchild.text? && subchild['key'].nil? }
          child.children.reject{|c| c.text? }.map { |subchild| hash_by_key_or_value_for(subchild, convert_1_to_true)}
        elsif child['list'] == 'true'
          hash_by_key_or_value_for(child, convert_1_to_true)
        else
          child.children.reject{|c| c.text? }.inject({}) do |subhash, subchild|
            subhash[subchild['key']] = hash_by_key_or_value_for(subchild, convert_1_to_true)
            subhash
          end
        end
        hash
      end
    end



    def self.booleanize(response)
      response == "1" ? true : false
    end

    def self.anonymous_field_from(child, hash)
      if child.name == 'anon'
        (hash[child.name] || []) << child.content.strip
      end
    end

  end

  class RevokeAuthorization < Parser#:nodoc:
    def self.process(data)
      booleanize(data)
    end
  end

  class RevokeExtendedPermission < Parser#:nodoc:
    def self.process(data)
      booleanize(element('auth_revokeExtendedPermission_response', data).content.strip)
    end
  end

  class CreateToken < Parser#:nodoc:
    def self.process(data)
      element('auth_createToken_response', data).content.strip
    end
  end

  class RegisterUsers < Parser
    def self.process(data)
      array_of_text_values(element("connect_registerUsers_response", data), "connect_registerUsers_response_elt")
    end
  end

  class UnregisterUsers < Parser
    def self.process(data)
      array_of_text_values(element("connect_unregisterUsers_response", data), "connect_unregisterUsers_response_elt")
    end
  end

  class GetUnconnectedFriendsCount < Parser
    def self.process(data)
      hash_or_value_for(element("connect_getUnconnectedFriendsCount_response",data)).to_i
    end
  end

  class GetSession < Parser#:nodoc:
    def self.process(data)
      hashinate(element('auth_getSession_response', data))
    end
  end

  class IsAppUser < Parser#:nodoc:
    def self.process(data)
      element('users_isAppUser_response', data).content == '1'
    end
  end

  class GetFriends < Parser#:nodoc:
    def self.process(data)
      array_of_text_values(element('friends_get_response', data), 'uid')
    end
  end

  class FriendListsGet < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('friends_getLists_response', data), 'friendlist')
    end
  end

  class UserInfo < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('users_getInfo_response', data), 'user')
    end
  end

  class UserStandardInfo < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('users_getStandardInfo_response', data), 'standard_user_info')
    end
  end

  class GetLoggedInUser < Parser#:nodoc:
    def self.process(data)
      Integer(element('users_getLoggedInUser_response', data).content.strip)
    end
  end

  class PagesIsAdmin < Parser#:nodoc:
    def self.process(data)
      element('pages_isAdmin_response', data).content.strip == '1'
    end
  end

  class PagesGetInfo < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('pages_getInfo_response', data), 'page')
    end
  end

  class PagesIsFan < Parser#:nodoc:
    def self.process(data)
      element('pages_isFan_response', data).content.strip == '1'
    end
  end

  class PublishStoryToUser < Parser#:nodoc:
    def self.process(data)
      element('feed_publishStoryToUser_response', data).content.strip
    end
  end

  class StreamPublish < Parser#:nodoc:
    def self.process(data)
      element('stream_publish_response', data).content.strip
    end
  end
  
  class StreamAddComment < Parser#:nodoc:
    def self.process(data)
      element('stream_addComment_response', data).content.strip
    end
  end  

  class StreamAddLike < Parser#:nodoc:
    def self.process(data)
      element('stream_addLike_response', data).content.strip
    end
  end  

  class StreamRemoveLike < Parser#:nodoc:
    def self.process(data)
      booleanize(element('stream_removeLike_response', data).content.strip)
    end
  end  

  class RegisterTemplateBundle < Parser#:nodoc:
    def self.process(data)
      element('feed_registerTemplateBundle_response', data).content.to_i
    end
  end

  class GetRegisteredTemplateBundles < Parser
    def self.process(data)
      array_of_hashes(element('feed_getRegisteredTemplateBundles_response',data), 'template_bundle')
    end
  end

  class DeactivateTemplateBundleByID < Parser#:nodoc:
    def self.process(data)
      element('feed_deactivateTemplateBundleByID_response', data).content.strip == '1'
    end
  end

  class PublishUserAction < Parser#:nodoc:
    def self.process(data)
      element('feed_publishUserAction_response', data).children[1].content.strip == "1"
    end
  end

  class UploadNativeStrings < Parser#:nodoc:
    def self.process(data)
      element('intl_uploadNativeStrings_response', data).content.strip
    end
  end

  class PublishActionOfUser < Parser#:nodoc:
    def self.process(data)
      element('feed_publishActionOfUser_response', data).content.strip
    end
  end

  class PublishTemplatizedAction < Parser#:nodoc:
    def self.process(data)
      element('feed_publishTemplatizedAction_response', data).children[1].content.strip
    end
  end

  class SetAppProperties < Parser#:nodoc:
    def self.process(data)
      element('admin_setAppProperties_response', data).content.strip
    end
  end

  class GetAppProperties < Parser#:nodoc:
    def self.process(data)
      element('admin_getAppProperties_response', data).content.strip
    end
  end

  class SetRestrictionInfo < Parser#:nodoc:
    def self.process(data)
      element('admin_setRestrictionInfo_response', data).content.strip
    end
  end

  class GetRestrictionInfo < Parser#:nodoc:
    def self.process(data)
      element('admin_getRestrictionInfo_response', data).content.strip
    end
  end

  class GetAllocation < Parser#:nodoc:
    def self.process(data)
      element('admin_getAllocation_response', data).content.strip
    end
  end
  
  class GetPublicInfo < Parser#:nodoc:
    def self.process(data)
      hashinate(element('application_getPublicInfo_response', data))
    end
  end

  class CommentsAdd < Parser#:nodoc:
    def self.process(data)
      element('comments_add_response', data).content.strip
    end
  end
  
  class CommentsRemove < Parser#:nodoc:
    def self.process(data)
      booleanize(data)
    end
  end
  
  class CommentsGet < Parser#:nodoc:
    def self.process(data)
       array_of_hashes(element('comments_get_response', data), 'comment')
    end
  end

  class BatchRun < Parser #:nodoc:
    class << self
      def current_batch=(current_batch)
        Thread.current[:facebooker_current_batch]=current_batch
      end
      def current_batch
        Thread.current[:facebooker_current_batch]
      end
    end
    def self.process(data)
      array_of_text_values(element('batch_run_response',data),"batch_run_response_elt").each_with_index do |response,i|
        batch_request=current_batch[i]
        body=Struct.new(:body).new
        body.body=response
        begin
          batch_request.result=Parser.parse(batch_request.method,body)
        rescue Exception=>ex
          batch_request.exception_raised=ex
        end
      end
    end
  end

  class GetAppUsers < Parser#:nodoc:
    def self.process(data)
      array_of_text_values(element('friends_getAppUsers_response', data), 'uid')
    end
  end

  class MessageGetThreadsInFolder < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('message_getThreadsInFolder_response', data), 'thread')
    end
  end

  class NotificationsGet < Parser#:nodoc:
    def self.process(data)
      hashinate(element('notifications_get_response', data))
    end
  end

  class NotificationsSend < Parser#:nodoc:
    def self.process(data)
      element('notifications_send_response', data).content.strip
    end
  end

  class NotificationsSendEmail < Parser#:nodoc:
    def self.process(data)
      element('notifications_sendEmail_response', data).content.strip
    end
  end

  class GetTags < Parser#nodoc:
    def self.process(data)
      array_of_hashes(element('photos_getTags_response', data), 'photo_tag')
    end
  end

  class AddTags < Parser#nodoc:
    def self.process(data)
      element('photos_addTag_response', data)
    end
  end

  class GetPhotos < Parser#nodoc:
    def self.process(data)
      array_of_hashes(element('photos_get_response', data), 'photo')
    end
  end

  class GetAlbums < Parser#nodoc:
    def self.process(data)
      array_of_hashes(element('photos_getAlbums_response', data), 'album')
    end
  end

  class GetStream < Parser #:nodoc:
    def self.process(data)
      response = {}
      response[:albums] = array_of_hashes(element('stream_get_response/albums', data), 'album')
      response[:posts] = array_of_hashes(element('stream_get_response/posts', data), 'stream_post')
      response[:profile] = array_of_hashes(element('stream_get_response/profiles', data), 'profile')
      response
    end
  end

  class CreateAlbum < Parser#:nodoc:
    def self.process(data)
      hashinate(element('photos_createAlbum_response', data))
    end
  end

  class UploadPhoto < Parser#:nodoc:
    def self.process(data)
      hashinate(element('photos_upload_response', data))
    end
  end

  class UploadVideo < Parser#:nodoc:
    def self.process(data)
      hashinate(element('video_upload_response', data))
    end
  end

  class SendRequest < Parser#:nodoc:
    def self.process(data)
      element('notifications_sendRequest_response', data).content.strip
    end
  end

  class ProfileFBML < Parser#:nodoc:
    def self.process(data)
      element('profile_getFBML_response', data).content.strip
    end
  end

  class ProfileFBMLSet < Parser#:nodoc:
    def self.process(data)
      element('profile_setFBML_response', data).content.strip
    end
  end

  class ProfileInfo < Parser#:nodoc:
    def self.process(data)
      hashinate(element('profile_getInfo_response info_fields', data))
    end
  end

  class ProfileInfoSet < Parser#:nodoc:
    def self.process(data)
      element('profile_setInfo_response', data).content.strip
    end
  end

  class FqlQuery < Parser#nodoc
    def self.process(data)
      root = element('fql_query_response', data)
      first_child = root.children.reject{|c| c.text? }.first
      first_child.nil? ? [] : [first_child.name, array_of_hashes(root, first_child.name)]
    end
  end

  class FqlMultiquery < Parser#nodoc
    def self.process(data)
      root = element('fql_multiquery_response', data)
      root.children.reject { |child| child.text? }.map do |elm|
        elm.children.reject { |child| child.text? }.map do |query|
          if 'name' == query.name
            query.text
          else
            list = query.children.reject { |child| child.text? }
            if list.length == 0
              []
            else
              [list.first.name, array_of_hashes(query, list.first.name)]
            end
          end
        end
      end
    end
  end

  class SetRefHandle < Parser#:nodoc:
    def self.process(data)
      element('fbml_setRefHandle_response', data).content.strip
    end
  end

  class RefreshRefURL < Parser#:nodoc:
    def self.process(data)
      element('fbml_refreshRefUrl_response', data).content.strip
    end
  end

  class RefreshImgSrc < Parser#:nodoc:
    def self.process(data)
      element('fbml_refreshImgSrc_response', data).content.strip
    end
  end

  class SetCookie < Parser#:nodoc:
    def self.process(data)
      element('data_setCookie_response', data).content.strip
    end
  end

  class GetCookies < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('data_getCookies_response', data), 'cookies')
    end
  end
  
  class EventsRsvp < Parser#:nodoc:
    def self.process(data)
       element('events_rsvp_response', data).content.strip
    end
  end
   
  class EventsCreate < Parser#:nodoc:
    def self.process(data)
      element('events_create_response', data).content.strip
    end
  end
  
  class EventsCancel < Parser#:nodoc:
    def self.process(data)
      element('events_cancel_response', data).content.strip
    end
  end

  class EventsGet < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('events_get_response', data), 'event')
    end
  end

  class GroupGetMembers < Parser#:nodoc:
    def self.process(data)
      root = element('groups_getMembers_response', data)
      result = ['members', 'admins', 'officers', 'not_replied'].map do |position|
        array_of(root, position) {|element| element}.map do |element|
          array_of_text_values(element, 'uid').map do |uid|
            {:position => position}.merge(:uid => uid)
          end
        end
      end.flatten
    end
  end

  class EventMembersGet < Parser#:nodoc:
    def self.process(data)
      root = element('events_getMembers_response', data)
      result = ['attending', 'declined', 'unsure', 'not_replied'].map do |rsvp_status|
        array_of(root, rsvp_status) {|element| element}.map do |element|
          array_of_text_values(element, 'uid').map do |uid|
            {:rsvp_status => rsvp_status}.merge(:uid => uid)
          end
        end
      end.flatten
    end
  end

  class GroupsGet < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('groups_get_response', data), 'group')
    end
  end

  class AreFriends < Parser#:nodoc:
    def self.process(data)
      array_of_hashes(element('friends_areFriends_response', data), 'friend_info').inject({}) do |memo, hash|
        memo[[Integer(hash['uid1']), Integer(hash['uid2'])].sort] = are_friends?(hash['are_friends'])
        memo
      end
    end

    private
    def self.are_friends?(raw_value)
      if raw_value == '1'
        true
      elsif raw_value == '0'
        false
      else
        nil
      end
    end
  end

  class SetStatus < Parser
    def self.process(data)
      element('users_setStatus_response',data).content.strip == '1'
    end
  end

  class GetStatus < Parser # :nodoc:
    def self.process(data)
      array_of_hashes(element('status_get_response',data),'user_status')
    end
  end

  class GetPreference < Parser#:nodoc:
    def self.process(data)
      element('data_getUserPreference_response', data).content.strip
    end
  end

  class SetPreference < Parser#:nodoc:
    def self.process(data)
      element('data_setUserPreference_response', data).content.strip
    end
  end

  class UserHasPermission < Parser
    def self.process(data)
      element('users_hasAppPermission_response', data).content.strip
    end
  end

  class SmsSend < Parser#:nodoc:
    def self.process(data)
      element('sms_send_response', data).content.strip == '0'
    end
  end

  class SmsCanSend < Parser#:nodoc:
    def self.process(data)
      element('sms_canSend_response', data).content.strip
    end
  end

  class DashboardGetCount < Parser
    def self.process(data)
      element('dashboard_getCount_response', data).content.strip
    end
  end

  class DashboardSetCount < Parser
    def self.process(data)
      element('dashboard_setCount_response', data).content.strip == '1'
    end
  end

  class DashboardIncrementCount < Parser
    def self.process(data)
      element('dashboard_incrementCount_response', data).content.strip == '1'
    end
  end

  class DashboardDecrementCount < Parser
    def self.process(data)
      element('dashboard_decrementCount_response', data).content.strip == '1'
    end
  end
  
  class DashboardMultiGetCount < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiGetCount_response', data))
    end
  end
  
  class DashboardMultiSetCount < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiSetCount_response', data), true)
    end
  end
  
  class DashboardMultiIncrementCount < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiIncrementCount_response', data), true)
    end
  end
  
  class DashboardMultiDecrementCount < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiDecrementCount_response', data), true)
    end
  end
  
  class DashboardAddGlobalNews < Parser
    def self.process(data)
      element('dashboard_addGlobalNews_response', data).content.strip
    end
  end
  
  # Currently, always returns all
  class DashboardGetGlobalNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_getGlobalNews_response', data))
    end
  end
  
  class DashboardClearGlobalNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_clearGlobalNews_response', data), true)
    end
  end
  
  class DashboardAddNews < Parser
    def self.process(data)
      element('dashboard_addNews_response', data).content.strip
    end
  end
  
  class DashboardGetNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_getNews_response', data))
    end
  end
  
  class DashboardClearNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_clearNews_response', data), true)
    end
  end
  
  class DashboardMultiAddNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiAddNews_response', data))
    end
  end
  
  class DashboardMultiClearNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiClearNews_response', data), true)
    end
  end
  
  class DashboardMultiGetNews < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_multiGetNews_response', data))
    end
  end
  
  class DashboardPublishActivity < Parser
    def self.process(data)
      element('dashboard_publishActivity_response', data).content.strip
    end
  end
  
  class DashboardRemoveActivity < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_removeActivity_response', data), true)
    end
  end
  
  class DashboardGetActivity < Parser
    def self.process(data)
      hashinate_by_key(element('dashboard_getActivity_response', data))
    end
  end


  class Errors < Parser#:nodoc:
    EXCEPTIONS = {
      1 	=> Facebooker::Session::UnknownError,
      2 	=> Facebooker::Session::ServiceUnavailable,
      4 	=> Facebooker::Session::MaxRequestsDepleted,
      5 	=> Facebooker::Session::HostNotAllowed,
      100 => Facebooker::Session::MissingOrInvalidParameter,
      101 => Facebooker::Session::InvalidAPIKey,
      102 => Facebooker::Session::SessionExpired,
      103 => Facebooker::Session::CallOutOfOrder,
      104 => Facebooker::Session::IncorrectSignature,
      120 => Facebooker::Session::InvalidAlbumId,
      200 => Facebooker::Session::PermissionError,
      250 => Facebooker::Session::ExtendedPermissionRequired,
      321 => Facebooker::Session::AlbumIsFull,
      324 => Facebooker::Session::MissingOrInvalidImageFile,
      325 => Facebooker::Session::TooManyUnapprovedPhotosPending,
      330 => Facebooker::Session::TemplateDataMissingRequiredTokens,
      340 => Facebooker::Session::TooManyUserCalls,
      341 => Facebooker::Session::TooManyUserActionCalls,
      342 => Facebooker::Session::InvalidFeedTitleLink,
      343 => Facebooker::Session::InvalidFeedTitleLength,
      344 => Facebooker::Session::InvalidFeedTitleName,
      345 => Facebooker::Session::BlankFeedTitle,
      346 => Facebooker::Session::FeedBodyLengthTooLong,
      347 => Facebooker::Session::InvalidFeedPhotoSource,
      348 => Facebooker::Session::InvalidFeedPhotoLink,
      330 => Facebooker::Session::FeedMarkupInvalid,
      360 => Facebooker::Session::FeedTitleDataInvalid,
      361 => Facebooker::Session::FeedTitleTemplateInvalid,
      362 => Facebooker::Session::FeedBodyDataInvalid,
      363 => Facebooker::Session::FeedBodyTemplateInvalid,
      364 => Facebooker::Session::FeedPhotosNotRetrieved,
      366 => Facebooker::Session::FeedTargetIdsInvalid,
      601 => Facebooker::Session::FQLParseError,
      602 => Facebooker::Session::FQLFieldDoesNotExist,
      603 => Facebooker::Session::FQLTableDoesNotExist,
      604 => Facebooker::Session::FQLStatementNotIndexable,
      605 => Facebooker::Session::FQLFunctionDoesNotExist,
      606 => Facebooker::Session::FQLWrongNumberArgumentsPassedToFunction,
      612 => Facebooker::Session::ReadMailboxExtendedPermissionRequired,
      807 => Facebooker::Session::TemplateBundleInvalid
    }
    def self.process(data)
      response_element = element('error_response', data) rescue nil
      if response_element
        hash = hashinate(response_element)
        exception = EXCEPTIONS[Integer(hash['error_code'])] || StandardError
        raise exception, hash['error_msg']
      end
    end
  end

  class Parser
    PARSERS = {
      'facebook.auth.revokeAuthorization' => RevokeAuthorization,
      'facebook.auth.revokeExtendedPermission' => RevokeExtendedPermission,
      'facebook.auth.createToken' => CreateToken,
      'facebook.auth.getSession' => GetSession,
      'facebook.connect.registerUsers' => RegisterUsers,
      'facebook.connect.unregisterUsers' => UnregisterUsers,
      'facebook.connect.getUnconnectedFriendsCount' => GetUnconnectedFriendsCount,
      'facebook.users.getInfo' => UserInfo,
      'facebook.users.getStandardInfo' => UserStandardInfo,
      'facebook.users.setStatus' => SetStatus,
      'facebook.status.get' => GetStatus,
      'facebook.users.getLoggedInUser' => GetLoggedInUser,
      'facebook.users.hasAppPermission' => UserHasPermission,
      'facebook.users.isAppUser' => IsAppUser,
      'facebook.pages.isAdmin' => PagesIsAdmin,
      'facebook.pages.getInfo' => PagesGetInfo,
      'facebook.pages.isFan' => PagesIsFan,
      'facebook.friends.get' => GetFriends,
      'facebook.friends.getLists' => FriendListsGet,
      'facebook.friends.areFriends' => AreFriends,
      'facebook.friends.getAppUsers' => GetAppUsers,
      'facebook.feed.publishStoryToUser' => PublishStoryToUser,
      'facebook.feed.publishActionOfUser' => PublishActionOfUser,
      'facebook.feed.publishTemplatizedAction' => PublishTemplatizedAction,
      'facebook.feed.registerTemplateBundle' => RegisterTemplateBundle,
      'facebook.feed.deactivateTemplateBundleByID' => DeactivateTemplateBundleByID,
      'facebook.feed.getRegisteredTemplateBundles' => GetRegisteredTemplateBundles,
      'facebook.feed.publishUserAction' => PublishUserAction,
      'facebook.message.getThreadsInFolder' => MessageGetThreadsInFolder,
      'facebook.notifications.get' => NotificationsGet,
      'facebook.notifications.send' => NotificationsSend,
      'facebook.notifications.sendRequest' => SendRequest,
      'facebook.profile.getFBML' => ProfileFBML,
      'facebook.profile.setFBML' => ProfileFBMLSet,
      'facebook.profile.getInfo' => ProfileInfo,
      'facebook.profile.setInfo' => ProfileInfoSet,
      'facebook.fbml.setRefHandle' => SetRefHandle,
      'facebook.fbml.refreshRefUrl' => RefreshRefURL,
      'facebook.fbml.refreshImgSrc' => RefreshImgSrc,
      'facebook.data.setCookie' => SetCookie,
      'facebook.data.getCookies' => GetCookies,
      'facebook.admin.setAppProperties' => SetAppProperties,
      'facebook.admin.getAppProperties' => GetAppProperties,
      'facebook.admin.setRestrictionInfo' => SetRestrictionInfo,
      'facebook.admin.getRestrictionInfo' => GetRestrictionInfo,
      'facebook.admin.getAllocation' => GetAllocation,
      'facebook.application.getPublicInfo' => GetPublicInfo,
      'facebook.batch.run' => BatchRun,
      'facebook.fql.query' => FqlQuery,
      'facebook.fql.multiquery' => FqlMultiquery,
      'facebook.photos.get' => GetPhotos,
      'facebook.photos.getAlbums' => GetAlbums,
      'facebook.photos.createAlbum' => CreateAlbum,
      'facebook.photos.getTags' => GetTags,
      'facebook.photos.addTag' => AddTags,
      'facebook.photos.upload' => UploadPhoto,
      'facebook.stream.get' => GetStream,
      'facebook.stream.publish' => StreamPublish,
      'facebook.stream.addComment' => StreamAddComment,
      'facebook.stream.addLike' => StreamAddLike,
      'facebook.stream.removeLike' => StreamRemoveLike,
      'facebook.events.create' => EventsCreate,
      'facebook.events.cancel' => EventsCancel,
      'facebook.events.get' => EventsGet,
      'facebook.events.rsvp' => EventsRsvp,
      'facebook.groups.get' => GroupsGet,
      'facebook.events.getMembers' => EventMembersGet,
      'facebook.groups.getMembers' => GroupGetMembers,
      'facebook.notifications.sendEmail' => NotificationsSendEmail,
      'facebook.data.getUserPreference' => GetPreference,
      'facebook.data.setUserPreference' => SetPreference,
      'facebook.video.upload' => UploadVideo,
      'facebook.sms.send' => SmsSend,
      'facebook.sms.canSend' => SmsCanSend,
      'facebook.comments.add' => CommentsAdd,
      'facebook.comments.remove' => CommentsRemove,
      'facebook.comments.get' => CommentsGet,
      'facebook.dashboard.setCount' => DashboardSetCount,
      'facebook.dashboard.getCount' => DashboardGetCount,
      'facebook.dashboard.incrementCount' => DashboardIncrementCount,
      'facebook.dashboard.decrementCount' => DashboardDecrementCount,
      'facebook.dashboard.multiGetCount' => DashboardMultiGetCount,
      'facebook.dashboard.multiSetCount' => DashboardMultiSetCount,
      'facebook.dashboard.multiIncrementCount' => DashboardMultiIncrementCount,
      'facebook.dashboard.multiDecrementCount' => DashboardMultiDecrementCount,
      'facebook.dashboard.addGlobalNews' => DashboardAddGlobalNews,
      'facebook.dashboard.getGlobalNews' => DashboardGetGlobalNews,
      'facebook.dashboard.clearGlobalNews' => DashboardClearGlobalNews,
      'facebook.dashboard.addNews' => DashboardAddNews,
      'facebook.dashboard.getNews' => DashboardGetNews,
      'facebook.dashboard.clearNews' => DashboardClearNews,
      'facebook.dashboard.multiAddNews' => DashboardMultiAddNews,
      'facebook.dashboard.multiGetNews' => DashboardMultiGetNews,
      'facebook.dashboard.multiClearNews' => DashboardMultiClearNews,
      'facebook.dashboard.publishActivity' => DashboardPublishActivity,
      'facebook.dashboard.removeActivity' => DashboardRemoveActivity,
      'facebook.dashboard.getActivity' => DashboardGetActivity,
      'facebook.intl.uploadNativeStrings' => UploadNativeStrings
    }
  end
end