application_controller.rb 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. # Copyright (C) 2012-2014 Zammad Foundation, http://zammad-foundation.org/
  2. class ApplicationController < ActionController::Base
  3. # http_basic_authenticate_with :name => "test", :password => "ttt"
  4. helper_method :current_user,
  5. :authentication_check,
  6. :authentication_check_action_token,
  7. :config_frontend,
  8. :role?,
  9. :model_create_render,
  10. :model_update_render,
  11. :model_restory_render,
  12. :mode_show_rendeder,
  13. :model_index_render
  14. skip_before_action :verify_authenticity_token
  15. before_action :set_user, :session_update
  16. before_action :cors_preflight_check
  17. after_action :set_access_control_headers
  18. after_action :trigger_events
  19. # For all responses in this controller, return the CORS access control headers.
  20. def set_access_control_headers
  21. headers['Access-Control-Allow-Origin'] = '*'
  22. headers['Access-Control-Allow-Methods'] = 'POST, GET, PUT, DELETE, OPTIONS'
  23. headers['Access-Control-Max-Age'] = '1728000'
  24. headers['Access-Control-Allow-Headers'] = 'Content-Type, Depth, User-Agent, X-File-Size, X-Requested-With, If-Modified-Since, X-File-Name, Cache-Control, Accept-Language'
  25. headers['Access-Control-Allow-Credentials'] = 'true'
  26. end
  27. # If this is a preflight OPTIONS request, then short-circuit the
  28. # request, return only the necessary headers and return an empty
  29. # text/plain.
  30. def cors_preflight_check
  31. return if request.method != 'OPTIONS'
  32. headers['Access-Control-Allow-Origin'] = '*'
  33. headers['Access-Control-Allow-Methods'] = 'POST, GET, PUT, DELETE, OPTIONS'
  34. headers['Access-Control-Allow-Headers'] = 'Content-Type, Depth, User-Agent, X-File-Size, X-Requested-With, If-Modified-Since, X-File-Name, Cache-Control, Accept-Language'
  35. headers['Access-Control-Max-Age'] = '1728000'
  36. headers['Access-Control-Allow-Credentials'] = 'true'
  37. render text: '', content_type: 'text/plain'
  38. false
  39. end
  40. private
  41. # execute events
  42. def trigger_events
  43. Observer::Ticket::Notification.transaction
  44. end
  45. # Finds the User with the ID stored in the session with the key
  46. # :current_user_id This is a common way to handle user login in
  47. # a Rails application; logging in sets the session value and
  48. # logging out removes it.
  49. def current_user
  50. return @_current_user if @_current_user
  51. return if !session[:user_id]
  52. @_current_user = User.find( session[:user_id] )
  53. end
  54. def current_user_set(user)
  55. session[:user_id] = user.id
  56. @_current_user = user
  57. set_user
  58. end
  59. # Sets the current user into a named Thread location so that it can be accessed
  60. # by models and observers
  61. def set_user
  62. return if !current_user
  63. UserInfo.current_user_id = current_user.id
  64. end
  65. # update session updated_at
  66. def session_update
  67. #sleep 0.6
  68. # on many paralell requests, session got reinitialised if Time. is used, as workaround use DateTime.
  69. #session[:ping] = Time.now.utc.iso8601
  70. session[:ping] = DateTime.now.iso8601
  71. # check if remote ip need to be updated
  72. if !session[:remote_id] || session[:remote_id] != request.remote_ip
  73. session[:remote_id] = request.remote_ip
  74. session[:geo] = GeoIp.location( request.remote_ip )
  75. end
  76. # fill user agent
  77. return if session[:user_agent]
  78. session[:user_agent] = request.env['HTTP_USER_AGENT']
  79. end
  80. def authentication_check_only
  81. logger.debug 'authentication_check'
  82. session[:request_type] = 1
  83. #logger.debug params.inspect
  84. #logger.debug session.inspect
  85. #logger.debug cookies.inspect
  86. # check http basic auth
  87. authenticate_with_http_basic do |username, password|
  88. logger.debug 'http basic auth check'
  89. session[:request_type] = 2
  90. userdata = User.authenticate( username, password )
  91. message = ''
  92. if !userdata
  93. message = 'authentication failed'
  94. end
  95. # return auth ok
  96. if message == ''
  97. # remember user
  98. session[:user_id] = userdata.id
  99. # set basic auth user to current user
  100. current_user_set(userdata)
  101. return {
  102. auth: true
  103. }
  104. end
  105. # return auth not ok
  106. return {
  107. auth: false,
  108. message: message,
  109. }
  110. end
  111. # check logon session
  112. if params['logon_session']
  113. logon_session = ActiveRecord::SessionStore::Session.where( session_id: params['logon_session'] ).first
  114. if logon_session
  115. userdata = User.find( logon_session.data[:user_id] )
  116. end
  117. session[:request_type] = 3
  118. # set logon session user to current user
  119. current_user_set(userdata)
  120. return {
  121. auth: true
  122. }
  123. end
  124. # check sso
  125. if !session[:user_id]
  126. user = User.sso(params)
  127. # Log the authorizing user in.
  128. if user
  129. session[:user_id] = user.id
  130. end
  131. end
  132. # return auth not ok (no session exists)
  133. if !session[:user_id]
  134. logger.debug 'no valid session, user_id'
  135. message = 'no valid session, user_id'
  136. return {
  137. auth: false,
  138. message: message,
  139. }
  140. end
  141. {
  142. auth: true
  143. }
  144. end
  145. def authentication_check
  146. result = authentication_check_only
  147. # return auth not ok
  148. if result[:auth] == false
  149. render(
  150. json: {
  151. error: result[:message],
  152. },
  153. status: :unauthorized
  154. )
  155. return false
  156. end
  157. # return auth ok
  158. true
  159. end
  160. def authentication_check_action_token(action)
  161. user = Token.check(
  162. action: action,
  163. name: params[:action_token],
  164. )
  165. if !user
  166. logger.debug params.inspect
  167. response_access_deny
  168. return
  169. end
  170. current_user_set( user )
  171. true
  172. end
  173. def role?( role_name )
  174. return false if !current_user
  175. current_user.role?( role_name )
  176. end
  177. def ticket_permission(ticket)
  178. return true if ticket.permission( current_user: current_user )
  179. response_access_deny
  180. false
  181. end
  182. def deny_if_not_role( role_name )
  183. return false if role?( role_name )
  184. response_access_deny
  185. true
  186. end
  187. def valid_session_with_user
  188. return true if current_user
  189. render json: { message: 'No session user!' }, status: :unprocessable_entity
  190. false
  191. end
  192. def response_access_deny
  193. render(
  194. json: {},
  195. status: :unauthorized
  196. )
  197. false
  198. end
  199. def config_frontend
  200. # config
  201. config = {}
  202. Setting.select('name').where( frontend: true ).each { |setting|
  203. config[setting.name] = Setting.get(setting.name)
  204. }
  205. # get all time zones
  206. config['timezones'] = {}
  207. TZInfo::Timezone.all.each { |t|
  208. # ignore the following time zones
  209. next if t.name =~ /^GMT/
  210. next if t.name =~ /^Etc/
  211. next if t.name =~ /^MET/
  212. next if t.name =~ /^MST/
  213. next if t.name =~ /^ROC/
  214. next if t.name =~ /^ROK/
  215. diff = t.current_period.utc_total_offset / 60 / 60
  216. config['timezones'][ t.name ] = diff
  217. }
  218. if session[:switched_from_user_id]
  219. config['switch_back_to_possible'] = true
  220. end
  221. config
  222. end
  223. # model helper
  224. def model_create_render (object, params)
  225. # create object
  226. generic_object = object.new( object.param_cleanup( params[object.to_app_model_url], true ) )
  227. # save object
  228. generic_object.save!
  229. # set relations
  230. generic_object.param_set_associations( params )
  231. model_create_render_item(generic_object)
  232. rescue Exception => e
  233. logger.error e.message
  234. logger.error e.backtrace.inspect
  235. render json: { error: e.message }, status: :unprocessable_entity
  236. end
  237. def model_create_render_item (generic_object)
  238. render json: generic_object.attributes_with_associations, status: :created
  239. end
  240. def model_update_render (object, params)
  241. # find object
  242. generic_object = object.find( params[:id] )
  243. # save object
  244. generic_object.update_attributes!( object.param_cleanup( params[object.to_app_model_url] ) )
  245. # set relations
  246. generic_object.param_set_associations( params )
  247. model_update_render_item( generic_object )
  248. rescue Exception => e
  249. logger.error e.message
  250. logger.error e.backtrace.inspect
  251. render json: { error: e.message }, status: :unprocessable_entity
  252. end
  253. def model_update_render_item (generic_object)
  254. render json: generic_object.attributes_with_associations, status: :ok
  255. end
  256. def model_destory_render (object, params)
  257. generic_object = object.find( params[:id] )
  258. generic_object.destroy
  259. model_destory_render_item()
  260. rescue Exception => e
  261. logger.error e.message
  262. logger.error e.backtrace.inspect
  263. render json: { error: e.message }, status: :unprocessable_entity
  264. end
  265. def model_destory_render_item ()
  266. render json: {}, status: :ok
  267. end
  268. def model_show_render (object, params)
  269. if params[:full]
  270. generic_object_full = object.full( params[:id] )
  271. render json: generic_object_full, status: :ok
  272. return
  273. end
  274. generic_object = object.find( params[:id] )
  275. model_show_render_item(generic_object)
  276. rescue Exception => e
  277. logger.error e.message
  278. logger.error e.backtrace.inspect
  279. render json: { error: e.message }, status: :unprocessable_entity
  280. end
  281. def model_show_render_item (generic_object)
  282. render json: generic_object.attributes_with_associations, status: :ok
  283. end
  284. def model_index_render (object, _params)
  285. generic_objects = object.all
  286. model_index_render_result( generic_objects )
  287. rescue Exception => e
  288. logger.error e.message
  289. logger.error e.backtrace.inspect
  290. render json: { error: e.message }, status: :unprocessable_entity
  291. end
  292. def model_index_render_result (generic_objects)
  293. render json: generic_objects, status: :ok
  294. end
  295. end