Aucune description http://j1x-huginn.herokuapp.com

agents_controller.rb 8.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. class AgentsController < ApplicationController
  2. include DotHelper
  3. include ActionView::Helpers::TextHelper
  4. include SortableTable
  5. def index
  6. set_table_sort sorts: %w[name created_at last_check_at last_event_at last_receive_at], default: { created_at: :desc }
  7. @agents = current_user.agents.preload(:scenarios, :controllers).reorder(table_sort).page(params[:page])
  8. if show_only_enabled_agents?
  9. @agents = @agents.where(disabled: false)
  10. end
  11. respond_to do |format|
  12. format.html
  13. format.json { render json: @agents }
  14. end
  15. end
  16. def toggle_visibility
  17. if show_only_enabled_agents?
  18. mark_all_agents_viewable
  19. else
  20. set_only_enabled_agents_as_viewable
  21. end
  22. redirect_to agents_path
  23. end
  24. def handle_details_post
  25. @agent = current_user.agents.find(params[:id])
  26. if @agent.respond_to?(:handle_details_post)
  27. render :json => @agent.handle_details_post(params) || {}
  28. else
  29. @agent.error "#handle_details_post called on an instance of #{@agent.class} that does not define it."
  30. head 500
  31. end
  32. end
  33. def run
  34. @agent = current_user.agents.find(params[:id])
  35. Agent.async_check(@agent.id)
  36. respond_to do |format|
  37. format.html { redirect_back "Agent run queued for '#{@agent.name}'" }
  38. format.json { head :ok }
  39. end
  40. end
  41. def dry_run
  42. attrs = params[:agent] || {}
  43. if agent = current_user.agents.find_by(id: params[:id])
  44. # POST /agents/:id/dry_run
  45. if attrs.present?
  46. type = agent.type
  47. agent = Agent.build_for_type(type, current_user, attrs)
  48. end
  49. else
  50. # POST /agents/dry_run
  51. type = attrs.delete(:type)
  52. agent = Agent.build_for_type(type, current_user, attrs)
  53. end
  54. agent.name ||= '(Untitled)'
  55. if agent.valid?
  56. if event_payload = params[:event]
  57. dummy_agent = Agent.build_for_type('ManualEventAgent', current_user, name: 'Dry-Runner')
  58. dummy_agent.readonly!
  59. event = dummy_agent.events.build(user: current_user, payload: event_payload)
  60. end
  61. results = agent.dry_run!(event)
  62. render json: {
  63. log: results[:log],
  64. events: Utils.pretty_print(results[:events], false),
  65. memory: Utils.pretty_print(results[:memory] || {}, false),
  66. }
  67. else
  68. render json: {
  69. log: [
  70. "#{pluralize(agent.errors.count, "error")} prohibited this Agent from being saved:",
  71. *agent.errors.full_messages
  72. ].join("\n- "),
  73. events: '',
  74. memory: '',
  75. }
  76. end
  77. end
  78. def type_details
  79. @agent = Agent.build_for_type(params[:type], current_user, {})
  80. initialize_presenter
  81. render json: {
  82. can_be_scheduled: @agent.can_be_scheduled?,
  83. default_schedule: @agent.default_schedule,
  84. can_receive_events: @agent.can_receive_events?,
  85. can_create_events: @agent.can_create_events?,
  86. can_control_other_agents: @agent.can_control_other_agents?,
  87. can_dry_run: @agent.can_dry_run?,
  88. options: @agent.default_options,
  89. description_html: @agent.html_description,
  90. oauthable: render_to_string(partial: 'oauth_dropdown', locals: { agent: @agent }),
  91. form_options: render_to_string(partial: 'options', locals: { agent: @agent })
  92. }
  93. end
  94. def event_descriptions
  95. html = current_user.agents.find(params[:ids].split(",")).group_by(&:type).map { |type, agents|
  96. agents.map(&:html_event_description).uniq.map { |desc|
  97. "<p><strong>#{type}</strong><br />" + desc + "</p>"
  98. }
  99. }.flatten.join()
  100. render :json => { :description_html => html }
  101. end
  102. def remove_events
  103. @agent = current_user.agents.find(params[:id])
  104. @agent.events.delete_all
  105. respond_to do |format|
  106. format.html { redirect_back "All emitted events removed for '#{@agent.name}'" }
  107. format.json { head :ok }
  108. end
  109. end
  110. def propagate
  111. respond_to do |format|
  112. if AgentPropagateJob.can_enqueue?
  113. details = Agent.receive! # Eventually this should probably be scoped to the current_user.
  114. format.html { redirect_back "Queued propagation calls for #{details[:event_count]} event(s) on #{details[:agent_count]} agent(s)" }
  115. format.json { head :ok }
  116. else
  117. format.html { redirect_back "Event propagation is already scheduled to run." }
  118. format.json { head :locked }
  119. end
  120. end
  121. end
  122. def destroy_memory
  123. @agent = current_user.agents.find(params[:id])
  124. @agent.update!(memory: {})
  125. respond_to do |format|
  126. format.html { redirect_back "Memory erased for '#{@agent.name}'" }
  127. format.json { head :ok }
  128. end
  129. end
  130. def show
  131. @agent = current_user.agents.find(params[:id])
  132. respond_to do |format|
  133. format.html
  134. format.json { render json: @agent }
  135. end
  136. end
  137. def new
  138. agents = current_user.agents
  139. if id = params[:id]
  140. @agent = agents.build_clone(agents.find(id))
  141. else
  142. @agent = agents.build
  143. end
  144. @agent.scenario_ids = [params[:scenario_id]] if params[:scenario_id] && current_user.scenarios.find_by(id: params[:scenario_id])
  145. initialize_presenter
  146. respond_to do |format|
  147. format.html
  148. format.json { render json: @agent }
  149. end
  150. end
  151. def edit
  152. @agent = current_user.agents.find(params[:id])
  153. initialize_presenter
  154. end
  155. def create
  156. build_agent
  157. respond_to do |format|
  158. if @agent.save
  159. format.html { redirect_back "'#{@agent.name}' was successfully created.", return: agents_path }
  160. format.json { render json: @agent, status: :ok, location: agent_path(@agent) }
  161. else
  162. initialize_presenter
  163. format.html { render action: "new" }
  164. format.json { render json: @agent.errors, status: :unprocessable_entity }
  165. end
  166. end
  167. end
  168. def update
  169. @agent = current_user.agents.find(params[:id])
  170. respond_to do |format|
  171. if @agent.update_attributes(params[:agent])
  172. format.html { redirect_back "'#{@agent.name}' was successfully updated.", return: agents_path }
  173. format.json { render json: @agent, status: :ok, location: agent_path(@agent) }
  174. else
  175. initialize_presenter
  176. format.html { render action: "edit" }
  177. format.json { render json: @agent.errors, status: :unprocessable_entity }
  178. end
  179. end
  180. end
  181. def leave_scenario
  182. @agent = current_user.agents.find(params[:id])
  183. @scenario = current_user.scenarios.find(params[:scenario_id])
  184. @agent.scenarios.destroy(@scenario)
  185. respond_to do |format|
  186. format.html { redirect_back "'#{@agent.name}' removed from '#{@scenario.name}'" }
  187. format.json { head :no_content }
  188. end
  189. end
  190. def destroy
  191. @agent = current_user.agents.find(params[:id])
  192. @agent.destroy
  193. respond_to do |format|
  194. format.html { redirect_back "'#{@agent.name}' deleted" }
  195. format.json { head :no_content }
  196. end
  197. end
  198. def validate
  199. build_agent
  200. if @agent.validate_option(params[:attribute])
  201. render text: 'ok'
  202. else
  203. render text: 'error', status: 403
  204. end
  205. end
  206. def complete
  207. build_agent
  208. render json: @agent.complete_option(params[:attribute])
  209. end
  210. protected
  211. # Sanitize params[:return] to prevent open redirect attacks, a common security issue.
  212. def redirect_back(message, options = {})
  213. if path = filtered_agent_return_link(options)
  214. redirect_to path, notice: message
  215. else
  216. super agents_path, notice: message
  217. end
  218. end
  219. def build_agent
  220. @agent = Agent.build_for_type(params[:agent].delete(:type),
  221. current_user,
  222. params[:agent])
  223. end
  224. def initialize_presenter
  225. if @agent.present? && @agent.is_form_configurable?
  226. @agent = FormConfigurableAgentPresenter.new(@agent, view_context)
  227. end
  228. end
  229. private
  230. def show_only_enabled_agents?
  231. !!cookies[:huginn_view_only_enabled_agents]
  232. end
  233. def set_only_enabled_agents_as_viewable
  234. cookies[:huginn_view_only_enabled_agents] = {
  235. value: "true",
  236. expires: 1.year.from_now
  237. }
  238. end
  239. def mark_all_agents_viewable
  240. cookies.delete(:huginn_view_only_enabled_agents)
  241. end
  242. end