You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

redmine.rb 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. # frozen_string_literal: true
  2. # Redmine - project management software
  3. # Copyright (C) 2006-2021 Jean-Philippe Lang
  4. #
  5. # This program is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU General Public License
  7. # as published by the Free Software Foundation; either version 2
  8. # of the License, or (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program; if not, write to the Free Software
  17. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  18. require 'redmine/core_ext'
  19. begin
  20. require 'mini_magick' unless Object.const_defined?(:MiniMagick)
  21. rescue LoadError
  22. # MiniMagick is not available
  23. end
  24. begin
  25. require 'redcarpet' unless Object.const_defined?(:Redcarpet)
  26. rescue LoadError
  27. # Redcarpet is not available
  28. end
  29. begin
  30. require 'commonmarker' unless Object.const_defined?(:CommonMarker)
  31. rescue LoadError
  32. # CommonMarker is not available
  33. end
  34. require 'redmine/acts/positioned'
  35. require 'redmine/scm/base'
  36. require 'redmine/access_control'
  37. require 'redmine/access_keys'
  38. require 'redmine/activity'
  39. require 'redmine/activity/fetcher'
  40. require 'redmine/ciphering'
  41. require 'redmine/codeset_util'
  42. require 'redmine/field_format'
  43. require 'redmine/info'
  44. require 'redmine/menu_manager'
  45. require 'redmine/notifiable'
  46. require 'redmine/platform'
  47. require 'redmine/project_jump_box'
  48. require 'redmine/mime_type'
  49. require 'redmine/search'
  50. require 'redmine/sort_criteria'
  51. require 'redmine/syntax_highlighting'
  52. require 'redmine/thumbnail'
  53. require 'redmine/unified_diff'
  54. require 'redmine/utils'
  55. require 'redmine/version'
  56. require 'redmine/wiki_formatting/links_helper'
  57. require 'redmine/wiki_formatting'
  58. require 'redmine/default_data/loader'
  59. require 'redmine/helpers/calendar'
  60. require 'redmine/helpers/diff'
  61. require 'redmine/helpers/gantt'
  62. require 'redmine/helpers/time_report'
  63. require 'redmine/views/other_formats_builder'
  64. require 'redmine/views/labelled_form_builder'
  65. require 'redmine/views/builders'
  66. require 'redmine/themes'
  67. require 'redmine/hook'
  68. require 'redmine/hook/listener'
  69. require 'redmine/hook/view_listener'
  70. require 'redmine/plugin'
  71. require 'redmine/twofa'
  72. Redmine::Scm::Base.add "Subversion"
  73. Redmine::Scm::Base.add "Mercurial"
  74. Redmine::Scm::Base.add "Cvs"
  75. Redmine::Scm::Base.add "Bazaar"
  76. Redmine::Scm::Base.add "Git"
  77. Redmine::Scm::Base.add "Filesystem"
  78. # Permissions
  79. Redmine::AccessControl.map do |map|
  80. map.permission :view_project, {:projects => [:show, :bookmark], :activities => [:index]}, :public => true, :read => true
  81. map.permission :search_project, {:search => :index}, :public => true, :read => true
  82. map.permission :add_project, {:projects => [:new, :create]}, :require => :loggedin
  83. map.permission :edit_project, {:projects => [:settings, :edit, :update]}, :require => :member
  84. map.permission :close_project, {:projects => [:close, :reopen]}, :require => :member, :read => true
  85. map.permission :delete_project, {:projects => :destroy}, :require => :member, :read => true
  86. map.permission :select_project_modules, {:projects => :modules}, :require => :member
  87. map.permission :view_members, {:members => [:index, :show]}, :public => true, :read => true
  88. map.permission :manage_members, {:projects => :settings, :members => [:index, :show, :new, :create, :edit, :update, :destroy, :autocomplete]}, :require => :member
  89. map.permission :manage_versions, {:projects => :settings, :versions => [:new, :create, :edit, :update, :close_completed, :destroy]}, :require => :member
  90. map.permission :add_subprojects, {:projects => [:new, :create]}, :require => :member
  91. # Queries
  92. map.permission :manage_public_queries, {:queries => [:new, :create, :edit, :update, :destroy]}, :require => :member
  93. map.permission :save_queries, {:queries => [:new, :create, :edit, :update, :destroy]}, :require => :loggedin
  94. map.project_module :issue_tracking do |map|
  95. # Issues
  96. map.permission :view_issues, {:issues => [:index, :show, :issue_tab],
  97. :auto_complete => [:issues],
  98. :context_menus => [:issues],
  99. :versions => [:index, :show, :status_by],
  100. :journals => [:index, :diff],
  101. :queries => :index,
  102. :reports => [:issue_report, :issue_report_details]},
  103. :read => true
  104. map.permission :add_issues, {:issues => [:new, :create], :attachments => :upload}
  105. map.permission :edit_issues, {:issues => [:edit, :update, :bulk_edit, :bulk_update], :journals => [:new], :attachments => :upload}
  106. map.permission :edit_own_issues, {:issues => [:edit, :update, :bulk_edit, :bulk_update], :journals => [:new], :attachments => :upload}
  107. map.permission :copy_issues, {:issues => [:new, :create, :bulk_edit, :bulk_update], :attachments => :upload}
  108. map.permission :manage_issue_relations, {:issue_relations => [:index, :show, :create, :destroy]}
  109. map.permission :manage_subtasks, {}
  110. map.permission :set_issues_private, {}
  111. map.permission :set_own_issues_private, {}, :require => :loggedin
  112. map.permission :add_issue_notes, {:issues => [:edit, :update], :journals => [:new], :attachments => :upload}
  113. map.permission :edit_issue_notes, {:journals => [:edit, :update]}, :require => :loggedin
  114. map.permission :edit_own_issue_notes, {:journals => [:edit, :update]}, :require => :loggedin
  115. map.permission :view_private_notes, {}, :read => true, :require => :member
  116. map.permission :set_notes_private, {}, :require => :member
  117. map.permission :delete_issues, {:issues => :destroy}, :require => :member
  118. # Watchers
  119. map.permission :view_issue_watchers, {}, :read => true
  120. map.permission :add_issue_watchers, {:watchers => [:new, :create, :append, :autocomplete_for_user]}
  121. map.permission :delete_issue_watchers, {:watchers => :destroy}
  122. map.permission :import_issues, {}
  123. # Issue categories
  124. map.permission :manage_categories, {:projects => :settings, :issue_categories => [:index, :show, :new, :create, :edit, :update, :destroy]}, :require => :member
  125. end
  126. map.project_module :time_tracking do |map|
  127. map.permission :view_time_entries, {:timelog => [:index, :report, :show]}, :read => true
  128. map.permission :log_time, {:timelog => [:new, :create]}, :require => :loggedin
  129. map.permission :edit_time_entries,
  130. {:timelog => [:edit, :update, :destroy, :bulk_edit, :bulk_update]},
  131. :require => :member
  132. map.permission :edit_own_time_entries,
  133. {:timelog => [:edit, :update, :destroy, :bulk_edit, :bulk_update]},
  134. :require => :loggedin
  135. map.permission :manage_project_activities,
  136. {:projects => :settings, :project_enumerations => [:update, :destroy]},
  137. :require => :member
  138. map.permission :log_time_for_other_users, :require => :member
  139. map.permission :import_time_entries, {}
  140. end
  141. map.project_module :news do |map|
  142. map.permission :view_news, {:news => [:index, :show]}, :read => true
  143. map.permission :manage_news, {:news => [:new, :create, :edit, :update, :destroy], :comments => [:destroy], :attachments => :upload}, :require => :member
  144. map.permission :comment_news, {:comments => :create}
  145. end
  146. map.project_module :documents do |map|
  147. map.permission :view_documents, {:documents => [:index, :show, :download]}, :read => true
  148. map.permission :add_documents, {:documents => [:new, :create, :add_attachment], :attachments => :upload}, :require => :loggedin
  149. map.permission :edit_documents, {:documents => [:edit, :update, :add_attachment], :attachments => :upload}, :require => :loggedin
  150. map.permission :delete_documents, {:documents => [:destroy]}, :require => :loggedin
  151. end
  152. map.project_module :files do |map|
  153. map.permission :view_files, {:files => :index, :versions => :download}, :read => true
  154. map.permission :manage_files, {:files => [:new, :create], :attachments => :upload}, :require => :loggedin
  155. end
  156. map.project_module :wiki do |map|
  157. map.permission :view_wiki_pages, {:wiki => [:index, :show, :special, :date_index], :auto_complete => [:wiki_pages]}, :read => true
  158. map.permission :view_wiki_edits, {:wiki => [:history, :diff, :annotate]}, :read => true
  159. map.permission :export_wiki_pages, {:wiki => [:export]}, :read => true
  160. map.permission :edit_wiki_pages, :wiki => [:new, :edit, :update, :preview, :add_attachment], :attachments => :upload
  161. map.permission :rename_wiki_pages, {:wiki => :rename}, :require => :member
  162. map.permission :delete_wiki_pages, {:wiki => [:destroy, :destroy_version]}, :require => :member
  163. map.permission :delete_wiki_pages_attachments, {}
  164. map.permission :view_wiki_page_watchers, {}, :read => true
  165. map.permission :add_wiki_page_watchers, {:watchers => [:new, :create, :autocomplete_for_user]}
  166. map.permission :delete_wiki_page_watchers, {:watchers => :destroy}
  167. map.permission :protect_wiki_pages, {:wiki => :protect}, :require => :member
  168. map.permission :manage_wiki, {:wikis => :destroy, :wiki => :rename}, :require => :member
  169. end
  170. map.project_module :repository do |map|
  171. map.permission :view_changesets, {:repositories => [:show, :revisions, :revision]}, :read => true
  172. map.permission :browse_repository, {:repositories => [:show, :browse, :entry, :raw, :annotate, :changes, :diff, :stats, :graph]}, :read => true
  173. map.permission :commit_access, {}
  174. map.permission :manage_related_issues, {:repositories => [:add_related_issue, :remove_related_issue]}
  175. map.permission :manage_repository, {:projects => :settings, :repositories => [:new, :create, :edit, :update, :committers, :destroy, :fetch_changesets]}, :require => :member
  176. end
  177. map.project_module :boards do |map|
  178. map.permission :view_messages, {:boards => [:index, :show], :messages => [:show]}, :read => true
  179. map.permission :add_messages, {:messages => [:new, :reply, :quote], :attachments => :upload}
  180. map.permission :edit_messages, {:messages => :edit, :attachments => :upload}, :require => :member
  181. map.permission :edit_own_messages, {:messages => :edit, :attachments => :upload}, :require => :loggedin
  182. map.permission :delete_messages, {:messages => :destroy}, :require => :member
  183. map.permission :delete_own_messages, {:messages => :destroy}, :require => :loggedin
  184. map.permission :view_message_watchers, {}, :read => true
  185. map.permission :add_message_watchers, {:watchers => [:new, :create, :autocomplete_for_user]}
  186. map.permission :delete_message_watchers, {:watchers => :destroy}
  187. map.permission :manage_boards, {:projects => :settings, :boards => [:new, :create, :edit, :update, :destroy]}, :require => :member
  188. end
  189. map.project_module :calendar do |map|
  190. map.permission :view_calendar, {:calendars => [:show, :update]}, :read => true
  191. end
  192. map.project_module :gantt do |map|
  193. map.permission :view_gantt, {:gantts => [:show, :update]}, :read => true
  194. end
  195. end
  196. Redmine::MenuManager.map :top_menu do |menu|
  197. menu.push :home, :home_path
  198. menu.push :my_page, {:controller => 'my', :action => 'page'},
  199. :if => Proc.new {User.current.logged?}
  200. menu.push :projects, {:controller => 'projects', :action => 'index'},
  201. :caption => :label_project_plural
  202. menu.push :administration, {:controller => 'admin', :action => 'index'},
  203. :if => Proc.new {User.current.admin?}, :last => true
  204. menu.push :help, Redmine::Info.help_url, :last => true
  205. end
  206. Redmine::MenuManager.map :account_menu do |menu|
  207. menu.push :login, :signin_path, :if => Proc.new {!User.current.logged?}
  208. menu.push :register, :register_path,
  209. :if => Proc.new {!User.current.logged? && Setting.self_registration?}
  210. menu.push :my_account, {:controller => 'my', :action => 'account'},
  211. :if => Proc.new {User.current.logged?}
  212. menu.push :logout, :signout_path, :html => {:method => 'post'},
  213. :if => Proc.new {User.current.logged?}
  214. end
  215. Redmine::MenuManager.map :application_menu do |menu|
  216. menu.push :projects, {:controller => 'projects', :action => 'index'},
  217. :permission => nil,
  218. :caption => :label_project_plural
  219. menu.push :activity, {:controller => 'activities', :action => 'index', :id => nil}
  220. menu.push(
  221. :issues,
  222. {:controller => 'issues', :action => 'index'},
  223. :if =>
  224. Proc.new do
  225. User.current.allowed_to?(:view_issues, nil, :global => true) &&
  226. EnabledModule.exists?(:project => Project.visible, :name => :issue_tracking)
  227. end,
  228. :caption => :label_issue_plural
  229. )
  230. menu.push(
  231. :time_entries,
  232. {:controller => 'timelog', :action => 'index'},
  233. :if =>
  234. Proc.new do
  235. User.current.allowed_to?(:view_time_entries, nil, :global => true) &&
  236. EnabledModule.exists?(:project => Project.visible, :name => :time_tracking)
  237. end,
  238. :caption => :label_spent_time
  239. )
  240. menu.push(
  241. :gantt,
  242. {:controller => 'gantts', :action => 'show'},
  243. :caption => :label_gantt,
  244. :if =>
  245. Proc.new do
  246. User.current.allowed_to?(:view_gantt, nil, :global => true) &&
  247. EnabledModule.exists?(:project => Project.visible, :name => :gantt)
  248. end
  249. )
  250. menu.push(
  251. :calendar,
  252. {:controller => 'calendars', :action => 'show'},
  253. :caption => :label_calendar,
  254. :if =>
  255. Proc.new do
  256. User.current.allowed_to?(:view_calendar, nil, :global => true) &&
  257. EnabledModule.exists?(:project => Project.visible, :name => :calendar)
  258. end
  259. )
  260. menu.push(
  261. :news,
  262. {:controller => 'news', :action => 'index'},
  263. :if =>
  264. Proc.new do
  265. User.current.allowed_to?(:view_news, nil, :global => true) &&
  266. EnabledModule.exists?(:project => Project.visible, :name => :news)
  267. end,
  268. :caption => :label_news_plural
  269. )
  270. end
  271. Redmine::MenuManager.map :admin_menu do |menu|
  272. menu.push :projects, {:controller => 'admin', :action => 'projects'},
  273. :caption => :label_project_plural,
  274. :html => {:class => 'icon icon-projects'}
  275. menu.push :users, {:controller => 'users'}, :caption => :label_user_plural,
  276. :html => {:class => 'icon icon-user'}
  277. menu.push :groups, {:controller => 'groups'}, :caption => :label_group_plural,
  278. :html => {:class => 'icon icon-group'}
  279. menu.push :roles, {:controller => 'roles'},
  280. :caption => :label_role_and_permissions,
  281. :html => {:class => 'icon icon-roles'}
  282. menu.push :trackers, {:controller => 'trackers'},
  283. :caption => :label_tracker_plural,
  284. :html => {:class => 'icon icon-issue'}
  285. menu.push :issue_statuses, {:controller => 'issue_statuses'},
  286. :caption => :label_issue_status_plural,
  287. :html => {:class => 'icon icon-issue-edit'}
  288. menu.push :workflows, {:controller => 'workflows', :action => 'edit'},
  289. :caption => :label_workflow,
  290. :html => {:class => 'icon icon-workflows'}
  291. menu.push :custom_fields, {:controller => 'custom_fields'},
  292. :caption => :label_custom_field_plural,
  293. :html => {:class => 'icon icon-custom-fields'}
  294. menu.push :enumerations, {:controller => 'enumerations'},
  295. :html => {:class => 'icon icon-list'}
  296. menu.push :settings, {:controller => 'settings'},
  297. :html => {:class => 'icon icon-settings'}
  298. menu.push :ldap_authentication,
  299. {:controller => 'auth_sources', :action => 'index'},
  300. :html => {:class => 'icon icon-server-authentication'}
  301. menu.push :plugins, {:controller => 'admin', :action => 'plugins'},
  302. :last => true,
  303. :html => {:class => 'icon icon-plugins'}
  304. menu.push :info, {:controller => 'admin', :action => 'info'},
  305. :caption => :label_information_plural,
  306. :last => true,
  307. :html => {:class => 'icon icon-help'}
  308. end
  309. Redmine::MenuManager.map :project_menu do |menu|
  310. menu.push(
  311. :new_object, nil, :caption => ' + ',
  312. :if => Proc.new {|p| Setting.new_item_menu_tab == '2'},
  313. :html => {:id => 'new-object',
  314. :onclick => 'toggleNewObjectDropdown(); return false;'}
  315. )
  316. menu.push(
  317. :new_issue_sub,
  318. {:controller => 'issues', :action => 'new', :copy_from => nil},
  319. :param => :project_id, :caption => :label_issue_new,
  320. :html => {:accesskey => Redmine::AccessKeys.key_for(:new_issue)},
  321. :if => Proc.new {|p| Issue.allowed_target_trackers(p).any?},
  322. :permission => :add_issues,
  323. :parent => :new_object
  324. )
  325. menu.push :new_issue_category,
  326. {:controller => 'issue_categories', :action => 'new'},
  327. :param => :project_id, :caption => :label_issue_category_new,
  328. :parent => :new_object
  329. menu.push :new_version, {:controller => 'versions', :action => 'new'},
  330. :param => :project_id, :caption => :label_version_new,
  331. :parent => :new_object
  332. menu.push :new_timelog, {:controller => 'timelog', :action => 'new'},
  333. :param => :project_id, :caption => :button_log_time,
  334. :parent => :new_object
  335. menu.push :new_news, {:controller => 'news', :action => 'new'},
  336. :param => :project_id, :caption => :label_news_new,
  337. :parent => :new_object
  338. menu.push :new_document, {:controller => 'documents', :action => 'new'},
  339. :param => :project_id, :caption => :label_document_new,
  340. :parent => :new_object
  341. menu.push :new_wiki_page, {:controller => 'wiki', :action => 'new'},
  342. :param => :project_id, :caption => :label_wiki_page_new,
  343. :parent => :new_object
  344. menu.push :new_file, {:controller => 'files', :action => 'new'},
  345. :param => :project_id, :caption => :label_attachment_new,
  346. :parent => :new_object
  347. menu.push :overview, {:controller => 'projects', :action => 'show'}
  348. menu.push :activity, {:controller => 'activities', :action => 'index'}
  349. menu.push(
  350. :roadmap,
  351. {:controller => 'versions', :action => 'index'},
  352. :param => :project_id,
  353. :if =>
  354. Proc.new do |p|
  355. if p.shared_versions.any?
  356. true
  357. else
  358. Setting.display_subprojects_issues? && p.rolled_up_versions.any?
  359. end
  360. end
  361. )
  362. menu.push :issues, {:controller => 'issues', :action => 'index'},
  363. :param => :project_id, :caption => :label_issue_plural
  364. menu.push(
  365. :new_issue,
  366. {:controller => 'issues', :action => 'new', :copy_from => nil},
  367. :param => :project_id, :caption => :label_issue_new,
  368. :html => {:accesskey => Redmine::AccessKeys.key_for(:new_issue)},
  369. :if =>
  370. Proc.new do |p|
  371. Setting.new_item_menu_tab == '1' &&
  372. Issue.allowed_target_trackers(p).any?
  373. end,
  374. :permission => :add_issues
  375. )
  376. menu.push :time_entries, {:controller => 'timelog', :action => 'index'},
  377. :param => :project_id, :caption => :label_spent_time
  378. menu.push :gantt, {:controller => 'gantts', :action => 'show'},
  379. :param => :project_id, :caption => :label_gantt
  380. menu.push :calendar, {:controller => 'calendars', :action => 'show'},
  381. :param => :project_id, :caption => :label_calendar
  382. menu.push :news, {:controller => 'news', :action => 'index'},
  383. :param => :project_id, :caption => :label_news_plural
  384. menu.push :documents, {:controller => 'documents', :action => 'index'},
  385. :param => :project_id, :caption => :label_document_plural
  386. menu.push :wiki, {:controller => 'wiki', :action => 'show', :id => nil},
  387. :param => :project_id,
  388. :if => Proc.new {|p| p.wiki && !p.wiki.new_record?}
  389. menu.push :boards, {:controller => 'boards', :action => 'index', :id => nil},
  390. :param => :project_id,
  391. :if => Proc.new {|p| p.boards.any?}, :caption => :label_board_plural
  392. menu.push :files, {:controller => 'files', :action => 'index'},
  393. :caption => :label_file_plural, :param => :project_id
  394. menu.push :repository,
  395. {:controller => 'repositories', :action => 'show',
  396. :repository_id => nil, :path => nil, :rev => nil},
  397. :if => Proc.new {|p| p.repository && !p.repository.new_record?}
  398. menu.push :settings, {:controller => 'projects', :action => 'settings'},
  399. :last => true
  400. end
  401. Redmine::Activity.map do |activity|
  402. activity.register :issues, :class_name => %w(Issue Journal)
  403. activity.register :changesets
  404. activity.register :news
  405. activity.register :documents, :class_name => %w(Document Attachment)
  406. activity.register :files, :class_name => 'Attachment'
  407. activity.register :wiki_edits, :class_name => 'WikiContentVersion', :default => false
  408. activity.register :messages, :default => false
  409. activity.register :time_entries, :default => false
  410. end
  411. Redmine::Search.map do |search|
  412. search.register :issues
  413. search.register :news
  414. search.register :documents
  415. search.register :changesets
  416. search.register :wiki_pages
  417. search.register :messages
  418. search.register :projects
  419. end
  420. Redmine::WikiFormatting.map do |format|
  421. format.register :textile
  422. format.register :markdown if Object.const_defined?(:Redcarpet)
  423. if Object.const_defined?(:CommonMarker)
  424. format.register :common_mark, label: 'CommonMark Markdown (GitHub Flavored) - experimental'
  425. end
  426. end
  427. ActionView::Template.register_template_handler :rsb, Redmine::Views::ApiTemplateHandler