Compare commits

...

393 Commits

Author SHA1 Message Date
Naoki Takezoe
f4865adecf Preparation of GitBucket 4.27.0 release 2018-07-28 14:01:17 +09:00
Naoki Takezoe
33cf58078e Merge pull request #2122 from jyuch/fix-database-export-error
Fix database export failure
2018-07-27 01:37:55 +09:00
jyuch
ec5d8560d8 (refs #2120) Fix database export failure 2018-07-26 21:33:30 +09:00
Naoki Takezoe
a6f6303bfa Merge pull request #2117 from uli-heller/postgresql-42.2.4
postgresql: 42.1.4 -> 42.2.4
2018-07-25 23:54:13 +09:00
Naoki Takezoe
377376d457 (Refs #2082)Fix to export orphan tables as well 2018-07-23 18:23:03 +09:00
Uli Heller
103800f911 postgresql: 42.1.4 -> 42.2.4 2018-07-23 11:13:20 +02:00
Naoki Takezoe
9c46be519e Merge branch 'kounoike-pr-create-tag' 2018-07-23 13:39:59 +09:00
Naoki Takezoe
a5e130db0b Change the create tag form to a dialog 2018-07-23 13:32:46 +09:00
Naoki Takezoe
f360a3ba9b Merge branch 'pr-create-tag' of https://github.com/kounoike/gitbucket into kounoike-pr-create-tag 2018-07-23 11:25:35 +09:00
Naoki Takezoe
251731e41a Merge pull request #2113 from uli-heller/mariadb-2.2.6
mariadb: 2.2.5 -> 2.2.6
2018-07-23 10:58:42 +09:00
Naoki Takezoe
f0129a3d4d Merge pull request #2115 from gitbucket/fix-file-remove
BugFix for the file remove page
2018-07-23 10:58:07 +09:00
Naoki Takezoe
78fbeb67d4 (Closes #2111)Fix comment toggle link on Firefox 2018-07-23 10:57:23 +09:00
Naoki Takezoe
bb59cbcb91 Fix the file remove page
- Cancel button didn’t work
- Redirection after remove didn’t work
- Diff wasn’t displayed at the remove confirmation page
2018-07-23 02:24:45 +09:00
Naoki Takezoe
9769041ad1 Merge pull request #2114 from gitbucket/fix-pullreq-comments
Keep pull request comment diff after new commits are pushed
2018-07-23 01:07:24 +09:00
Naoki Takezoe
8948c05080 Keep original commitId and lines of pull request comments 2018-07-22 23:03:29 +09:00
Uli Heller
3a9f67f862 mariadb: 2.2.5 -> 2.2.6 2018-07-21 18:13:57 +02:00
Naoki Takezoe
e179c8c56a Merge pull request #2110 from gitbucket/improve-repo-issues-search
Separate issues and pull requests search
2018-07-19 21:36:04 +09:00
Naoki Takezoe
403d5afedc Display "Closed" label on the search result of issues and pull requests 2018-07-19 21:13:13 +09:00
Naoki Takezoe
cb5a5b7b6f Fix search type parameter 2018-07-19 20:02:56 +09:00
Naoki Takezoe
aac232f33e Separate issues and pull requests search 2018-07-18 18:39:08 +09:00
Naoki Takezoe
55c973b760 (refs #2097)Fix mouse cursor 2018-07-18 17:18:10 +09:00
Naoki Takezoe
39a895cdc6 Merge pull request #2108 from gitbucket/sunjdk-ssl-provider
Create SunJDK's SSL Provider by reflection
2018-07-18 14:49:26 +09:00
Naoki Takezoe
d4a9a2b2ee Format 2018-07-18 13:44:59 +09:00
Naoki Takezoe
a02f020626 Create SunJDK's SSL Provider by reflection 2018-07-18 10:12:15 +09:00
Naoki Takezoe
0366f2f2ed Output warn log when accessing to the plugin repository failed 2018-07-17 00:43:33 +09:00
Naoki Takezoe
b6d5e34980 Merge pull request #2106 from x-way/catch-network-plugin-exceptions
Catch network exceptions (UnknownHostException etc.)
2018-07-17 00:38:06 +09:00
Naoki Takezoe
33a676f221 Fix downloading repository archive 2018-07-17 00:33:17 +09:00
Naoki Takezoe
3f6ca48f26 Merge pull request #2104 from gitbucket/fix-archive-slash-branch
Fix archiving for slashed branches
2018-07-16 11:49:42 +09:00
Andreas Jaggi
79aa55f741 Catch network exceptions (UnknownHostException etc.) 2018-07-15 08:09:31 +02:00
Naoki Takezoe
3d060ae82f Fix archiving for slashed branches 2018-07-15 10:39:46 +09:00
Naoki Takezoe
18a0c6c92a Fix plugin manager behavior 2018-07-09 16:46:48 +09:00
Naoki Takezoe
a92aae9544 Fix plugin manager behavior 2018-07-09 12:23:16 +09:00
Naoki Takezoe
87e301dd38 (refs #2092)Exclude lower versions than installed one from the list of upgradable plugins. 2018-07-09 11:47:38 +09:00
Naoki Takezoe
839bd6634f Merge pull request #2098 from tiqwab/remove-semicolon
Remove unnecessary semicolon
2018-07-09 10:59:48 +09:00
tiqwab
4512da7c03 Remove unnecessary semicolon 2018-07-08 12:03:58 +09:00
Naoki Takezoe
5e92815f96 (refs #2089)Fix the list of repositories on the sidebar 2018-07-05 12:07:28 +09:00
Naoki Takezoe
f5e2e5a0aa Merge pull request #2093 from gitbucket/improve-plugin-version-handling
Improve plugin version handling
2018-07-05 01:55:06 +09:00
Naoki Takezoe
61261dcb4e Improve plugin version handling
- Allow to include "-SNAPSHOT" (both plugin and gitbucket version)
- Allow to omit patch version like "x.y"
2018-07-04 14:42:46 +09:00
Naoki Takezoe
7ced7795ff Merge pull request #1966 from kounoike/pr-editorconfig-for-repository
Support EditorConfig for online browser/editor.
2018-07-01 22:46:53 +09:00
KOUNOIKE Yuusuke
d553771335 implement JGitResource/JGitResourcePath class with scala 2018-07-01 13:16:48 +09:00
KOUNOIKE Yuusuke
fd6b658565 Merge remote-tracking branch 'upstream/master' into pr-editorconfig-for-repository 2018-07-01 12:04:38 +09:00
Naoki Takezoe
f01c65d022 Remove document about JRebel 2018-07-01 02:40:07 +09:00
Naoki Takezoe
29d2014053 Merge pull request #2083 from gitbucket/drop-jrebel-config
Drop JRebel configuration
2018-07-01 02:38:17 +09:00
Naoki Takezoe
73681d7647 Merge pull request #2087 from xuwei-k/emojji
fix typo
2018-07-01 01:33:51 +09:00
Naoki Takezoe
79e5fb5dd8 Merge pull request #2086 from xuwei-k/foreach
use foreach instead of map
2018-07-01 01:33:02 +09:00
xuwei-k
563c69b4ef fix typo 2018-06-30 16:14:49 +09:00
xuwei-k
670d23c3c6 use foreach instead of map 2018-06-30 16:03:51 +09:00
Naoki Takezoe
3466097ab1 Update changelog 2018-06-30 12:54:06 +09:00
Naoki Takezoe
f29350a986 Upgrade bundled plugins 2018-06-30 12:49:34 +09:00
Naoki Takezoe
9f21b7775d Update release operation 2018-06-30 12:35:53 +09:00
Naoki Takezoe
a43aeff3fc 4.26.0 release 2018-06-30 12:06:17 +09:00
Naoki Takezoe
0e36ca0f71 Merge pull request #2084 from gitbucket/github-compatible-url-lfs
Add forwarding GitHub compatible LFS url
2018-06-30 01:45:48 +09:00
Naoki Takezoe
3de7a3b525 Update version to 4.26.0-SNAPSHOT 2018-06-29 17:47:43 +09:00
Naoki Takezoe
5d8f1a7678 Add forwarding GitHub compatible LFS url 2018-06-29 17:40:12 +09:00
Naoki Takezoe
809361d2e1 Merge remote-tracking branch 'origin/master' 2018-06-29 16:36:05 +09:00
Naoki Takezoe
427a372e94 Additional fix for #2005 2018-06-29 16:35:41 +09:00
Naoki Takezoe
f2759c6d7c Drop JRebel configuration 2018-06-29 14:44:49 +09:00
Naoki Takezoe
626ff932cd Merge pull request #2081 from uli-heller/jgit-5.0.1.201806211838-r
jgit: 5.0.0.201806131550-r -> 5.0.1.201806211838-r
2018-06-28 11:02:24 +09:00
Naoki Takezoe
328403f973 Merge pull request #2065 from gitbucket/plugin-network-install
Install plugins from the plugin registry
2018-06-27 11:55:54 +09:00
Uli Heller
55ae324878 jgit: 5.0.0.201806131550-r -> 5.0.1.201806211838-r
Bug Fixes
Respect unshallow lines in protocol v2
Fix maven site generation failing with javadoc errors
2018-06-26 07:04:19 +02:00
Naoki Takezoe
d322f772e8 Merge pull request #2005 from gitbucket/fork-button
Move Fork button to the header
2018-06-25 19:54:37 +09:00
Naoki Takezoe
e578f9548b Add a button to show forked repos to the fork account select dialog 2018-06-25 19:42:57 +09:00
Naoki Takezoe
2678c6939d Merge branch 'master' into plugin-network-install 2018-06-25 13:19:05 +09:00
Naoki Takezoe
9830392cb9 Remove unnecessary CSS style 2018-06-25 02:56:01 +09:00
Naoki Takezoe
bd689aa3c6 Merge pull request #2077 from uli-heller/jgit-5.0.0.201806131550-r
jgit: 5.0.0.201805301535-rc2 -> 5.0.0.201806131550-r
2018-06-24 20:31:41 +09:00
Naoki Takezoe
960b7834a7 Reload diff when display mode is changed 2018-06-24 14:57:12 +09:00
Naoki Takezoe
b5b37a1168 (refs #2049)Fix comment top diff handling 2018-06-24 01:01:55 +09:00
Naoki Takezoe
49f71a5ceb (refs #2057)Fix duplicate event handler registration in commit comments 2018-06-23 15:18:02 +09:00
Uli Heller
3febb27b1d jgit: 5.0.0.201805301535-rc2 -> 5.0.0.201806131550-r 2018-06-22 08:04:09 +02:00
Naoki Takezoe
79a7df254e Merge branch 'master' into fork-button 2018-06-20 18:15:54 +09:00
Naoki Takezoe
163fcd86e0 Merge pull request #2074 from gitbucket/dashboard-repo-list
Add "Repositories" tab to the dashboard
2018-06-20 03:35:28 +09:00
Naoki Takezoe
f8f1c7442e Add repsitory owner filter 2018-06-20 03:25:55 +09:00
Naoki Takezoe
91c885954a Fix sidebar to recent repositories 2018-06-20 02:35:34 +09:00
Naoki Takezoe
76d04e04f9 Merge pull request #2075 from gitbucket/improve-fork-popup
Fork account selector dialog to be list style
2018-06-19 21:54:35 +09:00
Naoki Takezoe
e77570efd4 Update the fork account selector dialog to list style 2018-06-19 18:52:00 +09:00
Naoki Takezoe
7be09563a2 Add "Repositories" tab to the dashboard 2018-06-19 16:55:48 +09:00
Naoki Takezoe
acb7a34f3f (refs #2073)Fix error in API when getting first commit 2018-06-19 14:53:36 +09:00
Naoki Takezoe
6e4a203f81 Hide fork button if its option is disabled 2018-06-18 16:02:12 +09:00
Naoki Takezoe
5f00b8354f Merge pull request #2067 from gitbucket/adjust-pullrequest-alert
Adjust condition to display pull request creation suggestion for branches
2018-06-13 22:43:49 +09:00
Naoki Takezoe
6029d9ef29 Adjust condition to display pull request creation suggestion for branches 2018-06-13 17:48:10 +09:00
Naoki Takezoe
db1d437a93 Fix style of some text fields 2018-06-13 15:39:50 +09:00
Naoki Takezoe
5de893f5d3 Merge pull request #2052 from uli-heller/jgit5
Upgraded to jgit-5.0-rc2
2018-06-13 15:32:41 +09:00
Naoki Takezoe
11d73816af Merge pull request #2047 from sraabe/patch-1
Added mime-type mapping for SVG files
2018-06-13 15:16:11 +09:00
Naoki Takezoe
4106276616 Merge pull request #2066 from gitbucket/render-html-as-text
Render HTML as text/plain
2018-06-13 10:51:32 +09:00
Naoki Takezoe
f1ffb25e43 Render HTML as text/plain 2018-06-12 20:02:49 +09:00
Naoki Takezoe
3c1bc32e5e Fix format 2018-06-12 17:06:48 +09:00
Naoki Takezoe
c93bbfca53 Fix testcase 2018-06-12 15:00:20 +09:00
Naoki Takezoe
033c3833d6 Fixup 2018-06-12 14:55:04 +09:00
Naoki Takezoe
151ecb956a Remove unused imports 2018-06-12 13:19:22 +09:00
Naoki Takezoe
8705793735 Bundle plugins 2018-06-11 20:37:04 +09:00
Naoki Takezoe
99f228bb94 Remove unused variables 2018-06-11 19:18:26 +09:00
Naoki Takezoe
a4cebcc3ac Turn on / off network plugin installation 2018-06-11 14:38:30 +09:00
Naoki Takezoe
ed7bb495ca Enhance plugin install / uninstall process 2018-06-11 13:36:53 +09:00
Naoki Takezoe
2586b436cf Merge pull request #2042 from kazuki43zoo/keep-showing-fold-comment
Keep showing a fold comment when tasklist is not complete
2018-06-11 09:09:51 +09:00
Naoki Takezoe
abeef42a53 Merge pull request #2037 from kazuki43zoo/focus-on-comment-area
Focus to textarea at add and edit a comment
2018-06-11 09:09:22 +09:00
Naoki Takezoe
f42792d9c3 Merge pull request #2018 from kounoike/pr-notification-hooks
Add notification hooks
2018-06-11 08:46:29 +09:00
Naoki Takezoe
42acd60527 Merge pull request #2061 from uli-heller/mariadb-2.2.5
org.mariadb.jdbc: 2.2.4 -> 2.2.5
2018-06-11 08:45:25 +09:00
Naoki Takezoe
5fc3ce34a3 Experiment of plugin installation from the remote repository 2018-06-10 20:29:09 +09:00
Uli Heller
a53c0a8386 org.mariadb.jdbc: 2.2.4 -> 2.2.5 2018-06-07 20:52:29 +02:00
Naoki Takezoe
97d2f6ba02 Merge pull request #2056 from kazuki43zoo/support-line-comment-on-split-mode
Support line comment on split mode
2018-06-07 11:22:43 +09:00
Kazuki Shimizu
05e1807a95 Fix to add a line comment correctly on split mode
Related with #2051
2018-06-05 22:33:48 +09:00
Kazuki Shimizu
3611b77c54 Merge branch 'gh-2050' into support-line-comment-on-split-mode 2018-06-05 22:18:31 +09:00
Naoki Takezoe
f2d561c557 (refs #2048)Fix wrong copy-and-pasted code :P 2018-06-05 13:56:30 +09:00
Naoki Takezoe
ea547a43a5 (refs #2040)Fix permission check for the repository menu 2018-06-05 12:36:50 +09:00
Naoki Takezoe
8068bb41ca Fix indent 2018-06-05 12:14:28 +09:00
Naoki Takezoe
c3eb30c01d (refs #2048)Fix archive downloading 2018-06-05 11:58:05 +09:00
Naoki Takezoe
7a795525ed Merge pull request #2044 from kazuki43zoo/fix-disable-comment-button
Release the disable status on line comment adding button when validation error occurred
2018-06-04 15:38:19 +09:00
Naoki Takezoe
082b2f7123 Merge pull request #2043 from kazuki43zoo/move-into-commentform-area
Move buttons into commentform area
2018-06-04 15:37:04 +09:00
Uli Heller
7a4bbccdd9 Upgraded to jgit-5.0-rc2 2018-06-04 08:29:06 +02:00
Kazuki Shimizu
c58f6377fc Find all text when collect diff data
Fixes #2050
2018-06-04 00:18:09 +09:00
kenji yoshida
1e330b42a8 sbt 1.1.6 (#2039) 2018-06-03 13:41:25 +09:00
Steffen Raabe
889f4f0a0f Added mime-type mapping for SVG files
I added a mime-type mapping for SVG files because when serving gitbucket from Wildfly, SVG files like the logo "gitbucket.svg" are being sent with "Content-Type: text/html", causing the image not to be displayed in the browser.
Adding this mapping to web.xml fixes that behavior for deployments on Wildfly and possibly other application servers.
2018-06-02 16:41:43 +02:00
Kazuki Shimizu
ff4ad2d903 Release the disable status on line comment adding button when validation error occurred 2018-06-01 07:01:01 +09:00
Kazuki Shimizu
42fc021939 Move buttons into commentform area 2018-06-01 01:18:45 +09:00
Kazuki Shimizu
20af5aa742 Keep showing a fold comment when tasklist is not complete 2018-05-31 08:35:08 +09:00
Kazuki Shimizu
d2047ac985 Focus to textarea at add and edit a comment 2018-05-30 08:57:21 +09:00
Naoki Takezoe
65ac7b7b13 Fixup 2018-05-29 03:23:04 +09:00
Naoki Takezoe
165cf88219 Bump gist-plugin to 4.15.0 which supports GitBucket 4.25.0 2018-05-29 02:09:19 +09:00
Naoki Takezoe
80b7e15d94 Bump to 4.15.0 2018-05-29 00:04:02 +09:00
Naoki Takezoe
6eadebede2 Merge pull request #2035 from kazuki43zoo/gh-2015_count-conversations
Count multiple comments on the same line
2018-05-28 23:54:25 +09:00
Kazuki Shimizu
beb0401500 Count multiple comments on the same line
See #2015
2018-05-28 22:58:34 +09:00
Naoki Takezoe
eface25cf8 Update README and CHANGELOG 2018-05-28 15:56:36 +09:00
Naoki Takezoe
9eff4cb485 (refs #2031) Fix permission check for the repository menu 2018-05-28 15:49:47 +09:00
Naoki Takezoe
c65c3e2c49 Escape HTML 2018-05-28 15:42:32 +09:00
Naoki Takezoe
d064ca85fb Encode parameters in url generation helpers 2018-05-28 12:30:07 +09:00
Naoki Takezoe
df9c34bcec Fix indent 2018-05-28 12:26:11 +09:00
Naoki Takezoe
172701105a URL encode username in user link 2018-05-28 12:15:26 +09:00
Naoki Takezoe
e2da18a763 (refs #2030) Fix isolation level issue in MySQL 2018-05-28 00:46:53 +09:00
Naoki Takezoe
77383c4e8f Merge pull request #2034 from gitbucket/extra-mail-address-migration
Migration to delete empty extra mail addresses
2018-05-28 00:45:02 +09:00
Naoki Takezoe
aba9db3857 Migration to delete empty extra mail addresses 2018-05-27 22:46:50 +09:00
Naoki Takezoe
d97677aaaa Merge branch 'ThaFridge-master' 2018-05-27 13:14:10 +09:00
Naoki Takezoe
d02a4baf47 Remove duplicated X-UA-Compatible meta tag 2018-05-27 13:13:51 +09:00
ThaFridge
4ce07ee3dd Update with propper mobile/tablet scalling 2018-05-27 13:11:52 +09:00
Naoki Takezoe
a354522406 Merge pull request #2032 from kounoike/pr-fix-extra-email-bug
fix extra email bug in account registration
2018-05-27 12:31:42 +09:00
Naoki Takezoe
c4bdf86253 Merge pull request #2015 from kazuki43zoo/count-conversation
Counting conversations on pull request screen
2018-05-27 03:36:45 +09:00
KOUNOIKE Yuusuke
87fa283b65 fix extra email bug in account registration 2018-05-26 12:58:31 +09:00
Kazuki Shimizu
71828e5d08 Counting conversations on pull request screen 2018-05-26 09:51:56 +09:00
Naoki Takezoe
89bf8db087 Add new line to the end of response 2018-05-25 02:07:13 +09:00
Naoki Takezoe
2b2669978f Merge pull request #2027 from gitbucket/fix-download
Fix for repository downloading improvement
2018-05-24 18:20:30 +09:00
Naoki Takezoe
e7493eff3b Merge pull request #2028 from yaroot/pubkey
expose user pubkey via /{user}.keys
2018-05-24 16:29:47 +09:00
Naoki Takezoe
1adb0b7bcf Include repository name and directory name in download filename 2018-05-24 16:25:43 +09:00
Naoki Takezoe
587970a477 Merge branch 'master' into fix-download 2018-05-24 16:02:16 +09:00
Naoki Takezoe
b45e6428c7 Merge pull request #2029 from gitbucket/revert-2026-pr-show-edited
Revert "show "edited" in edited comment"
2018-05-24 11:19:38 +09:00
Naoki Takezoe
7c758cbdee Revert "show "edited" in edited comment" 2018-05-24 11:03:41 +09:00
Naoki Takezoe
ffc0b59a58 Merge pull request #2026 from kounoike/pr-show-edited
show "edited" in edited comment
2018-05-24 10:48:22 +09:00
Naoki Takezoe
382250c243 Merge pull request #2017 from kounoike/pr-secure-password
use PBKDF2 for password
2018-05-24 10:16:31 +09:00
Yan Su
489ba2cd17 expose user pubkey via /{user}.keys 2018-05-24 08:30:02 +08:00
Naoki Takezoe
2a489870a1 Fix for repository downloading improvement in #2014 2018-05-24 01:25:07 +09:00
KOUNOIKE Yuusuke
99f1eaf3d8 fix by review comment 2018-05-24 00:30:13 +09:00
KOUNOIKE Yuusuke
e1c7cd0965 show "edited" in edited comment 2018-05-24 00:29:30 +09:00
Naoki Takezoe
fbe60a59d7 Merge pull request #2025 from uli-heller/mariadb-2.2.4
mariadb-java-client: 2.2.3 -> 2.2.4
2018-05-24 00:23:59 +09:00
Naoki Takezoe
efdf27df6b Merge pull request #2014 from kounoike/pr-archive-improve
improve archive download
2018-05-24 00:10:24 +09:00
KOUNOIKE Yuusuke
9ffda21bfd fix by review comment 2018-05-23 23:37:10 +09:00
Uli Heller
8ee7270986 mariadb-java-client: 2.2.3 -> 2.2.4 2018-05-23 08:16:36 +02:00
Naoki Takezoe
d95a6b8134 (refs #2023)Move GitBucket version from global header to admin page 2018-05-23 14:51:06 +09:00
Naoki Takezoe
31d546fd5a Merge pull request #2016 from kazuki43zoo/create-comment-on-change-title
Create comment on changing issue title
2018-05-23 02:18:28 +09:00
Naoki Takezoe
9812f66b0d Merge pull request #2013 from kazuki43zoo/avatarLink
Change to avatar image link from avatar image
2018-05-23 02:17:46 +09:00
Kazuki Shimizu
5ac8b87a76 Update 'About Action in Issue Comment' in document 2018-05-23 01:51:23 +09:00
Naoki Takezoe
0f52dc4d8c Merge pull request #2019 from kazuki43zoo/disabled-checkbox-on-preview
Disabled a checkbox of tasklist on preview
2018-05-22 20:56:56 +09:00
Kazuki Shimizu
3c956ac03e Disabled a checkbox of tasklist on preview 2018-05-19 23:23:22 +09:00
KOUNOIKE Yuusuke
511058dab4 scalafmt 2018-05-19 19:23:45 +09:00
KOUNOIKE Yuusuke
b1743b4c28 call hooks when issue closed by comment 2018-05-19 19:02:22 +09:00
KOUNOIKE Yuusuke
555b3465ed add hooks for notification. 2018-05-19 19:02:22 +09:00
KOUNOIKE Yuusuke
d45cba30c0 forgot add xml 2018-05-19 18:41:58 +09:00
KOUNOIKE Yuusuke
0840081dc8 use PBKDF2 for password. close #118 2018-05-19 18:19:49 +09:00
Kazuki Shimizu
0c0da0cbf7 Create comment on changing issue title 2018-05-19 17:48:30 +09:00
KOUNOIKE Yuusuke
e3641d0bf7 improve archive download 2018-05-19 15:02:48 +09:00
Kazuki Shimizu
1c118b8cd7 Change to avatar image link from avatar image 2018-05-19 14:32:40 +09:00
Naoki Takezoe
abf516682b Merge pull request #2011 from scf37/securerandom
Use SecureRandom to generate access tokens.
2018-05-18 14:16:24 +09:00
Scf37
72d07422a4 Use SecureRandom to generate access tokens.
scala.util.Random uses java.util.Random which only provides 64 bits of randomness.
2018-05-17 20:16:37 +03:00
Naoki Takezoe
ecc50cd2ae Merge pull request #2007 from kazuki43zoo/allow-task-list-on-commitcomment
Allow task list on commit comment area
2018-05-15 12:22:52 +09:00
Naoki Takezoe
acbcb60629 Merge pull request #1986 from kounoike/pr-ace-mode-by-js
set ace editor mode by ext-modelist.js
2018-05-12 23:03:36 +09:00
Naoki Takezoe
23a9bf46a2 Merge pull request #1989 from kounoike/pr-show-mail
Show mail address in profile page
2018-05-12 22:49:39 +09:00
Kazuki Shimizu
342ad68212 Allow task list on commit comment area 2018-05-12 21:26:19 +09:00
Naoki Takezoe
6d3dec518f Merge pull request #2006 from kazuki43zoo/gh-1976_download-patch-file-on-PR
Download a correct patch file on PR screen
2018-05-12 18:07:07 +09:00
Naoki Takezoe
e350633b69 (refs #2004) Fix review comments duplication 2018-05-12 18:06:26 +09:00
Kazuki Shimizu
4e3be1deb5 Download a correct patch file on PR screen
Fixes gh-1976
2018-05-12 17:49:29 +09:00
Naoki Takezoe
dc290614ca Merge pull request #2003 from kazuki43zoo/view-badge-on-pulls
View badges on pull request screen
2018-05-12 17:24:46 +09:00
Naoki Takezoe
1b22c2e29b Fix invisible tag location to fix button group style 2018-05-12 16:53:59 +09:00
Naoki Takezoe
ddeaffb705 Move Fork button to the header 2018-05-12 16:37:30 +09:00
Kazuki Shimizu
eedbd9f45a View badges on pull request screen 2018-05-12 12:08:59 +09:00
Naoki Takezoe
fa29acef54 (refs #2002) Recover commit comment for the specific commits 2018-05-11 16:09:44 +09:00
Naoki Takezoe
6355f8d0a4 Merge pull request #2001 from xuwei-k/sbt-1.1.5
sbt 1.1.5
2018-05-10 13:56:32 +09:00
Naoki Takezoe
173fc30211 (refs #2000) Recover title editing of pull request 2018-05-09 17:06:45 +09:00
kenji yoshida
4df9c36d82 sbt 1.1.5 2018-05-09 15:47:19 +09:00
Naoki Takezoe
33c0fb680e Merge pull request #1996 from kounoike/fix-json-diff
Fix JSON diff issue.
2018-05-06 02:26:06 +09:00
KOUNOIKE Yuusuke
378b3986dc Fix JSON diff issue. 2018-05-04 14:39:29 +09:00
Naoki Takezoe
f321d0974e Update README.md 2018-05-03 11:05:25 +09:00
Naoki Takezoe
227e2786e1 Merge pull request #1994 from kounoike/pr-show-email-settings-error
Show SMTP Error message in testing email settings
2018-05-02 16:50:56 +09:00
KOUNOIKE Yuusuke
1a90fd86ff format 2018-05-02 14:07:19 +09:00
KOUNOIKE Yuusuke
49f095bb26 Show SMTP Error message in testing email settings 2018-05-02 13:55:16 +09:00
Naoki Takezoe
b9acfc62c6 Use random UUID as blowfish key 2018-05-01 08:39:30 +09:00
Naoki Takezoe
864df6cdac Use random UUID as blowfish key 2018-05-01 08:27:54 +09:00
Naoki Takezoe
f0f4b8faa6 Bump to GitBucket 4.24.1 2018-05-01 01:41:52 +09:00
Naoki Takezoe
6350354942 Merge pull request #1991 from kounoike/fix-branch-protect
Fix branch protection problem
2018-05-01 01:39:16 +09:00
KOUNOIKE Yuusuke
bde66b2896 don't use :repo, should use :repository 2018-05-01 01:23:09 +09:00
Naoki Takezoe
173669f75e Fix cancel button style 2018-04-30 03:29:26 +09:00
KOUNOIKE Yuusuke
f53497da56 forgot radio group change. 2018-04-29 20:51:34 +09:00
Naoki Takezoe
5913bcd309 Merge pull request #1988 from kounoike/pr-fix-editor-1962
Fix editor preview bug
2018-04-29 20:38:47 +09:00
Naoki Takezoe
2a10acd76f Update README and CHANGELOG for GitBucket 4.24.0 2018-04-29 20:16:43 +09:00
KOUNOIKE Yuusuke
70dbee839a Show mail address in profile page, It can be controlled by settings. closes #673. 2018-04-29 19:05:45 +09:00
KOUNOIKE Yuusuke
2c6a8cf08a Fix editor preview bug introduced by #1962 2018-04-29 17:54:29 +09:00
KOUNOIKE Yuusuke
59859359ea set ace editor mode by ext-modelist.js 2018-04-29 16:37:51 +09:00
Naoki Takezoe
1cd0759325 (refs #1969) Bump gist plugin 2018-04-29 15:36:40 +09:00
Naoki Takezoe
915cfd06c3 Merge remote-tracking branch 'origin/master' 2018-04-29 11:34:45 +09:00
Naoki Takezoe
4cb198bc05 Bump to 4.24.0 2018-04-29 11:34:13 +09:00
Naoki Takezoe
1ea0e827a5 Bug fix for non-inline commit comment 2018-04-29 11:33:59 +09:00
Naoki Takezoe
8f4744f93d Merge pull request #1984 from kounoike/pr-call-webhook-when-close-by-comment
call issue closed webhook when pushed commit contains keywords
2018-04-29 09:36:59 +09:00
Naoki Takezoe
7c0d0be876 Merge pull request #1983 from kounoike/pr-close-pr-by-push
close and mark as merged PR by pushed commits
2018-04-29 09:35:53 +09:00
Naoki Takezoe
81c2d988a3 Merge pull request #1982 from kounoike/pr-fix-pr-message
don't separate user:branch in PR message
2018-04-29 09:31:52 +09:00
Naoki Takezoe
eef01262bd Add file name validation 2018-04-29 01:03:17 +09:00
KOUNOIKE Yuusuke
921298cf92 call issue closed webhook when pushed commit contains keywords. fix #1880 2018-04-28 23:58:04 +09:00
KOUNOIKE Yuusuke
56eb2435e3 close and mark as merged PR by pushed commits 2018-04-28 23:16:06 +09:00
KOUNOIKE Yuusuke
069d84e249 don't separate user:branch in PR message 2018-04-28 21:35:15 +09:00
Naoki Takezoe
c63c8d3cd2 Merge pull request #1980 from kounoike/pr-add-tags-on-commits
show tags on commits page
2018-04-28 20:28:45 +09:00
Naoki Takezoe
ca66faebdf Merge pull request #1979 from xuwei-k/Scala-2.12.6
Scala 2.12.6
2018-04-28 17:53:33 +09:00
KOUNOIKE Yuusuke
9b5530b3fa show tags on commits page. refs #1973 2018-04-28 16:04:50 +09:00
kenji yoshida
b3319daf95 Scala 2.12.6 2018-04-28 06:44:08 +09:00
Naoki Takezoe
cf038ebd38 Merge pull request #1961 from gitbucket/improve-pullreq-comments
Improve pull request comments presentation
2018-04-28 01:04:39 +09:00
Naoki Takezoe
163ed5c4a0 Fold outdated comments 2018-04-27 19:10:14 +09:00
Naoki Takezoe
9bff4b1e97 Set location information to reply comment correctly 2018-04-27 18:39:48 +09:00
Naoki Takezoe
332836119d Fix comment header 2018-04-27 17:33:56 +09:00
Naoki Takezoe
110646fe9f Fix comment deletion bug 2018-04-27 15:28:42 +09:00
Naoki Takezoe
e578875d51 Fix diff fragment 2018-04-27 15:06:30 +09:00
Naoki Takezoe
86903a7a22 Save diff fragment as a file instead of database 2018-04-27 14:57:59 +09:00
Naoki Takezoe
dd1dbd429c Display partial diff for commit comments 2018-04-27 12:04:23 +09:00
Naoki Takezoe
efe891a348 Store diff JSON data into the database 2018-04-27 08:55:14 +09:00
Naoki Takezoe
a720c6bee4 Merge branch 'master' into improve-pullreq-comments 2018-04-27 08:35:59 +09:00
Naoki Takezoe
c0a7c3537a Fix MatchError for CommitComment 2018-04-27 08:35:10 +09:00
Naoki Takezoe
2d1f08bc01 Send diff data of a commented location to the server 2018-04-26 20:50:01 +09:00
Naoki Takezoe
ddbc2e6a56 Fix comment deletion 2018-04-26 20:02:13 +09:00
Naoki Takezoe
2c201aae61 Merge pull request #1939 from kounoike/pr-multi-mailaddress
Add extra mail address support.
2018-04-26 02:19:15 +09:00
Naoki Takezoe
3f74745fc5 Fixup 2018-04-25 12:06:58 +09:00
Naoki Takezoe
0c17a23cca Merge branch 'master' into improve-pullreq-comments 2018-04-25 11:40:45 +09:00
Naoki Takezoe
8b2b36df0e Remove code in authenticators which keeps backward compatibility 2018-04-25 01:55:02 +09:00
Naoki Takezoe
07182cf946 Use parameter name to get repository owner and name from requested path 2018-04-24 18:22:04 +09:00
Naoki Takezoe
ea69463fd2 Remove unnecessary braces 2018-04-24 17:52:36 +09:00
Naoki Takezoe
3abd934d6a Merge pull request #1971 from yaroot/scalafmt
update scalafmt
2018-04-24 16:21:38 +09:00
Yan Su
7c10cb0ec7 update scalafmt 2018-04-24 13:40:23 +08:00
KOUNOIKE Yuusuke
d224fc1c5f add tag on commit. close #1265 2018-04-23 00:10:49 +09:00
KOUNOIKE Yuusuke
58381c3d30 Handle .editorconfig parse error 2018-04-22 16:46:34 +09:00
KOUNOIKE Yuusuke
e2d5382787 scalafmtSbt applied 2018-04-22 12:42:50 +09:00
KOUNOIKE Yuusuke
e350126794 Support EditorConfig for online browser/editor. 2018-04-22 12:24:28 +09:00
Naoki Takezoe
c1d6839c18 Update README.md 2018-04-20 20:00:25 +09:00
Naoki Takezoe
f62f83888c Fix comment list in the commit page 2018-04-20 02:35:41 +09:00
Naoki Takezoe
10cec316ee Adjust layout 2018-04-19 22:09:10 +09:00
Naoki Takezoe
545846cab5 Merge pull request #1962 from teruuuuuu/fix/#1871
Fix  #1871: Fixed breaking newlines when starting with multiple blank…
2018-04-19 19:38:51 +09:00
Naoki Takezoe
acf5767e4f Display error message as a toast 2018-04-19 18:56:00 +09:00
Naoki Takezoe
e4520247fc Recover review comment editing ability 2018-04-19 18:49:59 +09:00
Naoki Takezoe
a39a0292b6 Separate pull request tab contents as individual actions 2018-04-19 18:39:25 +09:00
Naoki Takezoe
6150625e99 Destruct issue-content style class 2018-04-19 16:46:32 +09:00
Naoki Takezoe
154f080c1c Tweaking commit comments presentation 2018-04-19 16:02:11 +09:00
Naoki Takezoe
b0ec5307a2 Modifying commit comments presentation 2018-04-19 13:42:06 +09:00
Naoki Takezoe
e5d4bc6653 Fix spelling 2018-04-19 13:12:19 +09:00
Naoki Takezoe
a701182f0c Formatted 2018-04-19 13:06:43 +09:00
Naoki Takezoe
f1db6e3c7c Grouping commit comments 2018-04-19 13:05:25 +09:00
arimuraterutoshiMac
6926aa7aec Fix #1871: Fixed breaking newlines when starting with multiple blank lines 2018-04-19 01:51:24 +09:00
Naoki Takezoe
1a0f282f23 Improve pull request comments presentation 2018-04-18 18:47:00 +09:00
Naoki Takezoe
255aa7476c Merge pull request #1953 from gitbucket/disable-removed-user-repos
Disable removed user's repositories
2018-04-17 10:16:32 +09:00
Naoki Takezoe
34df9ae739 Merge pull request #1959 from gitbucket/expand-api-auth
Apply ApiAuthenticationFilter to /api/* to cover APIs other than GitHub API
2018-04-17 10:15:23 +09:00
Naoki Takezoe
1fb0eeff22 Merge pull request #1952 from gitbucket/validate-case-difference
Disallow users and repositories which have different letter cases
2018-04-17 10:06:43 +09:00
Naoki Takezoe
8b963a32f0 Merge pull request #1955 from kounoike/pr-improve-payload-for-discord
Improve webhook payload compatibility for discord.
2018-04-17 10:06:22 +09:00
Naoki Takezoe
db17508559 Fix error response processing of APIs 2018-04-17 10:05:05 +09:00
Naoki Takezoe
79fb95c149 Merge pull request #1942 from yaroot/baseurl-xforwardedproto
Add X-Forwarded-Proto support for baseurl parsing
2018-04-16 23:57:41 +09:00
KOUNOIKE Yuusuke
f61149ae61 fixup by review comment 2018-04-16 21:28:39 +09:00
Naoki Takezoe
3abe398244 Apply ApiAuthenticationFilter to /api/* to cover APIs other than GitHub API 2018-04-16 19:41:11 +09:00
Naoki Takezoe
6e0deb6a6c Tiny update for the group editing form 2018-04-16 14:09:48 +09:00
Naoki Takezoe
13622c5970 Merge pull request #1957 from kounoike/fix-1956
fix #1956
2018-04-16 09:10:09 +09:00
KOUNOIKE Yuusuke
0c3c25b7c0 fix #1956 2018-04-15 21:53:08 +09:00
KOUNOIKE Yuusuke
6e31f19e9b Improve webhook payload compatibility for discord. close #1888 2018-04-15 15:13:10 +09:00
KOUNOIKE Yuusuke
52ec97b87e drop duplicated line. 2018-04-15 13:49:33 +09:00
KOUNOIKE Yuusuke
a5ccea6413 add comma 2018-04-15 13:34:57 +09:00
KOUNOIKE Yuusuke
cc091a65a4 Merge remote-tracking branch 'upstream/master' into pr-multi-mailaddress
# Conflicts:
#	src/main/scala/gitbucket/core/GitBucketCoreModule.scala
2018-04-15 13:33:08 +09:00
KOUNOIKE Yuusuke
884d4935cc Check duplicate across primary/extra mail addresses. 2018-04-15 13:28:03 +09:00
Naoki Takezoe
35655f33c7 Change local storage key 2018-04-15 12:11:31 +09:00
Naoki Takezoe
5022702796 Merge pull request #1954 from gitbucket/keep-editor-wrap
Keep wrap mode of Ace editor using localStorage
2018-04-15 12:05:15 +09:00
KOUNOIKE Yuusuke
243833c3fc improve getAccountByMailAddress query. 2018-04-15 11:34:55 +09:00
Naoki Takezoe
3f1ea419d6 Keep wrap mode of Ace editor using localStorage 2018-04-15 02:11:16 +09:00
Yan Su
0c2c590678 Add X-Forwarded-Proto support for baseurl parsing 2018-04-13 17:57:11 +08:00
Naoki Takezoe
3ca73aaafb Disable removed user’s repositories 2018-04-13 17:03:57 +09:00
Naoki Takezoe
b4cf4bfb17 Disallow users and repositories which have different letter cases 2018-04-13 15:56:56 +09:00
Naoki Takezoe
5cb26247fc Update README.md and CHANGELOG.md for GitBucket 4.23.1 release 2018-04-10 02:38:06 +09:00
Naoki Takezoe
2e391144f2 Merge branch 'master' of https://github.com/gitbucket/gitbucket 2018-04-09 19:46:01 +09:00
Naoki Takezoe
bd69821f1d Bump to 4.23.1 2018-04-09 19:45:47 +09:00
Naoki Takezoe
d802ed099b Merge pull request #1944 from xuwei-k/sbt-1.1.3
sbt 1.1.3
2018-04-09 14:49:21 +09:00
kenji yoshida
170a48de36 sbt 1.1.3 2018-04-08 14:45:37 +09:00
Naoki Takezoe
1d2d33ba71 Merge pull request #1940 from gitbucket/fix-get-contents-api
Fix a bug that fails to get the list of files of the repository root
2018-04-08 01:11:23 +09:00
Naoki Takezoe
9bcc2a3202 Fix a bug that fails to get the list of files of the repository root 2018-04-03 02:00:07 +09:00
KOUNOIKE Yuusuke
d65637ce0c Add extra mail address support. close #652 2018-04-02 01:01:18 +09:00
Naoki Takezoe
2e910aebdc Merge pull request #1938 from xuwei-k/Xfuture
add -Xfuture option. fix warnings
2018-04-02 00:23:12 +09:00
Naoki Takezoe
a1621b49a6 Merge pull request #1937 from xuwei-k/sbt-1.1.2
sbt 1.1.2
2018-04-02 00:22:46 +09:00
xuwei-k
ac6fbd0bfa add -Xfuture option. fix warnings
view bounds and procedure syntax are deprecated
2018-04-01 22:43:47 +09:00
kenji yoshida
5305b1b09a sbt 1.1.2 2018-04-01 22:38:37 +09:00
kenji yoshida
4f92739d73 check scalafmt in travis (#1936)
format test and sbt files
2018-04-01 22:24:51 +09:00
Naoki Takezoe
cadd128299 Merge pull request #1934 from gitbucket/apply-scalafmt
Apply scalafmt
2018-04-01 14:42:48 +09:00
Naoki Takezoe
b642783610 Apply scalafmt 2018-04-01 12:01:17 +09:00
Naoki Takezoe
4aaaff5de7 Merge pull request #1923 from yaroot/scalafmt
Add scalafmt
2018-04-01 11:55:21 +09:00
Naoki Takezoe
2fda39ddbe Merge pull request #1932 from stephengroat/patch-1
better looking gitter badge
2018-04-01 11:50:23 +09:00
Stephen
6753bd085f better looking gitter badge 2018-03-31 08:55:21 -07:00
Naoki Takezoe
2d8513e18a Merge pull request #1931 from gitbucket/fix-release-file-handling
Fix file upload bug in the release page
2018-03-31 23:11:57 +09:00
Naoki Takezoe
047c877d83 Fix file upload bug in the release page 2018-03-31 22:22:50 +09:00
Naoki Takezoe
b09c69afd7 Merge pull request #1928 from vmi/tomcat-shutdown
Fix that Tomcat deployed with GitBucket can not shutdown.
2018-03-31 02:35:44 +09:00
Naoki Takezoe
6713dd97f5 Merge pull request #1927 from kounoike/pr-render-for-binary
Render by plugin even if file is binary
2018-03-31 02:34:44 +09:00
KOUNOIKE Yuusuke
d5b2a9d6b5 fix typo in ApiPlugin 2018-03-31 02:33:16 +09:00
Naoki Takezoe
964f6d3c82 Merge remote-tracking branch 'origin/release/gitbucket-4.23.0' 2018-03-31 01:02:47 +09:00
Naoki Takezoe
f1164e7790 Revert "fix typo in ApiPlugin"
This reverts commit de8e553906.
2018-03-31 01:00:12 +09:00
Naoki Takezoe
b8f1736a09 Update README.md and CHANGELOG.md for 4.23.0 release 2018-03-30 15:51:22 +09:00
KOUNOIKE Yuusuke
de8e553906 fix typo in ApiPlugin 2018-03-30 01:17:48 +09:00
IWAMURO Motonori
d6c9101b83 Fix that Tomcat deployed with GitBucket can not shutdown. 2018-03-29 22:03:13 +09:00
Naoki Takezoe
50137c5546 Bump gitbucket-pages-plugin to 1.7.0 2018-03-28 00:53:53 +09:00
KOUNOIKE Yuusuke
9e131c8970 Render by plugin even if file is binary 2018-03-28 00:34:35 +09:00
Naoki Takezoe
8733b8eddd Bump gist plugin and notifications plugin 2018-03-27 09:22:55 +09:00
Naoki Takezoe
a0c06855d2 Bump to 4.23.0 2018-03-27 01:00:45 +09:00
Naoki Takezoe
64ce44243c Merge pull request #1921 from gitbucket/path-end-slash
Support all of paths which end with slash
2018-03-27 00:55:08 +09:00
Naoki Takezoe
66d2af1ef5 Fix redirection path 2018-03-26 16:19:17 +09:00
Yan Su
d155cb67b3 add scalafmt 2018-03-26 13:25:02 +08:00
Naoki Takezoe
7d7b13de6e Remove tail slash 2018-03-26 03:36:34 +09:00
Naoki Takezoe
329a8ebc2b Support all of paths which end with slash 2018-03-22 01:39:02 +09:00
Naoki Takezoe
1a16edd140 Merge pull request #1917 from kounoike/pr-list-plugins-api
Add plugin list API
2018-03-21 23:15:11 +09:00
Naoki Takezoe
28fed98924 Merge pull request #1919 from uli-heller/scala-2.12.5
scala: 2.12.4 -> 2.12.5
2018-03-21 23:14:30 +09:00
KOUNOIKE Yuusuke
08ac28902d Change plugin list API URL 2018-03-21 15:05:49 +09:00
Uli Heller
976f5e5a32 scala: 2.12.4 -> 2.12.5 2018-03-21 06:37:52 +01:00
Naoki Takezoe
ff53329cd6 Merge pull request #1918 from gitbucket/remove_unused_code
Remove unused action in AccountController
2018-03-21 13:03:32 +09:00
Naoki Takezoe
98de258abf Remove unused action in AccountController 2018-03-21 12:48:03 +09:00
Naoki Takezoe
5afef6713f Merge pull request #1902 from int128/oidc-ux-improvement
Show banner to create a personal access token for OIDC users
2018-03-21 12:36:29 +09:00
Naoki Takezoe
da21db8776 Merge remote-tracking branch 'origin/master' 2018-03-21 00:17:08 +09:00
Naoki Takezoe
f004c1c75b Commit transaction for each plugin migration 2018-03-21 00:16:54 +09:00
Naoki Takezoe
c96dbf4db5 Merge pull request #1916 from gitbucket/rescue_release_mysql
Rename table RELEASE to RELEASE_TAG
2018-03-21 00:08:36 +09:00
KOUNOIKE Yuusuke
742bdc0252 Add plugin list API 2018-03-20 09:28:58 +09:00
Naoki Takezoe
850429c507 Merge pull request #1914 from peccu/prevent-parsig-filename-as-md-in-conflict-message
Prevent parsing filename as Markdown in conflict message
2018-03-19 15:37:05 +09:00
Naoki Takezoe
17f9f066d8 Fix migration 2018-03-19 15:14:54 +09:00
Naoki Takezoe
d8e5a89ac5 Rename table RELEASE to RELEASE_TAG 2018-03-18 14:04:48 +09:00
peccu
33812cb337 set filename in conflict message as code 2018-03-17 11:52:20 +09:00
Naoki Takezoe
b89c99d388 Fix style classname 2018-03-12 18:12:00 +09:00
Naoki Takezoe
f5ec9ac1bf Tweak a style of titles of the system settings 2018-03-12 18:09:00 +09:00
Naoki Takezoe
3fd8cb2b3e Merge pull request #1908 from kounoike/pr-add-labels-to-api
Add labels to ApiIssue/ApiPullRequest
2018-03-12 16:38:56 +09:00
Naoki Takezoe
af62ccd72d Merge pull request #1910 from uli-heller/maria-2.2.3
mariadb-java-client: 2.2.2 -> 2.2.3
2018-03-12 16:37:50 +09:00
Naoki Takezoe
f097dd2316 Merge pull request #1909 from uli-heller/jgit-4.11.0.201803080745-r
jgit: Updated to 4.11.0.201803080745-r
2018-03-12 16:37:23 +09:00
Uli Heller
19640bd2d4 mariadb-java-client: 2.2.2 -> 2.2.3 2018-03-12 06:44:51 +01:00
Uli Heller
d2d5e9cc43 jgit: Updated to 4.11.0.201803080745-r 2018-03-12 06:09:25 +01:00
KOUNOIKE Yuusuke
3a7581b391 Fix tests for add labels to ApiIssue/ApiPullRequest 2018-03-11 17:10:17 +09:00
KOUNOIKE Yuusuke
d87305c11d Add labels to ApiIssue/ApiPullRequest 2018-03-11 16:57:05 +09:00
Naoki Takezoe
e935e6e6b1 (refs #1870) Insert comma before repeated table and column names 2018-03-11 16:37:11 +09:00
Naoki Takezoe
04a7548a26 Merge pull request #1907 from gitbucket/tag-message
Display commit message of tags in the release page
2018-03-11 12:37:29 +09:00
Naoki Takezoe
64528cb4a8 (refs #1898) Fixup 2018-03-11 02:09:11 +09:00
Naoki Takezoe
ae26e8ec6a (refs #1898) Display commit message of tags in the release page 2018-03-09 21:25:36 +09:00
Naoki Takezoe
8a13721c90 PostgreSQL doesn't accept back quotes. 2018-03-09 20:01:26 +09:00
Hidetake Iwata
e91d098055 Show banner to create token for OIDC users 2018-03-09 17:40:58 +09:00
Naoki Takezoe
5ca4a1a233 Merge pull request #1900 from int128/auth-personal-access-token
Git authentication by personal access tokens
2018-03-08 11:11:47 +09:00
Hidetake Iwata
a39fe7f51c Add Git authentication by personal access token 2018-03-06 21:50:50 +09:00
Naoki Takezoe
e1c05eb961 Update README.md 2018-03-03 02:41:08 +09:00
Naoki Takezoe
b25f97a96f Bump to 4.22.0 2018-03-03 02:29:51 +09:00
Naoki Takezoe
1c0f99bd64 Merge pull request #1893 from kounoike/pr-ace-1.3.1
update to Ace-1.3.1
2018-03-03 01:14:18 +09:00
KOUNOIKE Yuusuke
800d48d6d2 update to Ace-1.3.1 2018-02-28 01:18:45 +09:00
Naoki Takezoe
02a8540875 Merge pull request #1864 from kounoike/pr-more-action-issuecomments
Save more actions as issue comments.
2018-02-27 18:17:46 +09:00
Naoki Takezoe
048cdfc050 Merge pull request #1891 from uli-heller/mariadb-222
mariadb-java-client: 2.2.1 -> 2.2.2
2018-02-27 14:22:02 +09:00
Uli Heller
13f40d2b59 mariadb-java-client: 2.2.1 -> 2.2.2 2018-02-26 20:14:19 +01:00
Naoki Takezoe
fe2920a08f Merge pull request #1890 from gitbucket/confirm-passward-changing
Confirmation dialog for password changing
2018-02-23 18:41:10 +09:00
Naoki Takezoe
f59beded21 Confirmation dialog for password changing 2018-02-23 16:38:00 +09:00
Naoki Takezoe
c75119badf Tiny fixes for #1876 2018-02-23 16:14:26 +09:00
Naoki Takezoe
7b1292a9af Merge pull request #1876 from stheno/master
Added a couple features to the database viewer.
2018-02-23 13:12:57 +09:00
Naoki Takezoe
9ae26800c8 Merge pull request #1889 from gitbucket/autocomplete-off
Set autocomplete=off to almost forms
2018-02-22 17:29:47 +09:00
Naoki Takezoe
1eb8c83061 Merge pull request #1886 from uli-heller/jgit-4.10.0.201712302008-r
Updated jgit to 4.10.0.201712302008-r
2018-02-22 16:08:14 +09:00
Naoki Takezoe
a0069fde57 Set autocomplete=off to almost forms 2018-02-15 16:37:40 +09:00
Naoki Takezoe
ca5b121272 Fix year in CHANGELOG 2018-02-13 00:21:51 +09:00
Naoki Takezoe
6f4c081f10 Merge pull request #1885 from uli-heller/readme-md
Fixed years within README.md (closes #1884)
2018-02-13 00:20:22 +09:00
Uli Heller
227ee12e4c Updated jgit to 4.10.0.201712302008-r 2018-02-12 15:19:29 +01:00
Uli Heller
fc1cfe3f55 Fixed years within README.md (closes #1884) 2018-02-12 14:30:06 +01:00
Naoki Takezoe
76fa5fb474 (refs #1879) Support tag comparing 2018-02-12 02:13:57 +09:00
Naoki Takezoe
3c847bf957 (refs #1831) Add default merge strategy setting 2018-02-11 14:54:40 +09:00
Naoki Takezoe
88427b893c Merge pull request #1883 from gitbucket/disable-merge-strategies
Add a option to enable/disable merge strategies
2018-02-11 13:51:45 +09:00
Naoki Takezoe
737b0a9bdf (refs #1859) Add options to manage merge strategies 2018-02-11 13:38:03 +09:00
Naoki Takezoe
c8a7ef2fdb Merge pull request #1867 from gitbucket/refine-system-settings-page
Refine the system settings page
2018-02-11 01:13:33 +09:00
Naoki Takezoe
6d6331bbf3 Refine system settings page
Fixup
2018-02-11 00:58:29 +09:00
Naoki Takezoe
d1f2a72f06 (refs #1863) Fix path separator encoding in file finder 2018-02-10 02:00:07 +09:00
Naoki Takezoe
576daa0669 (refs #1881) Apply max file size setting to file uploading and wiki editing page 2018-02-10 01:28:49 +09:00
Naoki Takezoe
934d1df991 Merge pull request #1878 from alejandroliu/OIC-fix
Assign dummy password to newly created users
2018-02-10 00:25:38 +09:00
Naoki Takezoe
aa7db68e68 Merge pull request #1882 from xuwei-k/sbt-1.1.1
sbt 1.1.1
2018-02-10 00:17:03 +09:00
xuwei-k
a1bb667ec4 sbt 1.1.1 2018-02-09 12:15:41 +09:00
Naoki Takezoe
90ae489d35 Fix a position of the source url field 2018-02-09 02:00:43 +09:00
Naoki Takezoe
5bbd4f533f Merge pull request #1866 from kounoike/pr-initialize-empty-commit
Add new initialize option as `git commit -m "..." --allow-empty"`
2018-02-09 01:12:47 +09:00
A Liu Ly
283baaed57 Fixed failing tests. 2018-02-07 08:05:21 +00:00
A Liu Ly
c1e2191120 Assign dummy password to newly created users 2018-02-05 23:38:41 +00:00
stheno
28b0ea7f88 Added a couple features to the database viewer.
* Auto query on selection
* Choice of insert full query or just name based on auto checkbox
* A "select" on child menu item in left list.

I am not sure this is the right layout you want but this is a general idea if one chooses to adjust for project goals.
2018-02-04 19:28:18 -08:00
Naoki Takezoe
95acb8593a Merge pull request #1873 from kounoike/fix-1872
fix 1872.
2018-02-04 00:53:41 +09:00
KOUNOIKE Yuusuke
17d682f83b fix 1872.
java.io.File#getParent returns path with "\" instead of "/".
so don't use this method.
2018-02-01 02:41:43 +09:00
Naoki Takezoe
952c916e33 Remove context path from request uri in plugin routing 2018-01-30 11:26:15 +09:00
KOUNOIKE Yuusuke
3793a51e3f Add new initialize option as git commit -m "..." --allow-empty" 2018-01-28 18:59:39 +09:00
KOUNOIKE Yuusuke
770b49cc55 fix close_comment/reopen_comment/merge comment doesn't show 2018-01-27 18:23:51 +09:00
KOUNOIKE Yuusuke
95cd6b6e90 fix for build and test 2018-01-27 17:52:06 +09:00
KOUNOIKE Yuusuke
1f79ed95c2 fix table 2018-01-27 17:17:38 +09:00
KOUNOIKE Yuusuke
9aef90d214 insert comments when batch edit. 2018-01-27 16:52:20 +09:00
KOUNOIKE Yuusuke
f4d1c72f08 Add visibility of Labels/Priority/Milestone/Assignee changes in an issue. closes #1855 2018-01-27 16:46:11 +09:00
637 changed files with 165967 additions and 143020 deletions

11
.scalafmt.conf Normal file
View File

@@ -0,0 +1,11 @@
project.git = true
maxColumn = 120
docstrings = JavaDoc
align.tokens = ["%", "%%", {code = "=>", owner = "Case"}]
align.openParenCallSite = false
align.openParenDefnSite = false
continuationIndent.callSite = 2
continuationIndent.defnSite = 2
danglingParentheses = true

View File

@@ -3,7 +3,7 @@ sudo: true
jdk: jdk:
- oraclejdk8 - oraclejdk8
script: script:
- sbt test - sbt scalafmtSbtCheck scalafmtCheck test:scalafmtCheck test
before_script: before_script:
- sudo /etc/init.d/mysql stop - sudo /etc/init.d/mysql stop
- sudo /etc/init.d/postgresql stop - sudo /etc/init.d/postgresql stop

View File

@@ -1,10 +1,61 @@
# Changelog # Changelog
All changes to the project will be documented in this file. All changes to the project will be documented in this file.
### 4.21.2 - 27 Jan 2019 ### 4.27.0 - 29 Jul 2018
- Create new tag on the browser
- EditorConfig support
- Improve issues / pull requests search
### 4.26.0 - 30 Jun 2018
- Installing plugins from the central registry
- Repositories tab in the dashboard
- Fork dialog enhancement
- Adjust pull request creation suggestor
- Keep showing incompleted task list
- New notification hooks
### 4.25.0 - 29 May 2018
- Security improvements
- Show mail address at the profile page
- Task list on commit comments
- More detailed editing history of issues and pull requests
- Expose user public keys
- Download repository improvements
### 4.24.1 - 1 May 2018
- Fix bug in Web API authentication
### 4.24.0 - 30 Apr 2018
- Diff for each review comment on pull requests
- Extra mail addresses support
- Show tags at the commit list
- Keep wrap mode of the online editor
- Renew layout of gitbucket-gist-plugin
- Web API of gitbucket-ci-plugin
### 4.23.1 - 10 Apr 2018
- Fix bug that the contents API doesn't work for the repository root
- Fix shutdown problem in Tomcat deployment
- Render by plugins at the blob view even if it's a binary file
### 4.23.0 - 31 Mar 2018
- Allow tail slash in URL
- Display commit message of tags at the releases page
- Add labels property to issues and pull requests API response
- Plugins list API
- Git authentication with personal access token
- Max parallel builds and max stored history in CI plugin became configurable
### 4.22.0 - 3 Mar 2018
- Pull request merge strategy settings
- Create repository with an empty commit
- Improve database viewer
- Update maven-repository-plugin
### 4.21.2 - 27 Jan 2018
- Bugfix - Bugfix
### 4.21.1 - 27 Jan 2019 ### 4.21.1 - 27 Jan 2018
- Bugfix - Bugfix
### 4.21.0 - 27 Jan 2018 ### 4.21.0 - 27 Jan 2018

View File

@@ -1,4 +1,4 @@
GitBucket [![Gitter chat](https://badges.gitter.im/gitbucket/gitbucket.png)](https://gitter.im/gitbucket/gitbucket) [![Build Status](https://travis-ci.org/gitbucket/gitbucket.svg?branch=master)](https://travis-ci.org/gitbucket/gitbucket) GitBucket [![Gitter chat](https://badges.gitter.im/gitbucket/gitbucket.svg)](https://gitter.im/gitbucket/gitbucket) [![Build Status](https://travis-ci.org/gitbucket/gitbucket.svg?branch=master)](https://travis-ci.org/gitbucket/gitbucket) [![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.github.gitbucket/gitbucket_2.12/badge.svg)](https://maven-badges.herokuapp.com/maven-central/io.github.gitbucket/gitbucket_2.12) [![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://github.com/gitbucket/gitbucket/blob/master/LICENSE)
========= =========
GitBucket is a Git web platform powered by Scala offering: GitBucket is a Git web platform powered by Scala offering:
@@ -68,20 +68,11 @@ Support
- If you can't find same question and report, send it to [gitter room](https://gitter.im/gitbucket/gitbucket) before raising an issue. - If you can't find same question and report, send it to [gitter room](https://gitter.im/gitbucket/gitbucket) before raising an issue.
- The highest priority of GitBucket is the ease of installation and API compatibility with GitHub, so your feature request might be rejected if they go against those principles. - The highest priority of GitBucket is the ease of installation and API compatibility with GitHub, so your feature request might be rejected if they go against those principles.
What's New in 4.21.x What's New in 4.27.x
------------- -------------
### 4.27.0 - 29 Jul 2018
### 4.21.2 - 27 Jan 2019 - Create new tag on the browser
- Bugfix - EditorConfig support
- Improve issues / pull requests search
### 4.21.1 - 27 Jan 2019
- Bugfix
### 4.21.0 - 27 Jan 2018
- Release page
- OpenID Connect support
- New database viewer
- Submodule links to web page
- Clarify close/reopen button
See the [change log](CHANGELOG.md) for all of the updates. See the [change log](CHANGELOG.md) for all of the updates.

View File

@@ -3,19 +3,22 @@ import com.typesafe.sbt.pgp.PgpKeys._
val Organization = "io.github.gitbucket" val Organization = "io.github.gitbucket"
val Name = "gitbucket" val Name = "gitbucket"
val GitBucketVersion = "4.21.2" val GitBucketVersion = "4.27.0"
val ScalatraVersion = "2.6.1" val ScalatraVersion = "2.6.1"
val JettyVersion = "9.4.7.v20170914" val JettyVersion = "9.4.7.v20170914"
lazy val root = (project in file(".")).enablePlugins(SbtTwirl, ScalatraPlugin, JRebelPlugin).settings( lazy val root = (project in file("."))
.enablePlugins(SbtTwirl, ScalatraPlugin)
) .settings(
)
sourcesInBase := false sourcesInBase := false
organization := Organization organization := Organization
name := Name name := Name
version := GitBucketVersion version := GitBucketVersion
scalaVersion := "2.12.4" scalaVersion := "2.12.6"
scalafmtOnCompile := true
// dependency settings // dependency settings
resolvers ++= Seq( resolvers ++= Seq(
@@ -25,9 +28,10 @@ resolvers ++= Seq(
"sonatype-snapshot" at "https://oss.sonatype.org/content/repositories/snapshots/", "sonatype-snapshot" at "https://oss.sonatype.org/content/repositories/snapshots/",
"amateras-snapshot" at "http://amateras.sourceforge.jp/mvn-snapshot/" "amateras-snapshot" at "http://amateras.sourceforge.jp/mvn-snapshot/"
) )
libraryDependencies ++= Seq( libraryDependencies ++= Seq(
"org.eclipse.jgit" % "org.eclipse.jgit.http.server" % "4.9.2.201712150930-r", "org.eclipse.jgit" % "org.eclipse.jgit.http.server" % "5.0.1.201806211838-r",
"org.eclipse.jgit" % "org.eclipse.jgit.archive" % "4.9.2.201712150930-r", "org.eclipse.jgit" % "org.eclipse.jgit.archive" % "5.0.1.201806211838-r",
"org.scalatra" %% "scalatra" % ScalatraVersion, "org.scalatra" %% "scalatra" % ScalatraVersion,
"org.scalatra" %% "scalatra-json" % ScalatraVersion, "org.scalatra" %% "scalatra-json" % ScalatraVersion,
"org.scalatra" %% "scalatra-forms" % ScalatraVersion, "org.scalatra" %% "scalatra-forms" % ScalatraVersion,
@@ -38,13 +42,13 @@ libraryDependencies ++= Seq(
"org.apache.commons" % "commons-compress" % "1.15", "org.apache.commons" % "commons-compress" % "1.15",
"org.apache.commons" % "commons-email" % "1.5", "org.apache.commons" % "commons-email" % "1.5",
"org.apache.httpcomponents" % "httpclient" % "4.5.4", "org.apache.httpcomponents" % "httpclient" % "4.5.4",
"org.apache.sshd" % "apache-sshd" % "1.6.0" exclude("org.slf4j","slf4j-jdk14"), "org.apache.sshd" % "apache-sshd" % "1.6.0" exclude ("org.slf4j", "slf4j-jdk14"),
"org.apache.tika" % "tika-core" % "1.17", "org.apache.tika" % "tika-core" % "1.17",
"com.github.takezoe" %% "blocking-slick-32" % "0.0.10", "com.github.takezoe" %% "blocking-slick-32" % "0.0.10",
"com.novell.ldap" % "jldap" % "2009-10-07", "com.novell.ldap" % "jldap" % "2009-10-07",
"com.h2database" % "h2" % "1.4.196", "com.h2database" % "h2" % "1.4.196",
"org.mariadb.jdbc" % "mariadb-java-client" % "2.2.1", "org.mariadb.jdbc" % "mariadb-java-client" % "2.2.6",
"org.postgresql" % "postgresql" % "42.1.4", "org.postgresql" % "postgresql" % "42.2.4",
"ch.qos.logback" % "logback-classic" % "1.2.3", "ch.qos.logback" % "logback-classic" % "1.2.3",
"com.zaxxer" % "HikariCP" % "2.7.4", "com.zaxxer" % "HikariCP" % "2.7.4",
"com.typesafe" % "config" % "1.3.2", "com.typesafe" % "config" % "1.3.2",
@@ -52,7 +56,7 @@ libraryDependencies ++= Seq(
"fr.brouillard.oss.security.xhub" % "xhub4j-core" % "1.0.0", "fr.brouillard.oss.security.xhub" % "xhub4j-core" % "1.0.0",
"com.github.bkromhout" % "java-diff-utils" % "2.1.1", "com.github.bkromhout" % "java-diff-utils" % "2.1.1",
"org.cache2k" % "cache2k-all" % "1.0.1.Final", "org.cache2k" % "cache2k-all" % "1.0.1.Final",
"com.enragedginger" %% "akka-quartz-scheduler" % "1.6.1-akka-2.5.x" exclude("c3p0","c3p0"), "com.enragedginger" %% "akka-quartz-scheduler" % "1.6.1-akka-2.5.x" exclude ("c3p0", "c3p0"),
"net.coobird" % "thumbnailator" % "0.4.8", "net.coobird" % "thumbnailator" % "0.4.8",
"com.github.zafarkhaja" % "java-semver" % "0.9.0", "com.github.zafarkhaja" % "java-semver" % "0.9.0",
"com.nimbusds" % "oauth2-oidc-sdk" % "5.45", "com.nimbusds" % "oauth2-oidc-sdk" % "5.45",
@@ -64,18 +68,19 @@ libraryDependencies ++= Seq(
"com.wix" % "wix-embedded-mysql" % "3.0.0" % "test", "com.wix" % "wix-embedded-mysql" % "3.0.0" % "test",
"ru.yandex.qatools.embed" % "postgresql-embedded" % "2.6" % "test", "ru.yandex.qatools.embed" % "postgresql-embedded" % "2.6" % "test",
"net.i2p.crypto" % "eddsa" % "0.2.0", "net.i2p.crypto" % "eddsa" % "0.2.0",
"is.tagomor.woothee" % "woothee-java" % "1.7.0" "is.tagomor.woothee" % "woothee-java" % "1.7.0",
"org.ec4j.core" % "ec4j-core" % "0.0.1"
) )
// Compiler settings // Compiler settings
scalacOptions := Seq("-deprecation", "-language:postfixOps", "-opt:l:method") scalacOptions := Seq("-deprecation", "-language:postfixOps", "-opt:l:method", "-Xfuture")
javacOptions in compile ++= Seq("-target", "8", "-source", "8") javacOptions in compile ++= Seq("-target", "8", "-source", "8")
javaOptions in Jetty += "-Dlogback.configurationFile=/logback-dev.xml" javaOptions in Jetty += "-Dlogback.configurationFile=/logback-dev.xml"
// Test settings // Test settings
//testOptions in Test += Tests.Argument("-l", "ExternalDBTest") //testOptions in Test += Tests.Argument("-l", "ExternalDBTest")
javaOptions in Test += "-Dgitbucket.home=target/gitbucket_home_for_test" javaOptions in Test += "-Dgitbucket.home=target/gitbucket_home_for_test"
testOptions in Test += Tests.Setup( () => new java.io.File("target/gitbucket_home_for_test").mkdir() ) testOptions in Test += Tests.Setup(() => new java.io.File("target/gitbucket_home_for_test").mkdir())
fork in Test := true fork in Test := true
// Packaging options // Packaging options
@@ -85,31 +90,18 @@ packageOptions += Package.MainClass("JettyLauncher")
test in assembly := {} test in assembly := {}
assemblyMergeStrategy in assembly := { assemblyMergeStrategy in assembly := {
case PathList("META-INF", xs @ _*) => case PathList("META-INF", xs @ _*) =>
(xs map {_.toLowerCase}) match { (xs map { _.toLowerCase }) match {
case ("manifest.mf" :: Nil) => MergeStrategy.discard case ("manifest.mf" :: Nil) => MergeStrategy.discard
case _ => MergeStrategy.discard case _ => MergeStrategy.discard
} }
case x => MergeStrategy.first case x => MergeStrategy.first
} }
// JRebel
//Seq(jrebelSettings: _*)
//jrebel.webLinks += (target in webappPrepare).value
//jrebel.enabled := System.getenv().get("JREBEL") != null
javaOptions in Jetty ++= Option(System.getenv().get("JREBEL")).toSeq.flatMap { path =>
if (path.endsWith(".jar")) {
// Legacy JRebel agent
Seq("-noverify", "-XX:+UseConcMarkSweepGC", "-XX:+CMSClassUnloadingEnabled", s"-javaagent:${path}")
} else {
// New JRebel agent
Seq(s"-agentpath:${path}")
}
}
// Exclude a war file from published artifacts // Exclude a war file from published artifacts
signedArtifacts := { signedArtifacts := {
signedArtifacts.value.filterNot { case (_, file) => file.getName.endsWith(".war") || file.getName.endsWith(".war.asc") } signedArtifacts.value.filterNot {
case (_, file) => file.getName.endsWith(".war") || file.getName.endsWith(".war.asc")
}
} }
// Create executable war file // Create executable war file
@@ -145,10 +137,9 @@ executableKey := {
// include jetty classes // include jetty classes
val jettyJars = Keys.update.value select configurationFilter(name = ExecutableConfig.name) val jettyJars = Keys.update.value select configurationFilter(name = ExecutableConfig.name)
jettyJars foreach { jar => jettyJars foreach { jar =>
IO unzip (jar, temp, (name:String) => IO unzip (jar, temp, (name: String) =>
(name startsWith "javax/") || (name startsWith "javax/") ||
(name startsWith "org/") (name startsWith "org/"))
)
} }
// include original war file // include original war file
@@ -157,7 +148,7 @@ executableKey := {
// include launcher classes // include launcher classes
val classDir = (Keys.classDirectory in Compile).value val classDir = (Keys.classDirectory in Compile).value
val launchClasses = Seq("JettyLauncher.class" /*, "HttpsSupportConnector.class" */) val launchClasses = Seq("JettyLauncher.class" /*, "HttpsSupportConnector.class" */ )
launchClasses foreach { name => launchClasses foreach { name =>
IO copyFile (classDir / name, temp / name) IO copyFile (classDir / name, temp / name)
} }
@@ -165,30 +156,37 @@ executableKey := {
// include plugins // include plugins
val pluginsDir = temp / "WEB-INF" / "classes" / "plugins" val pluginsDir = temp / "WEB-INF" / "classes" / "plugins"
IO createDirectory (pluginsDir) IO createDirectory (pluginsDir)
IO copyFile(Keys.baseDirectory.value / "plugins.json", pluginsDir / "plugins.json")
val json = IO read(Keys.baseDirectory.value / "plugins.json") val plugins = IO readLines (Keys.baseDirectory.value / "src" / "main" / "resources" / "bundle-plugins.txt")
PluginsJson.getUrls(json).foreach { url => plugins.foreach { plugin =>
plugin.trim.split(":") match {
case Array(pluginId, pluginVersion) =>
val url = "https://plugins.gitbucket-community.org/releases/" +
s"gitbucket-${pluginId}-plugin/gitbucket-${pluginId}-plugin-gitbucket_${version.value}-${pluginVersion}.jar"
log info s"Download: ${url}" log info s"Download: ${url}"
IO transfer(new java.net.URL(url).openStream, pluginsDir / url.substring(url.lastIndexOf("/") + 1)) IO transfer (new java.net.URL(url).openStream, pluginsDir / url.substring(url.lastIndexOf("/") + 1))
case _ => ()
}
} }
// zip it up // zip it up
IO delete (temp / "META-INF" / "MANIFEST.MF") IO delete (temp / "META-INF" / "MANIFEST.MF")
val contentMappings = (temp.allPaths --- PathFinder(temp)).get pair { file => IO.relativizeFile(temp, file) } val contentMappings = (temp.allPaths --- PathFinder(temp)).get pair { file =>
IO.relativizeFile(temp, file)
}
val manifest = new JarManifest val manifest = new JarManifest
manifest.getMainAttributes put (AttrName.MANIFEST_VERSION, "1.0") manifest.getMainAttributes put (AttrName.MANIFEST_VERSION, "1.0")
manifest.getMainAttributes put (AttrName.MAIN_CLASS, "JettyLauncher") manifest.getMainAttributes put (AttrName.MAIN_CLASS, "JettyLauncher")
val outputFile = workDir / warName val outputFile = workDir / warName
IO jar (contentMappings.map { case (file, path) => (file, path.toString) } , outputFile, manifest) IO jar (contentMappings.map { case (file, path) => (file, path.toString) }, outputFile, manifest)
// generate checksums // generate checksums
Seq( Seq(
"md5" -> "MD5", "md5" -> "MD5",
"sha1" -> "SHA-1", "sha1" -> "SHA-1",
"sha256" -> "SHA-256" "sha256" -> "SHA-256"
) ).foreach {
.foreach { case (extension, algorithm) => case (extension, algorithm) =>
val checksumFile = workDir / (warName + "." + extension) val checksumFile = workDir / (warName + "." + extension)
Checksums generate (outputFile, checksumFile, algorithm) Checksums generate (outputFile, checksumFile, algorithm)
} }
@@ -203,7 +201,9 @@ publishTo := {
else Some("releases" at nexus + "service/local/staging/deploy/maven2") else Some("releases" at nexus + "service/local/staging/deploy/maven2")
} }
publishMavenStyle := true publishMavenStyle := true
pomIncludeRepository := { _ => false } pomIncludeRepository := { _ =>
false
}
pomExtra := ( pomExtra := (
<url>https://github.com/gitbucket/gitbucket</url> <url>https://github.com/gitbucket/gitbucket</url>
<licenses> <licenses>

View File

@@ -7,7 +7,7 @@ To determine if it was any operation, you see the `ACTION` column.
And in the case of some actions, `CONTENT` column value contains additional information. And in the case of some actions, `CONTENT` column value contains additional information.
|ACTION |CONTENT | |ACTION |CONTENT |
|---------------|-----------------| |----------------|--------------------------|
|comment |comment | |comment |comment |
|close_comment |comment | |close_comment |comment |
|reopen_comment |comment | |reopen_comment |comment |
@@ -17,6 +17,15 @@ And in the case of some actions, `CONTENT` column value contains additional info
|merge |comment | |merge |comment |
|delete_branch |branchName | |delete_branch |branchName |
|refer |issueId:title | |refer |issueId:title |
|add_label |labelName |
|delete_label |labelName |
|change_priority |oldPriority:priority |
|change_milestone|oldMilestone:milestone |
|assign |oldAssigned:assigned |
|change_title |oldTitle(CRLF)title \[1\] |
\[1\]: (CRLF) is "\r\n"
### comment ### comment
@@ -54,3 +63,27 @@ Therefore, this comment is not displayed, and not counted as a comment.
This value is saved when other issue or issue comment contains reference to the issue like `#issueId`. This value is saved when other issue or issue comment contains reference to the issue like `#issueId`.
At the same time, store id and title of the referrer issue as `id:title`. At the same time, store id and title of the referrer issue as `id:title`.
### add_label
This value is saved when users have added the label.
### delete_label
This value is saved when users have deleted the label.
### change_priority
This value is saved when users have changed the priority.
### change_milestone
This value is saved when users have changed the milestone.
### assign
This value is saved when users have assign issue/PR to user or remove the assign.
### change_title
This value is saved when users have changed the title.

View File

@@ -1,119 +0,0 @@
JRebel integration (optional)
=============================
[JRebel](https://zeroturnaround.com/software/jrebel/) is a JVM plugin that makes developing web apps much faster.
JRebel is generally able to eliminate the need for the slow "app restart" per modification of codes. Alsp it's only used during development, and doesn't change your deployed app in any way.
JRebel is not open source, but we can use it free for non-commercial use.
----
## 1. Get a JRebel license
Sign up for a [myJRebel](https://my.jrebel.com/register). You will need to create an account.
## 2. Download JRebel
Download the most recent ["nosetup" JRebel zip](https://zeroturnaround.com/software/jrebel/download/prev-releases/).
Next, unzip the downloaded file.
## 3. Activate
Follow `readme.txt` in the extracted directory to activate your downloaded JRebel.
You don't need to integrate with your IDE, since we're using sbt to do the servlet deployment.
## 4. Tell jvm where JRebel is
Fortunately, the gitbucket project is already set up to use JRebel.
You only need to tell jvm where to find the jrebel jar.
To do so, edit your shell resource file (usually `~/.bash_profile` on Mac, and `~/.bashrc` on Linux) and set the environment variable `JREBEL`.
For example, if you unzipped your JRebel download in your home directory, you would use:
```bash
export JREBEL=~/jrebel/legacy/jrebel.jar # legacy agent
export JREBEL=~/jrebel/lib/libjrebel64.dylib # new agent
```
You can choose the legacy JRebel agent or the new one.
See [the document](https://zeroturnaround.com/software/jrebel/jrebel7-agent-upgrade-cli/) for details.
Now reload your shell:
```
$ source ~/.bash_profile # on Mac
$ source ~/.bashrc # on Linux
```
## 5. See it in action!
Now you're ready to use JRebel with the gitbucket.
When you run sbt as normal, you will see a long message from JRebel, indicating it has loaded.
Here's an abbreviated version of what you will see:
```
$ ./sbt
[info] Loading project definition from /git/gitbucket/project
[info] Set current project to gitbucket (in build file:/git/gitbucket/)
>
```
You will start the servlet container slightly differently now that you're using sbt.
```
> jetty:quickstart
:
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel: #############################################################
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel: Legacy Agent 7.0.15 (201709080836)
2017-09-21 15:46:35 JRebel: (c) Copyright ZeroTurnaround AS, Estonia, Tartu.
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel: Over the last 2 days JRebel prevented
2017-09-21 15:46:35 JRebel: at least 8 redeploys/restarts saving you about 0.3 hours.
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel: Licensed to Naoki Takezoe (using myJRebel).
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel:
2017-09-21 15:46:35 JRebel: #############################################################
2017-09-21 15:46:35 JRebel:
:
> ~compile
[success] Total time: 2 s, completed 2017/09/21 15:50:06
1. Waiting for source changes... (press enter to interrupt)
```
Finally, change your code.
For example, you can change the title on `src/main/twirl/gitbucket/core/main.scala.html` like this:
```html
:
<a href="@context.path/" class="logo">
<img src="@helpers.assets("/common/images/gitbucket.svg")" style="width: 24px; height: 24px; display: inline;"/>
GitBucket
change code !!!!!!!!!!!!!!!!
<span class="header-version">@gitbucket.core.GitBucketCoreModule.getVersions.last.getVersion</span>
</a>
:
```
If JRebel is doing is correctly installed you will see a notice for you:
```
1. Waiting for source changes... (press enter to interrupt)
[info] Compiling 1 Scala source to /Users/naoki.takezoe/gitbucket/target/scala-2.12/classes...
[success] Total time: 1 s, completed 2017/09/21 15:55:40
```
And you reload browser, JRebel give notice of that it has reloaded classes:
```
2. Waiting for source changes... (press enter to interrupt)
2017-09-21 15:55:40 JRebel: Reloading class 'gitbucket.core.html.main$'.
```
## 6. Limitations
JRebel is nearly always able to eliminate the need to explicitly reload your container after a code change. However, if you change any of your routing patterns, there is nothing JRebel can do, you will have to restart by `jetty:quickstart`.

View File

@@ -8,5 +8,4 @@ Developer's Guide
* [Activity Types](activity.md) * [Activity Types](activity.md)
* [Automatic Schema Updating](auto_update.md) * [Automatic Schema Updating](auto_update.md)
* [Release Operation](release.md) * [Release Operation](release.md)
* [JRebel integration (optional)](jrebel.md)
* [Licenses](licenses.md) * [Licenses](licenses.md)

View File

@@ -47,7 +47,7 @@ $ sbt executable
For plug-in development, we have to publish the GitBucket jar file to the Maven central repository as well. At first, hit following command to publish artifacts to the sonatype OSS repository: For plug-in development, we have to publish the GitBucket jar file to the Maven central repository as well. At first, hit following command to publish artifacts to the sonatype OSS repository:
```bash ```bash
$ sbt publish-signed $ sbt publishSigned
``` ```
Then logged-in https://oss.sonatype.org/ and delete following files from the staging repository: Then logged-in https://oss.sonatype.org/ and delete following files from the staging repository:

View File

@@ -1,54 +0,0 @@
[
{
"id": "notifications",
"name": "Notifications Plugin",
"description": "Provides notifications feature on GitBucket.",
"versions": [
{
"version": "1.4.0",
"range": ">=4.19.0",
"url": "https://github.com/gitbucket/gitbucket-notifications-plugin/releases/download/1.4.0/gitbucket-notifications-plugin_2.12-1.4.0.jar"
}
],
"default": true
},
{
"id": "emoji",
"name": "Emoji Plugin",
"description": "Provides Emoji support for GitBucket.",
"versions": [
{
"version": "4.5.0",
"range": ">=4.18.0",
"url": "https://github.com/gitbucket/gitbucket-emoji-plugin/releases/download/4.5.0/gitbucket-emoji-plugin_2.12-4.5.0.jar"
}
],
"default": false
},
{
"id": "gist",
"name": "Gist Plugin",
"description": "Provides Gist feature on GitBucket.",
"versions": [
{
"version": "4.12.0",
"range": ">=4.21.0",
"url": "https://github.com/gitbucket/gitbucket-gist-plugin/releases/download/4.12.0/gitbucket-gist-plugin-assembly-4.12.0.jar"
}
],
"default": false
},
{
"id": "pages",
"name": "Pages Plugin",
"description": "Project pages for gitbucket",
"versions": [
{
"version": "1.6.0",
"range": ">=4.19.0",
"url": "https://github.com/gitbucket/gitbucket-pages-plugin/releases/download/v1.6.0/gitbucket-pages-plugin_2.12-1.6.0.jar"
}
],
"default": false
}
]

View File

@@ -6,19 +6,19 @@ import io._
object Checksums { object Checksums {
private val bufferSize = 2048 private val bufferSize = 2048
def generate(source:File, target:File, algorithm:String):Unit = def generate(source: File, target: File, algorithm: String): Unit =
sbt.IO write (target, compute(source, algorithm)) sbt.IO write (target, compute(source, algorithm))
def compute(file:File, algorithm:String):String = def compute(file: File, algorithm: String): String =
hex(raw(file, algorithm)) hex(raw(file, algorithm))
def raw(file:File, algorithm:String):Array[Byte] = def raw(file: File, algorithm: String): Array[Byte] =
(Using fileInputStream file) { is => (Using fileInputStream file) { is =>
val md = MessageDigest getInstance algorithm val md = MessageDigest getInstance algorithm
val buf = new Array[Byte](bufferSize) val buf = new Array[Byte](bufferSize)
md.reset() md.reset()
@tailrec @tailrec
def loop() { def loop(): Unit = {
val len = is read buf val len = is read buf
if (len != -1) { if (len != -1) {
md update (buf, 0, len) md update (buf, 0, len)
@@ -29,6 +29,8 @@ object Checksums {
md.digest() md.digest()
} }
def hex(bytes:Array[Byte]):String = def hex(bytes: Array[Byte]): String =
bytes map { it => "%02x" format (it.toInt & 0xff) } mkString "" bytes map { it =>
"%02x" format (it.toInt & 0xff)
} mkString ""
} }

View File

@@ -13,4 +13,3 @@ object PluginsJson {
} }
} }

View File

@@ -1 +1 @@
sbt.version=1.1.0 sbt.version=1.1.6

View File

@@ -1,9 +1,8 @@
scalacOptions ++= Seq("-unchecked", "-deprecation", "-feature") scalacOptions ++= Seq("-unchecked", "-deprecation", "-feature")
addSbtPlugin("com.geirsson" % "sbt-scalafmt" % "1.5.0")
addSbtPlugin("com.typesafe.sbt" % "sbt-twirl" % "1.3.13") addSbtPlugin("com.typesafe.sbt" % "sbt-twirl" % "1.3.13")
addSbtPlugin("com.eed3si9n" % "sbt-assembly" % "0.14.5") addSbtPlugin("com.eed3si9n" % "sbt-assembly" % "0.14.5")
//addSbtPlugin("com.earldouglas" % "xsbt-web-plugin" % "4.0.0")
//addSbtPlugin("fi.gekkio.sbtplugins" % "sbt-jrebel-plugin" % "0.10.0")
addSbtPlugin("org.scalatra.sbt" % "sbt-scalatra" % "1.0.1") addSbtPlugin("org.scalatra.sbt" % "sbt-scalatra" % "1.0.1")
addSbtPlugin("com.jsuereth" % "sbt-pgp" % "1.1.0") addSbtPlugin("com.jsuereth" % "sbt-pgp" % "1.1.0")
addSbtCoursier addSbtCoursier

View File

@@ -0,0 +1 @@
notifications:1.6.0

View File

@@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<changeSet>
<addColumn tableName="REPOSITORY">
<column name="MERGE_OPTIONS" type="varchar(200)" nullable="false" defaultValue="merge-commit,squash,rebase"/>
<column name="DEFAULT_MERGE_OPTION" type="varchar(100)" nullable="false" defaultValue="merge-commit"/>
</addColumn>
</changeSet>

View File

@@ -0,0 +1,14 @@
<?xml version="1.0" encoding="UTF-8"?>
<changeSet>
<dropForeignKeyConstraint baseTableName="RELEASE_ASSET" constraintName="IDX_RELEASE_ASSET_FK1"/>
<dropForeignKeyConstraint baseTableName="RELEASE" constraintName="IDX_RELEASE_FK0"/>
<dropPrimaryKey tableName="RELEASE" constraintName="IDX_RELEASE_PK"/>
<renameTable newTableName="RELEASE_TAG" oldTableName="RELEASE" />
<addPrimaryKey constraintName="IDX_RELEASE_TAG_PK" tableName="RELEASE_TAG" columnNames="USER_NAME, REPOSITORY_NAME, TAG"/>
<addForeignKeyConstraint constraintName="IDX_RELEASE_TAG_FK0" baseTableName="RELEASE_TAG" baseColumnNames="USER_NAME, REPOSITORY_NAME" referencedTableName="REPOSITORY" referencedColumnNames="USER_NAME, REPOSITORY_NAME"/>
<addForeignKeyConstraint constraintName="IDX_RELEASE_ASSET_FK0" baseTableName="RELEASE_ASSET" baseColumnNames="USER_NAME, REPOSITORY_NAME, TAG" referencedTableName="RELEASE_TAG" referencedColumnNames="USER_NAME, REPOSITORY_NAME, TAG"/>
</changeSet>

View File

@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<changeSet>
<createTable tableName="ACCOUNT_EXTRA_MAIL_ADDRESS">
<column name="USER_NAME" type="varchar(100)" nullable="false"/>
<column name="EXTRA_MAIL_ADDRESS" type="varchar(100)" nullable="false"/>
</createTable>
<addPrimaryKey constraintName="IDX_ACCOUNT_EXTRA_MAIL_ADDRESS_PK" tableName="ACCOUNT_EXTRA_MAIL_ADDRESS" columnNames="USER_NAME, EXTRA_MAIL_ADDRESS"/>
<addUniqueConstraint constraintName="IDX_ACCOUNT_EXTRA_MAIL_ADDRESS_1" tableName="ACCOUNT_EXTRA_MAIL_ADDRESS" columnNames="EXTRA_MAIL_ADDRESS"/>
</changeSet>

View File

@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<changeSet>
<modifyDataType columnName="PASSWORD" newDataType="varchar(200)" tableName="ACCOUNT"/>
<delete tableName="ACCOUNT_EXTRA_MAIL_ADDRESS">
<where>EXTRA_MAIL_ADDRESS = ''</where>
</delete>
</changeSet>

View File

@@ -0,0 +1,14 @@
<?xml version="1.0" encoding="UTF-8"?>
<changeSet>
<addColumn tableName="COMMIT_COMMENT">
<column name="ORIGINAL_COMMIT_ID" type="varchar(100)" nullable="true"/>
<column name="ORIGINAL_OLD_LINE" type="int" nullable="true"/>
<column name="ORIGINAL_NEW_LINE" type="int" nullable="true"/>
</addColumn>
<update tableName="COMMIT_COMMENT">
<column name="ORIGINAL_COMMIT_ID" valueComputed="COMMIT_ID"/>
<column name="ORIGINAL_OLD_LINE" valueComputed="OLD_LINE_NUMBER"/>
<column name="ORIGINAL_NEW_LINE" valueComputed="NEW_LINE_NUMBER"/>
</update>
<addNotNullConstraint columnName="ORIGINAL_COMMIT_ID" tableName="COMMIT_COMMENT" columnDataType="varchar(100)"/>
</changeSet>

View File

@@ -1,4 +1,3 @@
import java.util.EnumSet import java.util.EnumSet
import javax.servlet._ import javax.servlet._
@@ -9,28 +8,35 @@ import gitbucket.core.servlet._
import gitbucket.core.util.Directory import gitbucket.core.util.Directory
import org.scalatra._ import org.scalatra._
class ScalatraBootstrap extends LifeCycle with SystemSettingsService { class ScalatraBootstrap extends LifeCycle with SystemSettingsService {
override def init(context: ServletContext) { override def init(context: ServletContext): Unit = {
val settings = loadSystemSettings() val settings = loadSystemSettings()
if(settings.baseUrl.exists(_.startsWith("https://"))) { if (settings.baseUrl.exists(_.startsWith("https://"))) {
context.getSessionCookieConfig.setSecure(true) context.getSessionCookieConfig.setSecure(true)
} }
// Register TransactionFilter and BasicAuthenticationFilter at first // Register TransactionFilter and BasicAuthenticationFilter at first
context.addFilter("transactionFilter", new TransactionFilter) context.addFilter("transactionFilter", new TransactionFilter)
context.getFilterRegistration("transactionFilter").addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*") context
.getFilterRegistration("transactionFilter")
.addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*")
context.addFilter("gitAuthenticationFilter", new GitAuthenticationFilter) context.addFilter("gitAuthenticationFilter", new GitAuthenticationFilter)
context.getFilterRegistration("gitAuthenticationFilter").addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/git/*") context
.getFilterRegistration("gitAuthenticationFilter")
.addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/git/*")
context.addFilter("apiAuthenticationFilter", new ApiAuthenticationFilter) context.addFilter("apiAuthenticationFilter", new ApiAuthenticationFilter)
context.getFilterRegistration("apiAuthenticationFilter").addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/api/v3/*") context
.getFilterRegistration("apiAuthenticationFilter")
.addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/api/*")
// Register controllers // Register controllers
context.mount(new PreProcessController, "/*") context.mount(new PreProcessController, "/*")
context.addFilter("pluginControllerFilter", new PluginControllerFilter) context.addFilter("pluginControllerFilter", new PluginControllerFilter)
context.getFilterRegistration("pluginControllerFilter").addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*") context
.getFilterRegistration("pluginControllerFilter")
.addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*")
context.mount(new FileUploadController, "/upload") context.mount(new FileUploadController, "/upload")
@@ -51,11 +57,13 @@ class ScalatraBootstrap extends LifeCycle with SystemSettingsService {
filter.mount(new RepositorySettingsController, "/*") filter.mount(new RepositorySettingsController, "/*")
context.addFilter("compositeScalatraFilter", filter) context.addFilter("compositeScalatraFilter", filter)
context.getFilterRegistration("compositeScalatraFilter").addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*") context
.getFilterRegistration("compositeScalatraFilter")
.addMappingForUrlPatterns(EnumSet.allOf(classOf[DispatcherType]), true, "/*")
// Create GITBUCKET_HOME directory if it does not exist // Create GITBUCKET_HOME directory if it does not exist
val dir = new java.io.File(Directory.GitBucketHome) val dir = new java.io.File(Directory.GitBucketHome)
if(!dir.exists){ if (!dir.exists) {
dir.mkdirs() dir.mkdirs()
} }
} }

View File

@@ -3,39 +3,36 @@ package gitbucket.core
import io.github.gitbucket.solidbase.migration.{SqlMigration, LiquibaseMigration} import io.github.gitbucket.solidbase.migration.{SqlMigration, LiquibaseMigration}
import io.github.gitbucket.solidbase.model.{Version, Module} import io.github.gitbucket.solidbase.model.{Version, Module}
object GitBucketCoreModule extends Module("gitbucket-core", object GitBucketCoreModule
new Version("4.0.0", extends Module(
"gitbucket-core",
new Version(
"4.0.0",
new LiquibaseMigration("update/gitbucket-core_4.0.xml"), new LiquibaseMigration("update/gitbucket-core_4.0.xml"),
new SqlMigration("update/gitbucket-core_4.0.sql") new SqlMigration("update/gitbucket-core_4.0.sql")
), ),
new Version("4.1.0"), new Version("4.1.0"),
new Version("4.2.0", new Version("4.2.0", new LiquibaseMigration("update/gitbucket-core_4.2.xml")),
new LiquibaseMigration("update/gitbucket-core_4.2.xml")
),
new Version("4.2.1"), new Version("4.2.1"),
new Version("4.3.0"), new Version("4.3.0"),
new Version("4.4.0"), new Version("4.4.0"),
new Version("4.5.0"), new Version("4.5.0"),
new Version("4.6.0", new Version("4.6.0", new LiquibaseMigration("update/gitbucket-core_4.6.xml")),
new LiquibaseMigration("update/gitbucket-core_4.6.xml") new Version(
), "4.7.0",
new Version("4.7.0",
new LiquibaseMigration("update/gitbucket-core_4.7.xml"), new LiquibaseMigration("update/gitbucket-core_4.7.xml"),
new SqlMigration("update/gitbucket-core_4.7.sql") new SqlMigration("update/gitbucket-core_4.7.sql")
), ),
new Version("4.7.1"), new Version("4.7.1"),
new Version("4.8"), new Version("4.8"),
new Version("4.9.0", new Version("4.9.0", new LiquibaseMigration("update/gitbucket-core_4.9.xml")),
new LiquibaseMigration("update/gitbucket-core_4.9.xml")
),
new Version("4.10.0"), new Version("4.10.0"),
new Version("4.11.0", new Version("4.11.0", new LiquibaseMigration("update/gitbucket-core_4.11.xml")),
new LiquibaseMigration("update/gitbucket-core_4.11.xml")
),
new Version("4.12.0"), new Version("4.12.0"),
new Version("4.12.1"), new Version("4.12.1"),
new Version("4.13.0"), new Version("4.13.0"),
new Version("4.14.0", new Version(
"4.14.0",
new LiquibaseMigration("update/gitbucket-core_4.14.xml"), new LiquibaseMigration("update/gitbucket-core_4.14.xml"),
new SqlMigration("update/gitbucket-core_4.14.sql") new SqlMigration("update/gitbucket-core_4.14.sql")
), ),
@@ -49,9 +46,15 @@ object GitBucketCoreModule extends Module("gitbucket-core",
new Version("4.19.2"), new Version("4.19.2"),
new Version("4.19.3"), new Version("4.19.3"),
new Version("4.20.0"), new Version("4.20.0"),
new Version("4.21.0", new Version("4.21.0", new LiquibaseMigration("update/gitbucket-core_4.21.xml")),
new LiquibaseMigration("update/gitbucket-core_4.21.xml")
),
new Version("4.21.1"), new Version("4.21.1"),
new Version("4.21.2") new Version("4.21.2"),
) new Version("4.22.0", new LiquibaseMigration("update/gitbucket-core_4.22.xml")),
new Version("4.23.0", new LiquibaseMigration("update/gitbucket-core_4.23.xml")),
new Version("4.23.1"),
new Version("4.24.0", new LiquibaseMigration("update/gitbucket-core_4.24.xml")),
new Version("4.24.1"),
new Version("4.25.0", new LiquibaseMigration("update/gitbucket-core_4.25.xml")),
new Version("4.26.0"),
new Version("4.27.0", new LiquibaseMigration("update/gitbucket-core_4.27.xml"))
)

View File

@@ -6,13 +6,14 @@ import gitbucket.core.util.RepositoryName
* https://developer.github.com/v3/repos/#get-branch * https://developer.github.com/v3/repos/#get-branch
* https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection * https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection
*/ */
case class ApiBranch( case class ApiBranch(name: String, commit: ApiBranchCommit, protection: ApiBranchProtection)(
name: String, repositoryName: RepositoryName
commit: ApiBranchCommit, ) extends FieldSerializable {
protection: ApiBranchProtection)(repositoryName:RepositoryName) extends FieldSerializable { def _links =
def _links = Map( Map(
"self" -> ApiPath(s"/api/v3/repos/${repositoryName.fullName}/branches/${name}"), "self" -> ApiPath(s"/api/v3/repos/${repositoryName.fullName}/branches/${name}"),
"html" -> ApiPath(s"/${repositoryName.fullName}/tree/${name}")) "html" -> ApiPath(s"/${repositoryName.fullName}/tree/${name}")
)
} }
case class ApiBranchCommit(sha: String) case class ApiBranchCommit(sha: String)

View File

@@ -4,17 +4,22 @@ import gitbucket.core.service.ProtectedBranchService
import org.json4s._ import org.json4s._
/** https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection */ /** https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection */
case class ApiBranchProtection(enabled: Boolean, required_status_checks: Option[ApiBranchProtection.Status]){ case class ApiBranchProtection(enabled: Boolean, required_status_checks: Option[ApiBranchProtection.Status]) {
def status: ApiBranchProtection.Status = required_status_checks.getOrElse(ApiBranchProtection.statusNone) def status: ApiBranchProtection.Status = required_status_checks.getOrElse(ApiBranchProtection.statusNone)
} }
object ApiBranchProtection{ object ApiBranchProtection {
/** form for enabling-and-disabling-branch-protection */ /** form for enabling-and-disabling-branch-protection */
case class EnablingAndDisabling(protection: ApiBranchProtection) case class EnablingAndDisabling(protection: ApiBranchProtection)
def apply(info: ProtectedBranchService.ProtectedBranchInfo): ApiBranchProtection = ApiBranchProtection( def apply(info: ProtectedBranchService.ProtectedBranchInfo): ApiBranchProtection =
ApiBranchProtection(
enabled = info.enabled, enabled = info.enabled,
required_status_checks = Some(Status(EnforcementLevel(info.enabled && info.contexts.nonEmpty, info.includeAdministrators), info.contexts))) required_status_checks = Some(
Status(EnforcementLevel(info.enabled && info.contexts.nonEmpty, info.includeAdministrators), info.contexts)
)
)
val statusNone = Status(Off, Seq.empty) val statusNone = Status(Off, Seq.empty)
case class Status(enforcement_level: EnforcementLevel, contexts: Seq[String]) case class Status(enforcement_level: EnforcementLevel, contexts: Seq[String])
sealed class EnforcementLevel(val name: String) sealed class EnforcementLevel(val name: String)
@@ -22,25 +27,28 @@ object ApiBranchProtection{
case object NonAdmins extends EnforcementLevel("non_admins") case object NonAdmins extends EnforcementLevel("non_admins")
case object Everyone extends EnforcementLevel("everyone") case object Everyone extends EnforcementLevel("everyone")
object EnforcementLevel { object EnforcementLevel {
def apply(enabled: Boolean, includeAdministrators: Boolean): EnforcementLevel = if(enabled){ def apply(enabled: Boolean, includeAdministrators: Boolean): EnforcementLevel =
if(includeAdministrators){ if (enabled) {
if (includeAdministrators) {
Everyone Everyone
}else{ } else {
NonAdmins NonAdmins
} }
}else{ } else {
Off Off
} }
} }
implicit val enforcementLevelSerializer = new CustomSerializer[EnforcementLevel](format => ( implicit val enforcementLevelSerializer = new CustomSerializer[EnforcementLevel](
format =>
(
{ {
case JString("off") => Off case JString("off") => Off
case JString("non_admins") => NonAdmins case JString("non_admins") => NonAdmins
case JString("everyone") => Everyone case JString("everyone") => Everyone
}, }, {
{
case x: EnforcementLevel => JString(x.name) case x: EnforcementLevel => JString(x.name)
} }
)) )
)
} }

View File

@@ -2,7 +2,6 @@ package gitbucket.core.api
import gitbucket.core.model.{Account, CommitState, CommitStatus} import gitbucket.core.model.{Account, CommitState, CommitStatus}
/** /**
* https://developer.github.com/v3/repos/statuses/#get-the-combined-status-for-a-specific-ref * https://developer.github.com/v3/repos/statuses/#get-the-combined-status-for-a-specific-ref
*/ */
@@ -11,15 +10,22 @@ case class ApiCombinedCommitStatus(
sha: String, sha: String,
total_count: Int, total_count: Int,
statuses: Iterable[ApiCommitStatus], statuses: Iterable[ApiCommitStatus],
repository: ApiRepository){ repository: ApiRepository
) {
// val commit_url = ApiPath(s"/api/v3/repos/${repository.full_name}/${sha}") // val commit_url = ApiPath(s"/api/v3/repos/${repository.full_name}/${sha}")
val url = ApiPath(s"/api/v3/repos/${repository.full_name}/commits/${sha}/status") val url = ApiPath(s"/api/v3/repos/${repository.full_name}/commits/${sha}/status")
} }
object ApiCombinedCommitStatus { object ApiCombinedCommitStatus {
def apply(sha:String, statuses: Iterable[(CommitStatus, Account)], repository:ApiRepository): ApiCombinedCommitStatus = ApiCombinedCommitStatus( def apply(
sha: String,
statuses: Iterable[(CommitStatus, Account)],
repository: ApiRepository
): ApiCombinedCommitStatus =
ApiCombinedCommitStatus(
state = CommitState.combine(statuses.map(_._1.state).toSet).name, state = CommitState.combine(statuses.map(_._1.state).toSet).name,
sha = sha, sha = sha,
total_count= statuses.size, total_count = statuses.size,
statuses = statuses.map{ case (s, a)=> ApiCommitStatus(s, ApiUser(a)) }, statuses = statuses.map { case (s, a) => ApiCommitStatus(s, ApiUser(a)) },
repository = repository) repository = repository
)
} }

View File

@@ -5,25 +5,32 @@ import gitbucket.core.util.RepositoryName
import java.util.Date import java.util.Date
/** /**
* https://developer.github.com/v3/issues/comments/ * https://developer.github.com/v3/issues/comments/
*/ */
case class ApiComment( case class ApiComment(id: Int, user: ApiUser, body: String, created_at: Date, updated_at: Date)(
id: Int, repositoryName: RepositoryName,
user: ApiUser, issueId: Int,
body: String, isPullRequest: Boolean
created_at: Date, ) {
updated_at: Date)(repositoryName: RepositoryName, issueId: Int, isPullRequest: Boolean){ val html_url = ApiPath(
val html_url = ApiPath(s"/${repositoryName.fullName}/${if(isPullRequest){ "pull" }else{ "issues" }}/${issueId}#comment-${id}") s"/${repositoryName.fullName}/${if (isPullRequest) { "pull" } else { "issues" }}/${issueId}#comment-${id}"
)
} }
object ApiComment{ object ApiComment {
def apply(comment: IssueComment, repositoryName: RepositoryName, issueId: Int, user: ApiUser, isPullRequest: Boolean): ApiComment = def apply(
comment: IssueComment,
repositoryName: RepositoryName,
issueId: Int,
user: ApiUser,
isPullRequest: Boolean
): ApiComment =
ApiComment( ApiComment(
id = comment.commentId, id = comment.commentId,
user = user, user = user,
body = comment.content, body = comment.content,
created_at = comment.registeredDate, created_at = comment.registeredDate,
updated_at = comment.updatedDate)(repositoryName, issueId, isPullRequest) updated_at = comment.updatedDate
)(repositoryName, issueId, isPullRequest)
} }

View File

@@ -20,20 +20,22 @@ case class ApiCommit(
removed: List[String], removed: List[String],
modified: List[String], modified: List[String],
author: ApiPersonIdent, author: ApiPersonIdent,
committer: ApiPersonIdent)(repositoryName:RepositoryName, urlIsHtmlUrl: Boolean) extends FieldSerializable{ committer: ApiPersonIdent
val url = if(urlIsHtmlUrl){ )(repositoryName: RepositoryName, urlIsHtmlUrl: Boolean)
extends FieldSerializable {
val url = if (urlIsHtmlUrl) {
ApiPath(s"/${repositoryName.fullName}/commit/${id}") ApiPath(s"/${repositoryName.fullName}/commit/${id}")
}else{ } else {
ApiPath(s"/api/v3/${repositoryName.fullName}/commits/${id}") ApiPath(s"/api/v3/${repositoryName.fullName}/commits/${id}")
} }
val html_url = if(urlIsHtmlUrl){ val html_url = if (urlIsHtmlUrl) {
None None
}else{ } else {
Some(ApiPath(s"/${repositoryName.fullName}/commit/${id}")) Some(ApiPath(s"/${repositoryName.fullName}/commit/${id}"))
} }
} }
object ApiCommit{ object ApiCommit {
def apply(git: Git, repositoryName: RepositoryName, commit: CommitInfo, urlIsHtmlUrl: Boolean = false): ApiCommit = { def apply(git: Git, repositoryName: RepositoryName, commit: CommitInfo, urlIsHtmlUrl: Boolean = false): ApiCommit = {
val diffs = JGitUtil.getDiffs(git, None, commit.id, false, false) val diffs = JGitUtil.getDiffs(git, None, commit.id, false, false)
ApiCommit( ApiCommit(
@@ -53,5 +55,6 @@ object ApiCommit{
committer = ApiPersonIdent.committer(commit) committer = ApiPersonIdent.committer(commit)
)(repositoryName, urlIsHtmlUrl) )(repositoryName, urlIsHtmlUrl)
} }
def forWebhookPayload(git: Git, repositoryName: RepositoryName, commit: CommitInfo): ApiCommit = apply(git, repositoryName, commit, true) def forWebhookPayload(git: Git, repositoryName: RepositoryName, commit: CommitInfo): ApiCommit =
apply(git, repositoryName, commit, true)
} }

View File

@@ -4,7 +4,6 @@ import gitbucket.core.api.ApiCommitListItem._
import gitbucket.core.util.JGitUtil.CommitInfo import gitbucket.core.util.JGitUtil.CommitInfo
import gitbucket.core.util.RepositoryName import gitbucket.core.util.RepositoryName
/** /**
* https://developer.github.com/v3/repos/commits/ * https://developer.github.com/v3/repos/commits/
*/ */
@@ -13,12 +12,14 @@ case class ApiCommitListItem(
commit: Commit, commit: Commit,
author: Option[ApiUser], author: Option[ApiUser],
committer: Option[ApiUser], committer: Option[ApiUser],
parents: Seq[Parent])(repositoryName: RepositoryName) { parents: Seq[Parent]
)(repositoryName: RepositoryName) {
val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}") val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}")
} }
object ApiCommitListItem { object ApiCommitListItem {
def apply(commit: CommitInfo, repositoryName: RepositoryName): ApiCommitListItem = ApiCommitListItem( def apply(commit: CommitInfo, repositoryName: RepositoryName): ApiCommitListItem =
ApiCommitListItem(
sha = commit.id, sha = commit.id,
commit = Commit( commit = Commit(
message = commit.fullMessage, message = commit.fullMessage,
@@ -27,16 +28,17 @@ object ApiCommitListItem {
)(commit.id, repositoryName), )(commit.id, repositoryName),
author = None, author = None,
committer = None, committer = None,
parents = commit.parents.map(Parent(_)(repositoryName)))(repositoryName) parents = commit.parents.map(Parent(_)(repositoryName))
)(repositoryName)
case class Parent(sha: String)(repositoryName: RepositoryName){ case class Parent(sha: String)(repositoryName: RepositoryName) {
val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}") val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}")
} }
case class Commit( case class Commit(message: String, author: ApiPersonIdent, committer: ApiPersonIdent)(
message: String, sha: String,
author: ApiPersonIdent, repositoryName: RepositoryName
committer: ApiPersonIdent)(sha:String, repositoryName: RepositoryName) { ) {
val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/git/commits/${sha}") val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/git/commits/${sha}")
} }
} }

View File

@@ -5,7 +5,6 @@ import gitbucket.core.util.RepositoryName
import java.util.Date import java.util.Date
/** /**
* https://developer.github.com/v3/repos/statuses/#create-a-status * https://developer.github.com/v3/repos/statuses/#create-a-status
* https://developer.github.com/v3/repos/statuses/#list-statuses-for-a-specific-ref * https://developer.github.com/v3/repos/statuses/#list-statuses-for-a-specific-ref
@@ -23,14 +22,14 @@ case class ApiCommitStatus(
val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}/statuses") val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/commits/${sha}/statuses")
} }
object ApiCommitStatus { object ApiCommitStatus {
def apply(status: CommitStatus, creator:ApiUser): ApiCommitStatus = ApiCommitStatus( def apply(status: CommitStatus, creator: ApiUser): ApiCommitStatus =
ApiCommitStatus(
created_at = status.registeredDate, created_at = status.registeredDate,
updated_at = status.updatedDate, updated_at = status.updatedDate,
state = status.state.name, state = status.state.name,
target_url = status.targetUrl, target_url = status.targetUrl,
description= status.description, description = status.description,
id = status.commitStatusId, id = status.commitStatusId,
context = status.context, context = status.context,
creator = creator creator = creator

View File

@@ -51,20 +51,25 @@ object ApiCommits {
patch: String patch: String
) )
def apply(
def apply(repositoryName: RepositoryName, commitInfo: CommitInfo, diffs: Seq[DiffInfo], author: Account, committer: Account, repositoryName: RepositoryName,
commentCount: Int): ApiCommits = { commitInfo: CommitInfo,
diffs: Seq[DiffInfo],
author: Account,
committer: Account,
commentCount: Int
): ApiCommits = {
val files = diffs.map { diff => val files = diffs.map { diff =>
var additions = 0 var additions = 0
var deletions = 0 var deletions = 0
diff.patch.getOrElse("").split("\n").foreach { line => diff.patch.getOrElse("").split("\n").foreach { line =>
if(line.startsWith("+")) additions = additions + 1 if (line.startsWith("+")) additions = additions + 1
if(line.startsWith("-")) deletions = deletions + 1 if (line.startsWith("-")) deletions = deletions + 1
} }
File( File(
filename = if(diff.changeType == ChangeType.DELETE){ diff.oldPath } else { diff.newPath }, filename = if (diff.changeType == ChangeType.DELETE) { diff.oldPath } else { diff.newPath },
additions = additions, additions = additions,
deletions = deletions, deletions = deletions,
changes = additions + deletions, changes = additions + deletions,
@@ -75,12 +80,12 @@ object ApiCommits {
case ChangeType.RENAME => "renamed" case ChangeType.RENAME => "renamed"
case ChangeType.COPY => "copied" case ChangeType.COPY => "copied"
}, },
raw_url = if(diff.changeType == ChangeType.DELETE){ raw_url = if (diff.changeType == ChangeType.DELETE) {
ApiPath(s"/${repositoryName.fullName}/raw/${commitInfo.parents.head}/${diff.oldPath}") ApiPath(s"/${repositoryName.fullName}/raw/${commitInfo.parents.head}/${diff.oldPath}")
} else { } else {
ApiPath(s"/${repositoryName.fullName}/raw/${commitInfo.id}/${diff.newPath}") ApiPath(s"/${repositoryName.fullName}/raw/${commitInfo.id}/${diff.newPath}")
}, },
blob_url = if(diff.changeType == ChangeType.DELETE){ blob_url = if (diff.changeType == ChangeType.DELETE) {
ApiPath(s"/${repositoryName.fullName}/blob/${commitInfo.parents.head}/${diff.oldPath}") ApiPath(s"/${repositoryName.fullName}/blob/${commitInfo.parents.head}/${diff.oldPath}")
} else { } else {
ApiPath(s"/${repositoryName.fullName}/blob/${commitInfo.id}/${diff.newPath}") ApiPath(s"/${repositoryName.fullName}/blob/${commitInfo.id}/${diff.newPath}")

View File

@@ -11,18 +11,29 @@ case class ApiContents(
path: String, path: String,
sha: String, sha: String,
content: Option[String], content: Option[String],
encoding: Option[String])(repositoryName: RepositoryName){ encoding: Option[String]
)(repositoryName: RepositoryName) {
val download_url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/raw/${sha}/${path}") val download_url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/raw/${sha}/${path}")
} }
object ApiContents{ object ApiContents {
def apply(fileInfo: FileInfo, repositoryName: RepositoryName, content: Option[Array[Byte]]): ApiContents = { def apply(fileInfo: FileInfo, repositoryName: RepositoryName, content: Option[Array[Byte]]): ApiContents = {
if(fileInfo.isDirectory) { if (fileInfo.isDirectory) {
ApiContents("dir", fileInfo.name, fileInfo.path, fileInfo.commitId, None, None)(repositoryName) ApiContents("dir", fileInfo.name, fileInfo.path, fileInfo.commitId, None, None)(repositoryName)
} else { } else {
content.map(arr => content
ApiContents("file", fileInfo.name, fileInfo.path, fileInfo.commitId, Some(Base64.getEncoder.encodeToString(arr)), Some("base64"))(repositoryName) .map(
).getOrElse(ApiContents("file", fileInfo.name, fileInfo.path, fileInfo.commitId, None, None)(repositoryName)) arr =>
ApiContents(
"file",
fileInfo.name,
fileInfo.path,
fileInfo.commitId,
Some(Base64.getEncoder.encodeToString(arr)),
Some("base64")
)(repositoryName)
)
.getOrElse(ApiContents("file", fileInfo.name, fileInfo.path, fileInfo.commitId, None, None)(repositoryName))
} }
} }
} }

View File

@@ -1,5 +1,3 @@
package gitbucket.core.api package gitbucket.core.api
case class ApiError( case class ApiError(message: String, documentation_url: Option[String] = None)
message: String,
documentation_url: Option[String] = None)

View File

@@ -5,7 +5,6 @@ import gitbucket.core.util.RepositoryName
import java.util.Date import java.util.Date
/** /**
* https://developer.github.com/v3/issues/ * https://developer.github.com/v3/issues/
*/ */
@@ -13,33 +12,39 @@ case class ApiIssue(
number: Int, number: Int,
title: String, title: String,
user: ApiUser, user: ApiUser,
// labels, labels: List[ApiLabel],
state: String, state: String,
created_at: Date, created_at: Date,
updated_at: Date, updated_at: Date,
body: String)(repositoryName: RepositoryName, isPullRequest: Boolean){ body: String
)(repositoryName: RepositoryName, isPullRequest: Boolean) {
val id = 0 // dummy id
val comments_url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/issues/${number}/comments") val comments_url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/issues/${number}/comments")
val html_url = ApiPath(s"/${repositoryName.fullName}/${if(isPullRequest){ "pull" }else{ "issues" }}/${number}") val html_url = ApiPath(s"/${repositoryName.fullName}/${if (isPullRequest) { "pull" } else { "issues" }}/${number}")
val pull_request = if (isPullRequest) { val pull_request = if (isPullRequest) {
Some(Map( Some(
Map(
"url" -> ApiPath(s"/api/v3/repos/${repositoryName.fullName}/pulls/${number}"), "url" -> ApiPath(s"/api/v3/repos/${repositoryName.fullName}/pulls/${number}"),
"html_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}") "html_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}")
// "diff_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}.diff"), // "diff_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}.diff"),
// "patch_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}.patch") // "patch_url" -> ApiPath(s"/${repositoryName.fullName}/pull/${number}.patch")
)) )
)
} else { } else {
None None
} }
} }
object ApiIssue{ object ApiIssue {
def apply(issue: Issue, repositoryName: RepositoryName, user: ApiUser): ApiIssue = def apply(issue: Issue, repositoryName: RepositoryName, user: ApiUser, labels: List[ApiLabel]): ApiIssue =
ApiIssue( ApiIssue(
number = issue.issueId, number = issue.issueId,
title = issue.title, title = issue.title,
user = user, user = user,
state = if(issue.closed){ "closed" }else{ "open" }, labels = labels,
state = if (issue.closed) { "closed" } else { "open" },
body = issue.content.getOrElse(""), body = issue.content.getOrElse(""),
created_at = issue.registeredDate, created_at = issue.registeredDate,
updated_at = issue.updatedDate)(repositoryName, issue.isPullRequest) updated_at = issue.updatedDate
)(repositoryName, issue.isPullRequest)
} }

View File

@@ -6,14 +6,12 @@ import gitbucket.core.util.RepositoryName
/** /**
* https://developer.github.com/v3/issues/labels/ * https://developer.github.com/v3/issues/labels/
*/ */
case class ApiLabel( case class ApiLabel(name: String, color: String)(repositoryName: RepositoryName) {
name: String,
color: String)(repositoryName: RepositoryName){
var url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/labels/${name}") var url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/labels/${name}")
} }
object ApiLabel{ object ApiLabel {
def apply(label:Label, repositoryName: RepositoryName): ApiLabel = def apply(label: Label, repositoryName: RepositoryName): ApiLabel =
ApiLabel( ApiLabel(
name = label.labelName, name = label.labelName,
color = label.color color = label.color

View File

@@ -4,22 +4,11 @@ import gitbucket.core.util.JGitUtil.CommitInfo
import java.util.Date import java.util.Date
case class ApiPersonIdent(name: String, email: String, date: Date)
case class ApiPersonIdent(
name: String,
email: String,
date: Date)
object ApiPersonIdent { object ApiPersonIdent {
def author(commit: CommitInfo): ApiPersonIdent = def author(commit: CommitInfo): ApiPersonIdent =
ApiPersonIdent( ApiPersonIdent(name = commit.authorName, email = commit.authorEmailAddress, date = commit.authorTime)
name = commit.authorName,
email = commit.authorEmailAddress,
date = commit.authorTime)
def committer(commit: CommitInfo): ApiPersonIdent = def committer(commit: CommitInfo): ApiPersonIdent =
ApiPersonIdent( ApiPersonIdent(name = commit.committerName, email = commit.committerEmailAddress, date = commit.commitTime)
name = commit.committerName,
email = commit.committerEmailAddress,
date = commit.commitTime)
} }

View File

@@ -0,0 +1,17 @@
package gitbucket.core.api
import gitbucket.core.plugin.{PluginRegistry, PluginInfo}
case class ApiPlugin(
id: String,
name: String,
version: String,
description: String,
jarFileName: String
)
object ApiPlugin {
def apply(plugin: PluginInfo): ApiPlugin = {
ApiPlugin(plugin.pluginId, plugin.pluginName, plugin.pluginVersion, plugin.description, plugin.pluginJar.getName)
}
}

View File

@@ -20,7 +20,10 @@ case class ApiPullRequest(
title: String, title: String,
body: String, body: String,
user: ApiUser, user: ApiUser,
assignee: Option[ApiUser]){ labels: List[ApiLabel],
assignee: Option[ApiUser]
) {
val id = 0 // dummy id
val html_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}") val html_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}")
//val diff_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}.diff") //val diff_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}.diff")
//val patch_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}.patch") //val patch_url = ApiPath(s"${base.repo.html_url.path}/pull/${number}.patch")
@@ -33,13 +36,14 @@ case class ApiPullRequest(
val statuses_url = ApiPath(s"${base.repo.url.path}/statuses/${head.sha}") val statuses_url = ApiPath(s"${base.repo.url.path}/statuses/${head.sha}")
} }
object ApiPullRequest{ object ApiPullRequest {
def apply( def apply(
issue: Issue, issue: Issue,
pullRequest: PullRequest, pullRequest: PullRequest,
headRepo: ApiRepository, headRepo: ApiRepository,
baseRepo: ApiRepository, baseRepo: ApiRepository,
user: ApiUser, user: ApiUser,
labels: List[ApiLabel],
assignee: Option[ApiUser], assignee: Option[ApiUser],
mergedComment: Option[(IssueComment, Account)] mergedComment: Option[(IssueComment, Account)]
): ApiPullRequest = ): ApiPullRequest =
@@ -48,14 +52,8 @@ object ApiPullRequest{
state = if (issue.closed) "closed" else "open", state = if (issue.closed) "closed" else "open",
updated_at = issue.updatedDate, updated_at = issue.updatedDate,
created_at = issue.registeredDate, created_at = issue.registeredDate,
head = Commit( head = Commit(sha = pullRequest.commitIdTo, ref = pullRequest.requestBranch, repo = headRepo)(issue.userName),
sha = pullRequest.commitIdTo, base = Commit(sha = pullRequest.commitIdFrom, ref = pullRequest.branch, repo = baseRepo)(issue.userName),
ref = pullRequest.requestBranch,
repo = headRepo)(issue.userName),
base = Commit(
sha = pullRequest.commitIdFrom,
ref = pullRequest.branch,
repo = baseRepo)(issue.userName),
mergeable = None, // TODO: need check mergeable. mergeable = None, // TODO: need check mergeable.
merged = mergedComment.isDefined, merged = mergedComment.isDefined,
merged_at = mergedComment.map { case (comment, _) => comment.registeredDate }, merged_at = mergedComment.map { case (comment, _) => comment.registeredDate },
@@ -63,14 +61,12 @@ object ApiPullRequest{
title = issue.title, title = issue.title,
body = issue.content.getOrElse(""), body = issue.content.getOrElse(""),
user = user, user = user,
labels = labels,
assignee = assignee assignee = assignee
) )
case class Commit( case class Commit(sha: String, ref: String, repo: ApiRepository)(baseOwner: String) {
sha: String, val label = if (baseOwner == repo.owner.login) { ref } else { s"${repo.owner.login}:${ref}" }
ref: String,
repo: ApiRepository)(baseOwner:String){
val label = if( baseOwner == repo.owner.login ){ ref } else { s"${repo.owner.login}:${ref}" }
val user = repo.owner val user = repo.owner
} }

View File

@@ -17,10 +17,11 @@ case class ApiPullRequestReviewComment(
commit_id: String, // "0d1a26e67d8f5eaf1f6ba5c57fc3c7d91ac0fd1c", commit_id: String, // "0d1a26e67d8f5eaf1f6ba5c57fc3c7d91ac0fd1c",
// "original_commit_id": "0d1a26e67d8f5eaf1f6ba5c57fc3c7d91ac0fd1c", // "original_commit_id": "0d1a26e67d8f5eaf1f6ba5c57fc3c7d91ac0fd1c",
user: ApiUser, user: ApiUser,
body: String, // "Maybe you should use more emojji on this line.", body: String, // "Maybe you should use more emoji on this line.",
created_at: Date, // "2015-05-05T23:40:27Z", created_at: Date, // "2015-05-05T23:40:27Z",
updated_at: Date // "2015-05-05T23:40:27Z", updated_at: Date // "2015-05-05T23:40:27Z",
)(repositoryName:RepositoryName, issueId: Int) extends FieldSerializable { )(repositoryName: RepositoryName, issueId: Int)
extends FieldSerializable {
// "url": "https://api.github.com/repos/baxterthehacker/public-repo/pulls/comments/29724692", // "url": "https://api.github.com/repos/baxterthehacker/public-repo/pulls/comments/29724692",
val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/pulls/comments/${id}") val url = ApiPath(s"/api/v3/repos/${repositoryName.fullName}/pulls/comments/${id}")
// "html_url": "https://github.com/baxterthehacker/public-repo/pull/1#discussion_r29724692", // "html_url": "https://github.com/baxterthehacker/public-repo/pull/1#discussion_r29724692",
@@ -44,11 +45,17 @@ case class ApiPullRequestReviewComment(
val _links = Map( val _links = Map(
"self" -> Map("href" -> url), "self" -> Map("href" -> url),
"html" -> Map("href" -> html_url), "html" -> Map("href" -> html_url),
"pull_request" -> Map("href" -> pull_request_url)) "pull_request" -> Map("href" -> pull_request_url)
)
} }
object ApiPullRequestReviewComment{ object ApiPullRequestReviewComment {
def apply(comment: CommitComment, commentedUser: ApiUser, repositoryName: RepositoryName, issueId: Int): ApiPullRequestReviewComment = def apply(
comment: CommitComment,
commentedUser: ApiUser,
repositoryName: RepositoryName,
issueId: Int
): ApiPullRequestReviewComment =
new ApiPullRequestReviewComment( new ApiPullRequestReviewComment(
id = comment.commentId, id = comment.commentId,
path = comment.fileName.getOrElse(""), path = comment.fileName.getOrElse(""),

View File

@@ -5,7 +5,5 @@ import gitbucket.core.model.Account
case class ApiPusher(name: String, email: String) case class ApiPusher(name: String, email: String)
object ApiPusher { object ApiPusher {
def apply(user: Account): ApiPusher = ApiPusher( def apply(user: Account): ApiPusher = ApiPusher(name = user.userName, email = user.mailAddress)
name = user.userName,
email = user.mailAddress)
} }

View File

@@ -3,7 +3,6 @@ package gitbucket.core.api
import gitbucket.core.model.{Account, Repository} import gitbucket.core.model.{Account, Repository}
import gitbucket.core.service.RepositoryService.RepositoryInfo import gitbucket.core.service.RepositoryService.RepositoryInfo
// https://developer.github.com/v3/repos/ // https://developer.github.com/v3/repos/
case class ApiRepository( case class ApiRepository(
name: String, name: String,
@@ -13,10 +12,12 @@ case class ApiRepository(
forks: Int, forks: Int,
`private`: Boolean, `private`: Boolean,
default_branch: String, default_branch: String,
owner: ApiUser)(urlIsHtmlUrl: Boolean) { owner: ApiUser
)(urlIsHtmlUrl: Boolean) {
val id = 0 // dummy id
val forks_count = forks val forks_count = forks
val watchers_count = watchers val watchers_count = watchers
val url = if(urlIsHtmlUrl){ val url = if (urlIsHtmlUrl) {
ApiPath(s"/${full_name}") ApiPath(s"/${full_name}")
} else { } else {
ApiPath(s"/api/v3/repos/${full_name}") ApiPath(s"/api/v3/repos/${full_name}")
@@ -27,13 +28,14 @@ case class ApiRepository(
val ssh_url = Some(SshPath(s":${full_name}.git")) val ssh_url = Some(SshPath(s":${full_name}.git"))
} }
object ApiRepository{ object ApiRepository {
def apply( def apply(
repository: Repository, repository: Repository,
owner: ApiUser, owner: ApiUser,
forkedCount: Int =0, forkedCount: Int = 0,
watchers: Int = 0, watchers: Int = 0,
urlIsHtmlUrl: Boolean = false): ApiRepository = urlIsHtmlUrl: Boolean = false
): ApiRepository =
ApiRepository( ApiRepository(
name = repository.repositoryName, name = repository.repositoryName,
full_name = s"${repository.userName}/${repository.repositoryName}", full_name = s"${repository.userName}/${repository.repositoryName}",
@@ -46,13 +48,13 @@ object ApiRepository{
)(urlIsHtmlUrl) )(urlIsHtmlUrl)
def apply(repositoryInfo: RepositoryInfo, owner: ApiUser): ApiRepository = def apply(repositoryInfo: RepositoryInfo, owner: ApiUser): ApiRepository =
ApiRepository(repositoryInfo.repository, owner, forkedCount=repositoryInfo.forkedCount) ApiRepository(repositoryInfo.repository, owner, forkedCount = repositoryInfo.forkedCount)
def apply(repositoryInfo: RepositoryInfo, owner: Account): ApiRepository = def apply(repositoryInfo: RepositoryInfo, owner: Account): ApiRepository =
this(repositoryInfo.repository, ApiUser(owner)) this(repositoryInfo.repository, ApiUser(owner))
def forWebhookPayload(repositoryInfo: RepositoryInfo, owner: ApiUser): ApiRepository = def forWebhookPayload(repositoryInfo: RepositoryInfo, owner: ApiUser): ApiRepository =
ApiRepository(repositoryInfo.repository, owner, forkedCount=repositoryInfo.forkedCount, urlIsHtmlUrl=true) ApiRepository(repositoryInfo.repository, owner, forkedCount = repositoryInfo.forkedCount, urlIsHtmlUrl = true)
def forDummyPayload(owner: ApiUser): ApiRepository = def forDummyPayload(owner: ApiUser): ApiRepository =
ApiRepository( ApiRepository(

View File

@@ -4,13 +4,8 @@ import gitbucket.core.model.Account
import java.util.Date import java.util.Date
case class ApiUser(login: String, email: String, `type`: String, site_admin: Boolean, created_at: Date) {
case class ApiUser( val id = 0 // dummy id
login: String,
email: String,
`type`: String,
site_admin: Boolean,
created_at: Date) {
val url = ApiPath(s"/api/v3/users/${login}") val url = ApiPath(s"/api/v3/users/${login}")
val html_url = ApiPath(s"/${login}") val html_url = ApiPath(s"/${login}")
val avatar_url = ApiPath(s"/${login}/_avatar") val avatar_url = ApiPath(s"/${login}/_avatar")
@@ -25,12 +20,11 @@ case class ApiUser(
// val received_events_url = ApiPath(s"/api/v3/users/${login}/received_events") // val received_events_url = ApiPath(s"/api/v3/users/${login}/received_events")
} }
object ApiUser {
object ApiUser{
def apply(user: Account): ApiUser = ApiUser( def apply(user: Account): ApiUser = ApiUser(
login = user.userName, login = user.userName,
email = user.mailAddress, email = user.mailAddress,
`type` = if(user.isGroupAccount){ "Organization" } else { "User" }, `type` = if (user.isGroupAccount) { "Organization" } else { "User" },
site_admin = user.isAdmin, site_admin = user.isAdmin,
created_at = user.registeredDate created_at = user.registeredDate
) )

View File

@@ -9,10 +9,10 @@ case class CreateALabel(
color: String color: String
) { ) {
def isValid: Boolean = { def isValid: Boolean = {
name.length<=100 && name.length <= 100 &&
!name.startsWith("_") && !name.startsWith("_") &&
!name.startsWith("-") && !name.startsWith("-") &&
color.length==6 && color.length == 6 &&
color.matches("[a-fA-F0-9+_.]+") color.matches("[a-fA-F0-9+_.]+")
} }
} }

View File

@@ -8,4 +8,5 @@ case class CreateAnIssue(
body: Option[String], body: Option[String],
assignees: List[String], assignees: List[String],
milestone: Option[Int], milestone: Option[Int],
labels: List[String]) labels: List[String]
)

View File

@@ -15,10 +15,13 @@ object JsonFormat {
val parserISO = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss'Z'") val parserISO = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss'Z'")
val jsonFormats = Serialization.formats(NoTypeHints) + new CustomSerializer[Date](format => val jsonFormats = Serialization.formats(NoTypeHints) + new CustomSerializer[Date](
format =>
( (
{ case JString(s) => Try(Date.from(Instant.parse(s))).getOrElse(throw new MappingException("Can't convert " + s + " to Date")) }, {
{ case x: Date => JString(OffsetDateTime.ofInstant(x.toInstant, ZoneId.of("UTC")).format(parserISO)) } case JString(s) =>
Try(Date.from(Instant.parse(s))).getOrElse(throw new MappingException("Can't convert " + s + " to Date"))
}, { case x: Date => JString(OffsetDateTime.ofInstant(x.toInstant, ZoneId.of("UTC")).format(parserISO)) }
) )
) + FieldSerializer[ApiUser]() + ) + FieldSerializer[ApiUser]() +
FieldSerializer[ApiPullRequest]() + FieldSerializer[ApiPullRequest]() +
@@ -41,19 +44,31 @@ object JsonFormat {
FieldSerializer[ApiCommits.File]() + FieldSerializer[ApiCommits.File]() +
ApiBranchProtection.enforcementLevelSerializer ApiBranchProtection.enforcementLevelSerializer
def apiPathSerializer(c: Context) = new CustomSerializer[ApiPath](_ => ({ def apiPathSerializer(c: Context) =
new CustomSerializer[ApiPath](
_ =>
({
case JString(s) if s.startsWith(c.baseUrl) => ApiPath(s.substring(c.baseUrl.length)) case JString(s) if s.startsWith(c.baseUrl) => ApiPath(s.substring(c.baseUrl.length))
case JString(s) => throw new MappingException("Can't convert " + s + " to ApiPath") case JString(s) => throw new MappingException("Can't convert " + s + " to ApiPath")
}, { }, {
case ApiPath(path) => JString(c.baseUrl + path) case ApiPath(path) => JString(c.baseUrl + path)
})) })
)
def sshPathSerializer(c: Context) = new CustomSerializer[SshPath](_ => ({ def sshPathSerializer(c: Context) =
case JString(s) if c.sshUrl.exists(sshUrl => s.startsWith(sshUrl)) => SshPath(s.substring(c.sshUrl.get.length)) new CustomSerializer[SshPath](
_ =>
({
case JString(s) if c.sshUrl.exists(sshUrl => s.startsWith(sshUrl)) =>
SshPath(s.substring(c.sshUrl.get.length))
case JString(s) => throw new MappingException("Can't convert " + s + " to ApiPath") case JString(s) => throw new MappingException("Can't convert " + s + " to ApiPath")
}, { }, {
case SshPath(path) => c.sshUrl.map { sshUrl => JString(sshUrl + path) } getOrElse JNothing case SshPath(path) =>
})) c.sshUrl.map { sshUrl =>
JString(sshUrl + path)
} getOrElse JNothing
})
)
/** /**
* convert object to json string * convert object to json string

View File

@@ -1,8 +1,10 @@
package gitbucket.core.controller package gitbucket.core.controller
import java.io.File
import gitbucket.core.account.html import gitbucket.core.account.html
import gitbucket.core.helper import gitbucket.core.helper
import gitbucket.core.model.{AccountWebHook, GroupMember, RepositoryWebHook, Role, WebHook, WebHookContentType} import gitbucket.core.model._
import gitbucket.core.plugin.PluginRegistry import gitbucket.core.plugin.PluginRegistry
import gitbucket.core.service._ import gitbucket.core.service._
import gitbucket.core.service.WebHookService._ import gitbucket.core.service.WebHookService._
@@ -16,77 +18,141 @@ import org.scalatra.i18n.Messages
import org.scalatra.BadRequest import org.scalatra.BadRequest
import org.scalatra.forms._ import org.scalatra.forms._
class AccountController extends AccountControllerBase class AccountController
with AccountService with RepositoryService with ActivityService with WikiService with LabelsService with SshKeyService extends AccountControllerBase
with OneselfAuthenticator with UsersAuthenticator with GroupManagerAuthenticator with ReadableUsersAuthenticator with AccountService
with AccessTokenService with WebHookService with PrioritiesService with RepositoryCreationService with RepositoryService
with ActivityService
with WikiService
with LabelsService
with SshKeyService
with OneselfAuthenticator
with UsersAuthenticator
with GroupManagerAuthenticator
with ReadableUsersAuthenticator
with AccessTokenService
with WebHookService
with PrioritiesService
with RepositoryCreationService
trait AccountControllerBase extends AccountManagementControllerBase { trait AccountControllerBase extends AccountManagementControllerBase {
self: AccountService with RepositoryService with ActivityService with WikiService with LabelsService with SshKeyService self: AccountService
with OneselfAuthenticator with UsersAuthenticator with GroupManagerAuthenticator with ReadableUsersAuthenticator with RepositoryService
with AccessTokenService with WebHookService with PrioritiesService with RepositoryCreationService => with ActivityService
with WikiService
with LabelsService
with SshKeyService
with OneselfAuthenticator
with UsersAuthenticator
with GroupManagerAuthenticator
with ReadableUsersAuthenticator
with AccessTokenService
with WebHookService
with PrioritiesService
with RepositoryCreationService =>
case class AccountNewForm(userName: String, password: String, fullName: String, mailAddress: String, case class AccountNewForm(
description: Option[String], url: Option[String], fileId: Option[String]) userName: String,
password: String,
fullName: String,
mailAddress: String,
extraMailAddresses: List[String],
description: Option[String],
url: Option[String],
fileId: Option[String]
)
case class AccountEditForm(password: Option[String], fullName: String, mailAddress: String, case class AccountEditForm(
description: Option[String], url: Option[String], fileId: Option[String], clearImage: Boolean) password: Option[String],
fullName: String,
mailAddress: String,
extraMailAddresses: List[String],
description: Option[String],
url: Option[String],
fileId: Option[String],
clearImage: Boolean
)
case class SshKeyForm(title: String, publicKey: String) case class SshKeyForm(title: String, publicKey: String)
case class PersonalTokenForm(note: String) case class PersonalTokenForm(note: String)
val newForm = mapping( val newForm = mapping(
"userName" -> trim(label("User name" , text(required, maxlength(100), identifier, uniqueUserName, reservedNames))), "userName" -> trim(label("User name", text(required, maxlength(100), identifier, uniqueUserName, reservedNames))),
"password" -> trim(label("Password" , text(required, maxlength(20), password))), "password" -> trim(label("Password", text(required, maxlength(20), password))),
"fullName" -> trim(label("Full Name" , text(required, maxlength(100)))), "fullName" -> trim(label("Full Name", text(required, maxlength(100)))),
"mailAddress" -> trim(label("Mail Address" , text(required, maxlength(100), uniqueMailAddress()))), "mailAddress" -> trim(label("Mail Address", text(required, maxlength(100), uniqueMailAddress()))),
"description" -> trim(label("bio" , optional(text()))), "extraMailAddresses" -> list(
"url" -> trim(label("URL" , optional(text(maxlength(200))))), trim(label("Additional Mail Address", text(maxlength(100), uniqueExtraMailAddress())))
"fileId" -> trim(label("File ID" , optional(text()))) ),
"description" -> trim(label("bio", optional(text()))),
"url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID", optional(text())))
)(AccountNewForm.apply) )(AccountNewForm.apply)
val editForm = mapping( val editForm = mapping(
"password" -> trim(label("Password" , optional(text(maxlength(20), password)))), "password" -> trim(label("Password", optional(text(maxlength(20), password)))),
"fullName" -> trim(label("Full Name" , text(required, maxlength(100)))), "fullName" -> trim(label("Full Name", text(required, maxlength(100)))),
"mailAddress" -> trim(label("Mail Address" , text(required, maxlength(100), uniqueMailAddress("userName")))), "mailAddress" -> trim(label("Mail Address", text(required, maxlength(100), uniqueMailAddress("userName")))),
"description" -> trim(label("bio" , optional(text()))), "extraMailAddresses" -> list(
"url" -> trim(label("URL" , optional(text(maxlength(200))))), trim(label("Additional Mail Address", text(maxlength(100), uniqueExtraMailAddress("userName"))))
"fileId" -> trim(label("File ID" , optional(text()))), ),
"clearImage" -> trim(label("Clear image" , boolean())) "description" -> trim(label("bio", optional(text()))),
"url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID", optional(text()))),
"clearImage" -> trim(label("Clear image", boolean()))
)(AccountEditForm.apply) )(AccountEditForm.apply)
val sshKeyForm = mapping( val sshKeyForm = mapping(
"title" -> trim(label("Title", text(required, maxlength(100)))), "title" -> trim(label("Title", text(required, maxlength(100)))),
"publicKey" -> trim2(label("Key" , text(required, validPublicKey))) "publicKey" -> trim2(label("Key", text(required, validPublicKey)))
)(SshKeyForm.apply) )(SshKeyForm.apply)
val personalTokenForm = mapping( val personalTokenForm = mapping(
"note" -> trim(label("Token", text(required, maxlength(100)))) "note" -> trim(label("Token", text(required, maxlength(100))))
)(PersonalTokenForm.apply) )(PersonalTokenForm.apply)
case class NewGroupForm(groupName: String, description: Option[String], url: Option[String], fileId: Option[String], members: String) case class NewGroupForm(
case class EditGroupForm(groupName: String, description: Option[String], url: Option[String], fileId: Option[String], members: String, clearImage: Boolean) groupName: String,
description: Option[String],
url: Option[String],
fileId: Option[String],
members: String
)
case class EditGroupForm(
groupName: String,
description: Option[String],
url: Option[String],
fileId: Option[String],
members: String,
clearImage: Boolean
)
val newGroupForm = mapping( val newGroupForm = mapping(
"groupName" -> trim(label("Group name" ,text(required, maxlength(100), identifier, uniqueUserName, reservedNames))), "groupName" -> trim(label("Group name", text(required, maxlength(100), identifier, uniqueUserName, reservedNames))),
"description" -> trim(label("Group description", optional(text()))), "description" -> trim(label("Group description", optional(text()))),
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), "url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID" ,optional(text()))), "fileId" -> trim(label("File ID", optional(text()))),
"members" -> trim(label("Members" ,text(required, members))) "members" -> trim(label("Members", text(required, members)))
)(NewGroupForm.apply) )(NewGroupForm.apply)
val editGroupForm = mapping( val editGroupForm = mapping(
"groupName" -> trim(label("Group name" ,text(required, maxlength(100), identifier))), "groupName" -> trim(label("Group name", text(required, maxlength(100), identifier))),
"description" -> trim(label("Group description", optional(text()))), "description" -> trim(label("Group description", optional(text()))),
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), "url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID" ,optional(text()))), "fileId" -> trim(label("File ID", optional(text()))),
"members" -> trim(label("Members" ,text(required, members))), "members" -> trim(label("Members", text(required, members))),
"clearImage" -> trim(label("Clear image" ,boolean())) "clearImage" -> trim(label("Clear image", boolean()))
)(EditGroupForm.apply) )(EditGroupForm.apply)
case class RepositoryCreationForm(owner: String, name: String, description: Option[String], isPrivate: Boolean, initOption: String, sourceUrl: Option[String]) case class RepositoryCreationForm(
owner: String,
name: String,
description: Option[String],
isPrivate: Boolean,
initOption: String,
sourceUrl: Option[String]
)
case class ForkRepositoryForm(owner: String, name: String) case class ForkRepositoryForm(owner: String, name: String)
val newRepositoryForm = mapping( val newRepositoryForm = mapping(
@@ -110,9 +176,15 @@ trait AccountControllerBase extends AccountManagementControllerBase {
)(AccountForm.apply) )(AccountForm.apply)
// for account web hook url addition. // for account web hook url addition.
case class AccountWebHookForm(url: String, events: Set[WebHook.Event], ctype: WebHookContentType, token: Option[String]) case class AccountWebHookForm(
url: String,
events: Set[WebHook.Event],
ctype: WebHookContentType,
token: Option[String]
)
def accountWebHookForm(update:Boolean) = mapping( def accountWebHookForm(update: Boolean) =
mapping(
"url" -> trim(label("url", text(required, accountWebHook(update)))), "url" -> trim(label("url", text(required, accountWebHook(update)))),
"events" -> accountWebhookEvents, "events" -> accountWebhookEvents,
"ctype" -> label("ctype", text()), "ctype" -> label("ctype", text()),
@@ -120,13 +192,14 @@ trait AccountControllerBase extends AccountManagementControllerBase {
)( )(
(url, events, ctype, token) => AccountWebHookForm(url, events, WebHookContentType.valueOf(ctype), token) (url, events, ctype, token) => AccountWebHookForm(url, events, WebHookContentType.valueOf(ctype), token)
) )
/** /**
* Provides duplication check for web hook url. duplicated from RepositorySettingsController.scala * Provides duplication check for web hook url. duplicated from RepositorySettingsController.scala
*/ */
private def accountWebHook(needExists: Boolean): Constraint = new Constraint(){ private def accountWebHook(needExists: Boolean): Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(getAccountWebHook(params("userName"), value).isDefined != needExists){ if (getAccountWebHook(params("userName"), value).isDefined != needExists) {
Some(if(needExists){ Some(if (needExists) {
"URL had not been registered yet." "URL had not been registered yet."
} else { } else {
"URL had been registered already." "URL had been registered already."
@@ -136,48 +209,68 @@ trait AccountControllerBase extends AccountManagementControllerBase {
} }
} }
private def accountWebhookEvents = new ValueType[Set[WebHook.Event]]{ private def accountWebhookEvents = new ValueType[Set[WebHook.Event]] {
def convert(name: String, params: Map[String, Seq[String]], messages: Messages): Set[WebHook.Event] = { def convert(name: String, params: Map[String, Seq[String]], messages: Messages): Set[WebHook.Event] = {
WebHook.Event.values.flatMap { t => WebHook.Event.values.flatMap { t =>
params.optionValue(name + "." + t.name).map(_ => t) params.optionValue(name + "." + t.name).map(_ => t)
}.toSet }.toSet
} }
def validate(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] = def validate(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] =
if(convert(name, params, messages).isEmpty){ if (convert(name, params, messages).isEmpty) {
Seq(name -> messages("error.required").format(name)) Seq(name -> messages("error.required").format(name))
} else { } else {
Nil Nil
} }
} }
/** /**
* Displays user information. * Displays user information.
*/ */
get("/:userName") { get("/:userName") {
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName).map { account => getAccountByUserName(userName).map { account =>
val extraMailAddresses = getAccountExtraMailAddresses(userName)
params.getOrElse("tab", "repositories") match { params.getOrElse("tab", "repositories") match {
// Public Activity // Public Activity
case "activity" => case "activity" =>
gitbucket.core.account.html.activity(account, gitbucket.core.account.html.activity(
if(account.isGroupAccount) Nil else getGroupsByUserName(userName), account,
getActivitiesByUser(userName, true)) if (account.isGroupAccount) Nil else getGroupsByUserName(userName),
getActivitiesByUser(userName, true),
extraMailAddresses
)
// Members // Members
case "members" if(account.isGroupAccount) => { case "members" if (account.isGroupAccount) => {
val members = getGroupMembers(account.userName) val members = getGroupMembers(account.userName)
gitbucket.core.account.html.members(account, members, gitbucket.core.account.html.members(
context.loginAccount.exists(x => members.exists { member => member.userName == x.userName && member.isManager })) account,
members,
extraMailAddresses,
context.loginAccount.exists(
x =>
members.exists { member =>
member.userName == x.userName && member.isManager
}
)
)
} }
// Repositories // Repositories
case _ => { case _ => {
val members = getGroupMembers(account.userName) val members = getGroupMembers(account.userName)
gitbucket.core.account.html.repositories(account, gitbucket.core.account.html.repositories(
if(account.isGroupAccount) Nil else getGroupsByUserName(userName), account,
if (account.isGroupAccount) Nil else getGroupsByUserName(userName),
getVisibleRepositories(context.loginAccount, Some(userName)), getVisibleRepositories(context.loginAccount, Some(userName)),
context.loginAccount.exists(x => members.exists { member => member.userName == x.userName && member.isManager })) extraMailAddresses,
context.loginAccount.exists(
x =>
members.exists { member =>
member.userName == x.userName && member.isManager
}
)
)
} }
} }
} getOrElse NotFound() } getOrElse NotFound()
@@ -189,21 +282,33 @@ trait AccountControllerBase extends AccountManagementControllerBase {
helper.xml.feed(getActivitiesByUser(userName, true)) helper.xml.feed(getActivitiesByUser(userName, true))
} }
get("/:userName/_avatar"){ get("/:userName.keys") {
val keys = getPublicKeys(params("userName"))
contentType = "text/plain; charset=utf-8"
keys.map(_.publicKey).mkString("", "\n", "\n")
}
get("/:userName/_avatar") {
val userName = params("userName") val userName = params("userName")
contentType = "image/png" contentType = "image/png"
getAccountByUserName(userName).flatMap{ account => getAccountByUserName(userName)
.flatMap { account =>
response.setDateHeader("Last-Modified", account.updatedDate.getTime) response.setDateHeader("Last-Modified", account.updatedDate.getTime)
account.image.map{ image => account.image
Some(RawData(FileUtil.getMimeType(image), new java.io.File(getUserUploadDir(userName), image))) .map { image =>
}.getOrElse{ Some(
RawData(FileUtil.getMimeType(image), new File(getUserUploadDir(userName), FileUtil.checkFilename(image)))
)
}
.getOrElse {
if (account.isGroupAccount) { if (account.isGroupAccount) {
TextAvatarUtil.textGroupAvatar(account.fullName) TextAvatarUtil.textGroupAvatar(account.fullName)
} else { } else {
TextAvatarUtil.textAvatar(account.fullName) TextAvatarUtil.textAvatar(account.fullName)
} }
} }
}.getOrElse{ }
.getOrElse {
response.setHeader("Cache-Control", "max-age=3600") response.setHeader("Cache-Control", "max-age=3600")
Thread.currentThread.getContextClassLoader.getResourceAsStream("noimage.png") Thread.currentThread.getContextClassLoader.getResourceAsStream("noimage.png")
} }
@@ -212,36 +317,39 @@ trait AccountControllerBase extends AccountManagementControllerBase {
get("/:userName/_edit")(oneselfOnly { get("/:userName/_edit")(oneselfOnly {
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName).map { x => getAccountByUserName(userName).map { x =>
html.edit(x, flash.get("info"), flash.get("error")) val extraMails = getAccountExtraMailAddresses(userName)
html.edit(x, extraMails, flash.get("info"), flash.get("error"))
} getOrElse NotFound() } getOrElse NotFound()
}) })
post("/:userName/_edit", editForm)(oneselfOnly { form => post("/:userName/_edit", editForm)(oneselfOnly { form =>
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName).map { account => getAccountByUserName(userName).map {
updateAccount(account.copy( account =>
password = form.password.map(sha1).getOrElse(account.password), updateAccount(
account.copy(
password = form.password.map(pbkdf2_sha256).getOrElse(account.password),
fullName = form.fullName, fullName = form.fullName,
mailAddress = form.mailAddress, mailAddress = form.mailAddress,
description = form.description, description = form.description,
url = form.url)) url = form.url
)
)
updateImage(userName, form.fileId, form.clearImage) updateImage(userName, form.fileId, form.clearImage)
updateAccountExtraMailAddresses(userName, form.extraMailAddresses.filter(_ != ""))
flash += "info" -> "Account information has been updated." flash += "info" -> "Account information has been updated."
redirect(s"/${userName}/_edit") redirect(s"/${userName}/_edit")
} getOrElse NotFound() } getOrElse NotFound()
}) })
get("/captures/(.*)".r) {
multiParams("captures").head
}
get("/:userName/_delete")(oneselfOnly { get("/:userName/_delete")(oneselfOnly {
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName, true).map { account => getAccountByUserName(userName, true).map {
if(isLastAdministrator(account)){ account =>
if (isLastAdministrator(account)) {
flash += "error" -> "Account can't be removed because this is last one administrator." flash += "error" -> "Account can't be removed because this is last one administrator."
redirect(s"/${userName}/_edit") redirect(s"/${userName}/_edit")
} else { } else {
@@ -290,9 +398,9 @@ trait AccountControllerBase extends AccountManagementControllerBase {
getAccountByUserName(userName).map { x => getAccountByUserName(userName).map { x =>
var tokens = getAccessTokens(x.userName) var tokens = getAccessTokens(x.userName)
val generatedToken = flash.get("generatedToken") match { val generatedToken = flash.get("generatedToken") match {
case Some((tokenId:Int, token:String)) => { case Some((tokenId: Int, token: String)) => {
val gt = tokens.find(_.accessTokenId == tokenId) val gt = tokens.find(_.accessTokenId == tokenId)
gt.map{ t => gt.map { t =>
tokens = tokens.filterNot(_ == t) tokens = tokens.filterNot(_ == t)
(t, token) (t, token)
} }
@@ -363,7 +471,8 @@ trait AccountControllerBase extends AccountManagementControllerBase {
get("/:userName/_hooks/edit")(oneselfOnly { get("/:userName/_hooks/edit")(oneselfOnly {
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName).flatMap { account => getAccountByUserName(userName).flatMap { account =>
getAccountWebHook(userName, params("url")).map { case (webhook, events) => getAccountWebHook(userName, params("url")).map {
case (webhook, events) =>
html.edithook(webhook, events, account, false) html.edithook(webhook, events, account, false)
} }
} getOrElse NotFound() } getOrElse NotFound()
@@ -390,7 +499,9 @@ trait AccountControllerBase extends AccountManagementControllerBase {
import org.apache.http.util.EntityUtils import org.apache.http.util.EntityUtils
import scala.concurrent.ExecutionContext.Implicits.global import scala.concurrent.ExecutionContext.Implicits.global
def _headers(h: Array[org.apache.http.Header]): Array[Array[String]] = h.map { h => Array(h.getName, h.getValue) } def _headers(h: Array[org.apache.http.Header]): Array[Array[String]] = h.map { h =>
Array(h.getName, h.getValue)
}
val userName = params("userName") val userName = params("userName")
val url = params("url") val url = params("url")
@@ -404,31 +515,49 @@ trait AccountControllerBase extends AccountManagementControllerBase {
val (webHook, json, reqFuture, resFuture) = callWebHook(WebHook.Push, List(dummyWebHookInfo), dummyPayload).head val (webHook, json, reqFuture, resFuture) = callWebHook(WebHook.Push, List(dummyWebHookInfo), dummyPayload).head
val toErrorMap: PartialFunction[Throwable, Map[String,String]] = { val toErrorMap: PartialFunction[Throwable, Map[String, String]] = {
case e: java.net.UnknownHostException => Map("error"-> ("Unknown host " + e.getMessage)) case e: java.net.UnknownHostException => Map("error" -> ("Unknown host " + e.getMessage))
case e: java.lang.IllegalArgumentException => Map("error"-> ("invalid url")) case e: java.lang.IllegalArgumentException => Map("error" -> ("invalid url"))
case e: org.apache.http.client.ClientProtocolException => Map("error"-> ("invalid url")) case e: org.apache.http.client.ClientProtocolException => Map("error" -> ("invalid url"))
case NonFatal(e) => Map("error"-> (e.getClass + " "+ e.getMessage)) case NonFatal(e) => Map("error" -> (e.getClass + " " + e.getMessage))
} }
contentType = formats("json") contentType = formats("json")
org.json4s.jackson.Serialization.write(Map( org.json4s.jackson.Serialization.write(
Map(
"url" -> url, "url" -> url,
"request" -> Await.result(reqFuture.map(req => Map( "request" -> Await.result(
reqFuture
.map(
req =>
Map(
"headers" -> _headers(req.getAllHeaders), "headers" -> _headers(req.getAllHeaders),
"payload" -> json "payload" -> json
)).recover(toErrorMap), 20 seconds), )
"response" -> Await.result(resFuture.map(res => Map( )
.recover(toErrorMap),
20 seconds
),
"response" -> Await.result(
resFuture
.map(
res =>
Map(
"status" -> res.getStatusLine(), "status" -> res.getStatusLine(),
"body" -> EntityUtils.toString(res.getEntity()), "body" -> EntityUtils.toString(res.getEntity()),
"headers" -> _headers(res.getAllHeaders()) "headers" -> _headers(res.getAllHeaders())
)).recover(toErrorMap), 20 seconds) )
)) )
.recover(toErrorMap),
20 seconds
)
)
)
}) })
get("/register"){ get("/register") {
if(context.settings.allowAccountRegistration){ if (context.settings.allowAccountRegistration) {
if(context.loginAccount.isDefined){ if (context.loginAccount.isDefined) {
redirect("/") redirect("/")
} else { } else {
html.register() html.register()
@@ -436,10 +565,19 @@ trait AccountControllerBase extends AccountManagementControllerBase {
} else NotFound() } else NotFound()
} }
post("/register", newForm){ form => post("/register", newForm) { form =>
if(context.settings.allowAccountRegistration){ if (context.settings.allowAccountRegistration) {
createAccount(form.userName, sha1(form.password), form.fullName, form.mailAddress, false, form.description, form.url) createAccount(
form.userName,
pbkdf2_sha256(form.password),
form.fullName,
form.mailAddress,
false,
form.description,
form.url
)
updateImage(form.userName, form.fileId, false) updateImage(form.userName, form.fileId, false)
updateAccountExtraMailAddresses(form.userName, form.extraMailAddresses.filter(_ != ""))
redirect("/signin") redirect("/signin")
} else NotFound() } else NotFound()
} }
@@ -450,17 +588,23 @@ trait AccountControllerBase extends AccountManagementControllerBase {
post("/groups/new", newGroupForm)(usersOnly { form => post("/groups/new", newGroupForm)(usersOnly { form =>
createGroup(form.groupName, form.description, form.url) createGroup(form.groupName, form.description, form.url)
updateGroupMembers(form.groupName, form.members.split(",").map { updateGroupMembers(
form.groupName,
form.members
.split(",")
.map {
_.split(":") match { _.split(":") match {
case Array(userName, isManager) => (userName, isManager.toBoolean) case Array(userName, isManager) => (userName, isManager.toBoolean)
} }
}.toList) }
.toList
)
updateImage(form.groupName, form.fileId, false) updateImage(form.groupName, form.fileId, false)
redirect(s"/${form.groupName}") redirect(s"/${form.groupName}")
}) })
get("/:groupName/_editgroup")(managersOnly { get("/:groupName/_editgroup")(managersOnly {
defining(params("groupName")){ groupName => defining(params("groupName")) { groupName =>
getAccountByUserName(groupName, true).map { account => getAccountByUserName(groupName, true).map { account =>
html.editgroup(account, getGroupMembers(groupName), flash.get("info")) html.editgroup(account, getGroupMembers(groupName), flash.get("info"))
} getOrElse NotFound() } getOrElse NotFound()
@@ -468,7 +612,8 @@ trait AccountControllerBase extends AccountManagementControllerBase {
}) })
get("/:groupName/_deletegroup")(managersOnly { get("/:groupName/_deletegroup")(managersOnly {
defining(params("groupName")){ groupName => defining(params("groupName")) {
groupName =>
// Remove from GROUP_MEMBER // Remove from GROUP_MEMBER
updateGroupMembers(groupName, Nil) updateGroupMembers(groupName, Nil)
// Disable group // Disable group
@@ -487,11 +632,18 @@ trait AccountControllerBase extends AccountManagementControllerBase {
}) })
post("/:groupName/_editgroup", editGroupForm)(managersOnly { form => post("/:groupName/_editgroup", editGroupForm)(managersOnly { form =>
defining(params("groupName"), form.members.split(",").map { defining(
params("groupName"),
form.members
.split(",")
.map {
_.split(":") match { _.split(":") match {
case Array(userName, isManager) => (userName, isManager.toBoolean) case Array(userName, isManager) => (userName, isManager.toBoolean)
} }
}.toList){ case (groupName, members) => }
.toList
) {
case (groupName, members) =>
getAccountByUserName(groupName, true).map { account => getAccountByUserName(groupName, true).map { account =>
updateGroup(groupName, form.description, form.url, false) updateGroup(groupName, form.description, form.url, false)
@@ -525,9 +677,17 @@ trait AccountControllerBase extends AccountManagementControllerBase {
* Create new repository. * Create new repository.
*/ */
post("/new", newRepositoryForm)(usersOnly { form => post("/new", newRepositoryForm)(usersOnly { form =>
LockUtil.lock(s"${form.owner}/${form.name}"){ LockUtil.lock(s"${form.owner}/${form.name}") {
if(getRepository(form.owner, form.name).isEmpty){ if (getRepository(form.owner, form.name).isEmpty) {
createRepository(context.loginAccount.get, form.owner, form.name, form.description, form.isPrivate, form.initOption, form.sourceUrl) createRepository(
context.loginAccount.get,
form.owner,
form.name,
form.description,
form.isPrivate,
form.initOption,
form.sourceUrl
)
} }
} }
@@ -536,7 +696,7 @@ trait AccountControllerBase extends AccountManagementControllerBase {
}) })
get("/:owner/:repository/fork")(readableUsersOnly { repository => get("/:owner/:repository/fork")(readableUsersOnly { repository =>
if(repository.repository.options.allowFork){ if (repository.repository.options.allowFork) {
val loginAccount = context.loginAccount.get val loginAccount = context.loginAccount.get
val loginUserName = loginAccount.userName val loginUserName = loginAccount.userName
val groups = getGroupsByUserName(loginUserName) val groups = getGroupsByUserName(loginUserName)
@@ -544,11 +704,16 @@ trait AccountControllerBase extends AccountManagementControllerBase {
case _: List[String] => case _: List[String] =>
val managerPermissions = groups.map { group => val managerPermissions = groups.map { group =>
val members = getGroupMembers(group) val members = getGroupMembers(group)
context.loginAccount.exists(x => members.exists { member => member.userName == x.userName && member.isManager }) context.loginAccount.exists(
x =>
members.exists { member =>
member.userName == x.userName && member.isManager
}
)
} }
helper.html.forkrepository( helper.html.forkrepository(
repository, repository,
(groups zip managerPermissions).toMap (groups zip managerPermissions).sortBy(_._1)
) )
case _ => redirect(s"/${loginUserName}") case _ => redirect(s"/${loginUserName}")
} }
@@ -556,7 +721,7 @@ trait AccountControllerBase extends AccountManagementControllerBase {
}) })
post("/:owner/:repository/fork", accountForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/fork", accountForm)(readableUsersOnly { (form, repository) =>
if(repository.repository.options.allowFork){ if (repository.repository.options.allowFork) {
val loginAccount = context.loginAccount.get val loginAccount = context.loginAccount.get
val loginUserName = loginAccount.userName val loginUserName = loginAccount.userName
val accountName = form.accountName val accountName = form.accountName
@@ -574,38 +739,45 @@ trait AccountControllerBase extends AccountManagementControllerBase {
} else BadRequest() } else BadRequest()
}) })
private def existsAccount: Constraint = new Constraint(){ private def existsAccount: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(getAccountByUserName(value).isEmpty) Some("User or group does not exist.") else None if (getAccountByUserNameIgnoreCase(value).isEmpty) Some("User or group does not exist.") else None
} }
private def uniqueRepository: Constraint = new Constraint(){ private def uniqueRepository: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = { override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
for { for {
userName <- params.optionValue("owner") userName <- params.optionValue("owner")
_ <- getRepositoryNamesOfUser(userName).find(_ == value) _ <- getRepositoryNamesOfUser(userName).find(_.equalsIgnoreCase(value))
} yield { } yield {
"Repository already exists." "Repository already exists."
} }
} }
} }
private def members: Constraint = new Constraint(){ private def members: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
if(value.split(",").exists { if (value.split(",").exists {
_.split(":") match { case Array(userName, isManager) => isManager.toBoolean } _.split(":") match { case Array(userName, isManager) => isManager.toBoolean }
}) None else Some("Must select one manager at least.") }) None
else Some("Must select one manager at least.")
} }
} }
private def validPublicKey: Constraint = new Constraint(){ private def validPublicKey: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = SshUtil.str2PublicKey(value) match { override def validate(name: String, value: String, messages: Messages): Option[String] =
SshUtil.str2PublicKey(value) match {
case Some(_) if !getAllKeys().exists(_.publicKey == value) => None case Some(_) if !getAllKeys().exists(_.publicKey == value) => None
case _ => Some("Key is invalid.") case _ => Some("Key is invalid.")
} }
} }
private def validAccountName: Constraint = new Constraint(){ private def validAccountName: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
getAccountByUserName(value) match { getAccountByUserName(value) match {
case Some(_) => None case Some(_) => None

View File

@@ -10,6 +10,9 @@ import gitbucket.core.util.Implicits._
import gitbucket.core.util.JGitUtil._ import gitbucket.core.util.JGitUtil._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
import gitbucket.core.util._ import gitbucket.core.util._
import gitbucket.core.plugin.PluginRegistry
import gitbucket.core.servlet.Database
import gitbucket.core.model.Profile.profile.blockingApi._
import gitbucket.core.view.helpers.{isRenderable, renderMarkup} import gitbucket.core.view.helpers.{isRenderable, renderMarkup}
import org.eclipse.jgit.api.Git import org.eclipse.jgit.api.Git
import org.eclipse.jgit.revwalk.RevWalk import org.eclipse.jgit.revwalk.RevWalk
@@ -19,7 +22,8 @@ import scala.collection.JavaConverters._
import scala.concurrent.Await import scala.concurrent.Await
import scala.concurrent.duration.Duration import scala.concurrent.duration.Duration
class ApiController extends ApiControllerBase class ApiController
extends ApiControllerBase
with RepositoryService with RepositoryService
with AccountService with AccountService
with ProtectedBranchService with ProtectedBranchService
@@ -76,7 +80,7 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* https://developer.github.com/v3/#root-endpoint * https://developer.github.com/v3/#root-endpoint
*/ */
get("/api/v3/") { get("/api/v3") {
JsonFormat(ApiEndPoint()) JsonFormat(ApiEndPoint())
} }
@@ -103,66 +107,89 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/repos/#list-organization-repositories * https://developer.github.com/v3/repos/#list-organization-repositories
*/ */
get("/api/v3/orgs/:orgName/repos") { get("/api/v3/orgs/:orgName/repos") {
JsonFormat(getVisibleRepositories(context.loginAccount, Some(params("orgName"))).map{ r => ApiRepository(r, getAccountByUserName(r.owner).get)}) JsonFormat(getVisibleRepositories(context.loginAccount, Some(params("orgName"))).map { r =>
ApiRepository(r, getAccountByUserName(r.owner).get)
})
} }
/** /**
* https://developer.github.com/v3/repos/#list-user-repositories * https://developer.github.com/v3/repos/#list-user-repositories
*/ */
get("/api/v3/users/:userName/repos") { get("/api/v3/users/:userName/repos") {
JsonFormat(getVisibleRepositories(context.loginAccount, Some(params("userName"))).map{ r => ApiRepository(r, getAccountByUserName(r.owner).get)}) JsonFormat(getVisibleRepositories(context.loginAccount, Some(params("userName"))).map { r =>
ApiRepository(r, getAccountByUserName(r.owner).get)
})
} }
/* /*
* https://developer.github.com/v3/repos/branches/#list-branches * https://developer.github.com/v3/repos/branches/#list-branches
*/ */
get ("/api/v3/repos/:owner/:repo/branches")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/branches")(referrersOnly { repository =>
JsonFormat(JGitUtil.getBranches( JsonFormat(
JGitUtil
.getBranches(
owner = repository.owner, owner = repository.owner,
name = repository.name, name = repository.name,
defaultBranch = repository.repository.defaultBranch, defaultBranch = repository.repository.defaultBranch,
origin = repository.repository.originUserName.isEmpty origin = repository.repository.originUserName.isEmpty
).map { br => )
.map { br =>
ApiBranchForList(br.name, ApiBranchCommit(br.commitId)) ApiBranchForList(br.name, ApiBranchCommit(br.commitId))
}) }
)
}) })
/** /**
* https://developer.github.com/v3/repos/branches/#get-branch * https://developer.github.com/v3/repos/branches/#get-branch
*/ */
get ("/api/v3/repos/:owner/:repo/branches/*")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/branches/*")(referrersOnly { repository =>
//import gitbucket.core.api._ //import gitbucket.core.api._
(for{ (for {
branch <- params.get("splat") if repository.branchList.contains(branch) branch <- params.get("splat") if repository.branchList.contains(branch)
br <- getBranches(repository.owner, repository.name, repository.repository.defaultBranch, repository.repository.originUserName.isEmpty).find(_.name == branch) br <- getBranches(
repository.owner,
repository.name,
repository.repository.defaultBranch,
repository.repository.originUserName.isEmpty
).find(_.name == branch)
} yield { } yield {
val protection = getProtectedBranchInfo(repository.owner, repository.name, branch) val protection = getProtectedBranchInfo(repository.owner, repository.name, branch)
JsonFormat(ApiBranch(branch, ApiBranchCommit(br.commitId), ApiBranchProtection(protection))(RepositoryName(repository))) JsonFormat(
ApiBranch(branch, ApiBranchCommit(br.commitId), ApiBranchProtection(protection))(RepositoryName(repository))
)
}) getOrElse NotFound() }) getOrElse NotFound()
}) })
/* /*
* https://developer.github.com/v3/repos/contents/#get-contents * https://developer.github.com/v3/repos/contents/#get-contents
*/ */
get("/api/v3/repos/:owner/:repo/contents/*")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/contents")(referrersOnly { repository =>
getContents(repository, ".", params.getOrElse("ref", repository.repository.defaultBranch))
})
/*
* https://developer.github.com/v3/repos/contents/#get-contents
*/
get("/api/v3/repos/:owner/:repository/contents/*")(referrersOnly { repository =>
getContents(repository, multiParams("splat").head, params.getOrElse("ref", repository.repository.defaultBranch))
})
private def getContents(repository: RepositoryService.RepositoryInfo, path: String, refStr: String) = {
def getFileInfo(git: Git, revision: String, pathStr: String): Option[FileInfo] = { def getFileInfo(git: Git, revision: String, pathStr: String): Option[FileInfo] = {
val path = new java.io.File(pathStr) val (dirName, fileName) = pathStr.lastIndexOf('/') match {
val dirName = path.getParent match { case -1 =>
case null => "." (".", pathStr)
case s => s case n =>
(pathStr.take(n), pathStr.drop(n + 1))
} }
getFileList(git, revision, dirName).find(f => f.name.equals(path.getName)) getFileList(git, revision, dirName).find(f => f.name.equals(fileName))
} }
val path = multiParams("splat").head match { using(Git.open(getRepositoryDir(params("owner"), params("repository")))) { git =>
case s if s.isEmpty => "."
case s => s
}
val refStr = params.getOrElse("ref", repository.repository.defaultBranch)
using(Git.open(getRepositoryDir(params("owner"), params("repo")))){ git =>
val fileList = getFileList(git, refStr, path) val fileList = getFileList(git, refStr, path)
if (fileList.isEmpty) { // file or NotFound if (fileList.isEmpty) { // file or NotFound
getFileInfo(git, refStr, path).flatMap(f => { getFileInfo(git, refStr, path)
.flatMap { f =>
val largeFile = params.get("large_file").exists(s => s.equals("true")) val largeFile = params.get("large_file").exists(s => s.equals("true"))
val content = getContentFromId(git, f.id, largeFile) val content = getContentFromId(git, f.id, largeFile)
request.getHeader("Accept") match { request.getHeader("Accept") match {
@@ -172,48 +199,65 @@ trait ApiControllerBase extends ControllerBase {
} }
case "application/vnd.github.v3.html" if isRenderable(f.name) => { case "application/vnd.github.v3.html" if isRenderable(f.name) => {
contentType = "application/vnd.github.v3.html" contentType = "application/vnd.github.v3.html"
content.map(c => content.map { c =>
List( List(
"<div data-path=\"", path, "\" id=\"file\">", "<article>", "<div data-path=\"",
path,
"\" id=\"file\">",
"<article>",
renderMarkup(path.split("/").toList, new String(c), refStr, repository, false, false, true).body, renderMarkup(path.split("/").toList, new String(c), refStr, repository, false, false, true).body,
"</article>", "</div>" "</article>",
"</div>"
).mkString ).mkString
) }
} }
case "application/vnd.github.v3.html" => { case "application/vnd.github.v3.html" => {
contentType = "application/vnd.github.v3.html" contentType = "application/vnd.github.v3.html"
content.map(c => content.map { c =>
List( List(
"<div data-path=\"", path, "\" id=\"file\">", "<div class=\"plain\">", "<pre>", "<div data-path=\"",
path,
"\" id=\"file\">",
"<div class=\"plain\">",
"<pre>",
play.twirl.api.HtmlFormat.escape(new String(c)).body, play.twirl.api.HtmlFormat.escape(new String(c)).body,
"</pre>", "</div>", "</div>" "</pre>",
"</div>",
"</div>"
).mkString ).mkString
) }
} }
case _ => case _ =>
Some(JsonFormat(ApiContents(f, RepositoryName(repository), content))) Some(JsonFormat(ApiContents(f, RepositoryName(repository), content)))
} }
}).getOrElse(NotFound()) }
.getOrElse(NotFound())
} else { // directory } else { // directory
JsonFormat(fileList.map{f => ApiContents(f, RepositoryName(repository), None)}) JsonFormat(fileList.map { f =>
} ApiContents(f, RepositoryName(repository), None)
}
}) })
}
}
}
/* /*
* https://developer.github.com/v3/git/refs/#get-a-reference * https://developer.github.com/v3/git/refs/#get-a-reference
*/ */
get("/api/v3/repos/:owner/:repo/git/refs/*") (referrersOnly { repository => get("/api/v3/repos/:owner/:repository/git/refs/*")(referrersOnly { repository =>
val revstr = multiParams("splat").head val revstr = multiParams("splat").head
using(Git.open(getRepositoryDir(params("owner"), params("repo")))) { git => using(Git.open(getRepositoryDir(params("owner"), params("repository")))) { git =>
val ref = git.getRepository().findRef(revstr) val ref = git.getRepository().findRef(revstr)
if(ref != null){ if (ref != null) {
val sha = ref.getObjectId().name() val sha = ref.getObjectId().name()
JsonFormat(ApiRef(revstr, ApiObject(sha))) JsonFormat(ApiRef(revstr, ApiObject(sha)))
} else { } else {
val refs = git.getRepository().getAllRefs().asScala val refs = git
.getRepository()
.getAllRefs()
.asScala
.collect { case (str, ref) if str.startsWith("refs/" + revstr) => ref } .collect { case (str, ref) if str.startsWith("refs/" + revstr) => ref }
JsonFormat(refs.map { ref => JsonFormat(refs.map { ref =>
@@ -227,9 +271,11 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* https://developer.github.com/v3/repos/collaborators/#list-collaborators * https://developer.github.com/v3/repos/collaborators/#list-collaborators
*/ */
get("/api/v3/repos/:owner/:repo/collaborators") (referrersOnly { repository => get("/api/v3/repos/:owner/:repository/collaborators")(referrersOnly { repository =>
// TODO Should ApiUser take permission? getCollaboratorUserNames does not return owner group members. // TODO Should ApiUser take permission? getCollaboratorUserNames does not return owner group members.
JsonFormat(getCollaboratorUserNames(params("owner"), params("repo")).map(u => ApiUser(getAccountByUserName(u).get))) JsonFormat(
getCollaboratorUserNames(params("owner"), params("repository")).map(u => ApiUser(getAccountByUserName(u).get))
)
}) })
/** /**
@@ -245,9 +291,9 @@ trait ApiControllerBase extends ControllerBase {
* List user's own repository * List user's own repository
* https://developer.github.com/v3/repos/#list-your-repositories * https://developer.github.com/v3/repos/#list-your-repositories
*/ */
get("/api/v3/user/repos")(usersOnly{ get("/api/v3/user/repos")(usersOnly {
JsonFormat(getVisibleRepositories(context.loginAccount, Option(context.loginAccount.get.userName)).map{ JsonFormat(getVisibleRepositories(context.loginAccount, Option(context.loginAccount.get.userName)).map { r =>
r => ApiRepository(r, getAccountByUserName(r.owner).get) ApiRepository(r, getAccountByUserName(r.owner).get)
}) })
}) })
@@ -261,10 +307,20 @@ trait ApiControllerBase extends ControllerBase {
data <- extractFromJsonBody[CreateARepository] if data.isValid data <- extractFromJsonBody[CreateARepository] if data.isValid
} yield { } yield {
LockUtil.lock(s"${owner}/${data.name}") { LockUtil.lock(s"${owner}/${data.name}") {
if(getRepository(owner, data.name).isEmpty){ if (getRepository(owner, data.name).isEmpty) {
val f = createRepository(context.loginAccount.get, owner, data.name, data.description, data.`private`, data.auto_init) val f = createRepository(
context.loginAccount.get,
owner,
data.name,
data.description,
data.`private`,
data.auto_init
)
Await.result(f, Duration.Inf) Await.result(f, Duration.Inf)
val repository = getRepository(owner, data.name).get
val repository = Database() withTransaction { session =>
getRepository(owner, data.name)(session).get
}
JsonFormat(ApiRepository(repository, ApiUser(getAccountByUserName(owner).get))) JsonFormat(ApiRepository(repository, ApiUser(getAccountByUserName(owner).get)))
} else { } else {
ApiError( ApiError(
@@ -286,8 +342,15 @@ trait ApiControllerBase extends ControllerBase {
data <- extractFromJsonBody[CreateARepository] if data.isValid data <- extractFromJsonBody[CreateARepository] if data.isValid
} yield { } yield {
LockUtil.lock(s"${groupName}/${data.name}") { LockUtil.lock(s"${groupName}/${data.name}") {
if(getRepository(groupName, data.name).isEmpty){ if (getRepository(groupName, data.name).isEmpty) {
val f = createRepository(context.loginAccount.get, groupName, data.name, data.description, data.`private`, data.auto_init) val f = createRepository(
context.loginAccount.get,
groupName,
data.name,
data.description,
data.`private`,
data.auto_init
)
Await.result(f, Duration.Inf) Await.result(f, Duration.Inf)
val repository = getRepository(groupName, data.name).get val repository = getRepository(groupName, data.name).get
JsonFormat(ApiRepository(repository, ApiUser(getAccountByUserName(groupName).get))) JsonFormat(ApiRepository(repository, ApiUser(getAccountByUserName(groupName).get)))
@@ -304,15 +367,26 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection * https://developer.github.com/v3/repos/#enabling-and-disabling-branch-protection
*/ */
patch("/api/v3/repos/:owner/:repo/branches/*")(ownerOnly { repository => patch("/api/v3/repos/:owner/:repository/branches/*")(ownerOnly { repository =>
import gitbucket.core.api._ import gitbucket.core.api._
(for{ (for {
branch <- params.get("splat") if repository.branchList.contains(branch) branch <- params.get("splat") if repository.branchList.contains(branch)
protection <- extractFromJsonBody[ApiBranchProtection.EnablingAndDisabling].map(_.protection) protection <- extractFromJsonBody[ApiBranchProtection.EnablingAndDisabling].map(_.protection)
br <- getBranches(repository.owner, repository.name, repository.repository.defaultBranch, repository.repository.originUserName.isEmpty).find(_.name == branch) br <- getBranches(
repository.owner,
repository.name,
repository.repository.defaultBranch,
repository.repository.originUserName.isEmpty
).find(_.name == branch)
} yield { } yield {
if(protection.enabled){ if (protection.enabled) {
enableBranchProtection(repository.owner, repository.name, branch, protection.status.enforcement_level == ApiBranchProtection.Everyone, protection.status.contexts) enableBranchProtection(
repository.owner,
repository.name,
branch,
protection.status.enforcement_level == ApiBranchProtection.Everyone,
protection.status.contexts
)
} else { } else {
disableBranchProtection(repository.owner, repository.name, branch) disableBranchProtection(repository.owner, repository.name, branch)
} }
@@ -324,7 +398,7 @@ trait ApiControllerBase extends ControllerBase {
* @see https://developer.github.com/v3/rate_limit/#get-your-current-rate-limit-status * @see https://developer.github.com/v3/rate_limit/#get-your-current-rate-limit-status
* but not enabled. * but not enabled.
*/ */
get("/api/v3/rate_limit"){ get("/api/v3/rate_limit") {
contentType = formats("json") contentType = formats("json")
// this message is same as github enterprise... // this message is same as github enterprise...
org.scalatra.NotFound(ApiError("Rate limiting is not enabled.")) org.scalatra.NotFound(ApiError("Rate limiting is not enabled."))
@@ -347,11 +421,14 @@ trait ApiControllerBase extends ControllerBase {
repos = repository.owner -> repository.name repos = repository.owner -> repository.name
) )
JsonFormat(issues.map { case (issue, issueUser) => JsonFormat(issues.map {
case (issue, issueUser) =>
ApiIssue( ApiIssue(
issue = issue, issue = issue,
repositoryName = RepositoryName(repository), repositoryName = RepositoryName(repository),
user = ApiUser(issueUser) user = ApiUser(issueUser),
labels = getIssueLabels(repository.owner, repository.name, issue.issueId)
.map(ApiLabel(_, RepositoryName(repository)))
) )
}) })
}) })
@@ -360,12 +437,19 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/#get-a-single-issue * https://developer.github.com/v3/issues/#get-a-single-issue
*/ */
get("/api/v3/repos/:owner/:repository/issues/:id")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/issues/:id")(referrersOnly { repository =>
(for{ (for {
issueId <- params("id").toIntOpt issueId <- params("id").toIntOpt
issue <- getIssue(repository.owner, repository.name, issueId.toString) issue <- getIssue(repository.owner, repository.name, issueId.toString)
openedUser <- getAccountByUserName(issue.openedUserName) openedUser <- getAccountByUserName(issue.openedUserName)
} yield { } yield {
JsonFormat(ApiIssue(issue, RepositoryName(repository), ApiUser(openedUser))) JsonFormat(
ApiIssue(
issue,
RepositoryName(repository),
ApiUser(openedUser),
getIssueLabels(repository.owner, repository.name, issue.issueId).map(ApiLabel(_, RepositoryName(repository)))
)
)
}) getOrElse NotFound() }) getOrElse NotFound()
}) })
@@ -373,8 +457,8 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/#create-an-issue * https://developer.github.com/v3/issues/#create-an-issue
*/ */
post("/api/v3/repos/:owner/:repository/issues")(readableUsersOnly { repository => post("/api/v3/repos/:owner/:repository/issues")(readableUsersOnly { repository =>
if(isIssueEditable(repository)){ // TODO Should this check is provided by authenticator? if (isIssueEditable(repository)) { // TODO Should this check is provided by authenticator?
(for{ (for {
data <- extractFromJsonBody[CreateAnIssue] data <- extractFromJsonBody[CreateAnIssue]
loginAccount <- context.loginAccount loginAccount <- context.loginAccount
} yield { } yield {
@@ -387,8 +471,17 @@ trait ApiControllerBase extends ControllerBase {
milestone.map(_.milestoneId), milestone.map(_.milestoneId),
None, None,
data.labels, data.labels,
loginAccount) loginAccount
JsonFormat(ApiIssue(issue, RepositoryName(repository), ApiUser(loginAccount))) )
JsonFormat(
ApiIssue(
issue,
RepositoryName(repository),
ApiUser(loginAccount),
getIssueLabels(repository.owner, repository.name, issue.issueId)
.map(ApiLabel(_, RepositoryName(repository)))
)
)
}) getOrElse NotFound() }) getOrElse NotFound()
} else Unauthorized() } else Unauthorized()
}) })
@@ -397,11 +490,14 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/comments/#list-comments-on-an-issue * https://developer.github.com/v3/issues/comments/#list-comments-on-an-issue
*/ */
get("/api/v3/repos/:owner/:repository/issues/:id/comments")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/issues/:id/comments")(referrersOnly { repository =>
(for{ (for {
issueId <- params("id").toIntOpt issueId <- params("id").toIntOpt
comments = getCommentsForApi(repository.owner, repository.name, issueId) comments = getCommentsForApi(repository.owner, repository.name, issueId)
} yield { } yield {
JsonFormat(comments.map{ case (issueComment, user, issue) => ApiComment(issueComment, RepositoryName(repository), issueId, ApiUser(user), issue.isPullRequest) }) JsonFormat(comments.map {
case (issueComment, user, issue) =>
ApiComment(issueComment, RepositoryName(repository), issueId, ApiUser(user), issue.isPullRequest)
})
}) getOrElse NotFound() }) getOrElse NotFound()
}) })
@@ -409,15 +505,23 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/comments/#create-a-comment * https://developer.github.com/v3/issues/comments/#create-a-comment
*/ */
post("/api/v3/repos/:owner/:repository/issues/:id/comments")(readableUsersOnly { repository => post("/api/v3/repos/:owner/:repository/issues/:id/comments")(readableUsersOnly { repository =>
(for{ (for {
issueId <- params("id").toIntOpt issueId <- params("id").toIntOpt
issue <- getIssue(repository.owner, repository.name, issueId.toString) issue <- getIssue(repository.owner, repository.name, issueId.toString)
body <- extractFromJsonBody[CreateAComment].map(_.body) if ! body.isEmpty body <- extractFromJsonBody[CreateAComment].map(_.body) if !body.isEmpty
action = params.get("action").filter(_ => isEditable(issue.userName, issue.repositoryName, issue.openedUserName)) action = params.get("action").filter(_ => isEditable(issue.userName, issue.repositoryName, issue.openedUserName))
(issue, id) <- handleComment(issue, Some(body), repository, action) (issue, id) <- handleComment(issue, Some(body), repository, action)
issueComment <- getComment(repository.owner, repository.name, id.toString()) issueComment <- getComment(repository.owner, repository.name, id.toString())
} yield { } yield {
JsonFormat(ApiComment(issueComment, RepositoryName(repository), issueId, ApiUser(context.loginAccount.get), issue.isPullRequest)) JsonFormat(
ApiComment(
issueComment,
RepositoryName(repository),
issueId,
ApiUser(context.loginAccount.get),
issue.isPullRequest
)
)
}) getOrElse NotFound() }) getOrElse NotFound()
}) })
@@ -446,7 +550,7 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/labels/#create-a-label * https://developer.github.com/v3/issues/labels/#create-a-label
*/ */
post("/api/v3/repos/:owner/:repository/labels")(writableUsersOnly { repository => post("/api/v3/repos/:owner/:repository/labels")(writableUsersOnly { repository =>
(for{ (for {
data <- extractFromJsonBody[CreateALabel] if data.isValid data <- extractFromJsonBody[CreateALabel] if data.isValid
} yield { } yield {
LockUtil.lock(RepositoryName(repository).fullName) { LockUtil.lock(RepositoryName(repository).fullName) {
@@ -457,10 +561,12 @@ trait ApiControllerBase extends ControllerBase {
} getOrElse NotFound() } getOrElse NotFound()
} else { } else {
// TODO ApiError should support errors field to enhance compatibility of GitHub API // TODO ApiError should support errors field to enhance compatibility of GitHub API
UnprocessableEntity(ApiError( UnprocessableEntity(
ApiError(
"Validation Failed", "Validation Failed",
Some("https://developer.github.com/v3/issues/labels/#create-a-label") Some("https://developer.github.com/v3/issues/labels/#create-a-label")
)) )
)
} }
} }
}) getOrElse NotFound() }) getOrElse NotFound()
@@ -471,23 +577,28 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/issues/labels/#update-a-label * https://developer.github.com/v3/issues/labels/#update-a-label
*/ */
patch("/api/v3/repos/:owner/:repository/labels/:labelName")(writableUsersOnly { repository => patch("/api/v3/repos/:owner/:repository/labels/:labelName")(writableUsersOnly { repository =>
(for{ (for {
data <- extractFromJsonBody[CreateALabel] if data.isValid data <- extractFromJsonBody[CreateALabel] if data.isValid
} yield { } yield {
LockUtil.lock(RepositoryName(repository).fullName) { LockUtil.lock(RepositoryName(repository).fullName) {
getLabel(repository.owner, repository.name, params("labelName")).map { label => getLabel(repository.owner, repository.name, params("labelName")).map {
label =>
if (getLabel(repository.owner, repository.name, data.name).isEmpty) { if (getLabel(repository.owner, repository.name, data.name).isEmpty) {
updateLabel(repository.owner, repository.name, label.labelId, data.name, data.color) updateLabel(repository.owner, repository.name, label.labelId, data.name, data.color)
JsonFormat(ApiLabel( JsonFormat(
ApiLabel(
getLabel(repository.owner, repository.name, label.labelId).get, getLabel(repository.owner, repository.name, label.labelId).get,
RepositoryName(repository) RepositoryName(repository)
)) )
)
} else { } else {
// TODO ApiError should support errors field to enhance compatibility of GitHub API // TODO ApiError should support errors field to enhance compatibility of GitHub API
UnprocessableEntity(ApiError( UnprocessableEntity(
ApiError(
"Validation Failed", "Validation Failed",
Some("https://developer.github.com/v3/issues/labels/#create-a-label") Some("https://developer.github.com/v3/issues/labels/#create-a-label")
)) )
)
} }
} getOrElse NotFound() } getOrElse NotFound()
} }
@@ -524,13 +635,16 @@ trait ApiControllerBase extends ControllerBase {
repos = repository.owner -> repository.name repos = repository.owner -> repository.name
) )
JsonFormat(issues.map { case (issue, issueUser, commentCount, pullRequest, headRepo, headOwner, assignee) => JsonFormat(issues.map {
case (issue, issueUser, commentCount, pullRequest, headRepo, headOwner, assignee) =>
ApiPullRequest( ApiPullRequest(
issue = issue, issue = issue,
pullRequest = pullRequest, pullRequest = pullRequest,
headRepo = ApiRepository(headRepo, ApiUser(headOwner)), headRepo = ApiRepository(headRepo, ApiUser(headOwner)),
baseRepo = ApiRepository(repository, ApiUser(baseOwner)), baseRepo = ApiRepository(repository, ApiUser(baseOwner)),
user = ApiUser(issueUser), user = ApiUser(issueUser),
labels = getIssueLabels(repository.owner, repository.name, issue.issueId)
.map(ApiLabel(_, RepositoryName(repository))),
assignee = assignee.map(ApiUser.apply), assignee = assignee.map(ApiUser.apply),
mergedComment = getMergedComment(repository.owner, repository.name, issue.issueId) mergedComment = getMergedComment(repository.owner, repository.name, issue.issueId)
) )
@@ -541,25 +655,34 @@ trait ApiControllerBase extends ControllerBase {
* https://developer.github.com/v3/pulls/#get-a-single-pull-request * https://developer.github.com/v3/pulls/#get-a-single-pull-request
*/ */
get("/api/v3/repos/:owner/:repository/pulls/:id")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/pulls/:id")(referrersOnly { repository =>
(for{ (for {
issueId <- params("id").toIntOpt issueId <- params("id").toIntOpt
(issue, pullRequest) <- getPullRequest(repository.owner, repository.name, issueId) (issue, pullRequest) <- getPullRequest(repository.owner, repository.name, issueId)
users = getAccountsByUserNames(Set(repository.owner, pullRequest.requestUserName, issue.openedUserName), Set.empty) users = getAccountsByUserNames(
Set(repository.owner, pullRequest.requestUserName, issue.openedUserName),
Set.empty
)
baseOwner <- users.get(repository.owner) baseOwner <- users.get(repository.owner)
headOwner <- users.get(pullRequest.requestUserName) headOwner <- users.get(pullRequest.requestUserName)
issueUser <- users.get(issue.openedUserName) issueUser <- users.get(issue.openedUserName)
assignee = issue.assignedUserName.flatMap { userName => getAccountByUserName(userName, false) } assignee = issue.assignedUserName.flatMap { userName =>
getAccountByUserName(userName, false)
}
headRepo <- getRepository(pullRequest.requestUserName, pullRequest.requestRepositoryName) headRepo <- getRepository(pullRequest.requestUserName, pullRequest.requestRepositoryName)
} yield { } yield {
JsonFormat(ApiPullRequest( JsonFormat(
ApiPullRequest(
issue = issue, issue = issue,
pullRequest = pullRequest, pullRequest = pullRequest,
headRepo = ApiRepository(headRepo, ApiUser(headOwner)), headRepo = ApiRepository(headRepo, ApiUser(headOwner)),
baseRepo = ApiRepository(repository, ApiUser(baseOwner)), baseRepo = ApiRepository(repository, ApiUser(baseOwner)),
user = ApiUser(issueUser), user = ApiUser(issueUser),
labels = getIssueLabels(repository.owner, repository.name, issue.issueId)
.map(ApiLabel(_, RepositoryName(repository))),
assignee = assignee.map(ApiUser.apply), assignee = assignee.map(ApiUser.apply),
mergedComment = getMergedComment(repository.owner, repository.name, issue.issueId) mergedComment = getMergedComment(repository.owner, repository.name, issue.issueId)
)) )
)
}) getOrElse NotFound() }) getOrElse NotFound()
}) })
@@ -569,13 +692,23 @@ trait ApiControllerBase extends ControllerBase {
get("/api/v3/repos/:owner/:repository/pulls/:id/commits")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/pulls/:id/commits")(referrersOnly { repository =>
val owner = repository.owner val owner = repository.owner
val name = repository.name val name = repository.name
params("id").toIntOpt.flatMap{ issueId => params("id").toIntOpt.flatMap {
getPullRequest(owner, name, issueId) map { case(issue, pullreq) => issueId =>
using(Git.open(getRepositoryDir(owner, name))){ git => getPullRequest(owner, name, issueId) map {
case (issue, pullreq) =>
using(Git.open(getRepositoryDir(owner, name))) { git =>
val oldId = git.getRepository.resolve(pullreq.commitIdFrom) val oldId = git.getRepository.resolve(pullreq.commitIdFrom)
val newId = git.getRepository.resolve(pullreq.commitIdTo) val newId = git.getRepository.resolve(pullreq.commitIdTo)
val repoFullName = RepositoryName(repository) val repoFullName = RepositoryName(repository)
val commits = git.log.addRange(oldId, newId).call.iterator.asScala.map { c => ApiCommitListItem(new CommitInfo(c), repoFullName) }.toList val commits = git.log
.addRange(oldId, newId)
.call
.iterator
.asScala
.map { c =>
ApiCommitListItem(new CommitInfo(c), repoFullName)
}
.toList
JsonFormat(commits) JsonFormat(commits)
} }
} }
@@ -592,15 +725,24 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* https://developer.github.com/v3/repos/statuses/#create-a-status * https://developer.github.com/v3/repos/statuses/#create-a-status
*/ */
post("/api/v3/repos/:owner/:repo/statuses/:sha")(writableUsersOnly { repository => post("/api/v3/repos/:owner/:repository/statuses/:sha")(writableUsersOnly { repository =>
(for{ (for {
ref <- params.get("sha") ref <- params.get("sha")
sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref) sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref)
data <- extractFromJsonBody[CreateAStatus] if data.isValid data <- extractFromJsonBody[CreateAStatus] if data.isValid
creator <- context.loginAccount creator <- context.loginAccount
state <- CommitState.valueOf(data.state) state <- CommitState.valueOf(data.state)
statusId = createCommitStatus(repository.owner, repository.name, sha, data.context.getOrElse("default"), statusId = createCommitStatus(
state, data.target_url, data.description, new java.util.Date(), creator) repository.owner,
repository.name,
sha,
data.context.getOrElse("default"),
state,
data.target_url,
data.description,
new java.util.Date(),
creator
)
status <- getCommitStatus(repository.owner, repository.name, statusId) status <- getCommitStatus(repository.owner, repository.name, statusId)
} yield { } yield {
JsonFormat(ApiCommitStatus(status, ApiUser(creator))) JsonFormat(ApiCommitStatus(status, ApiUser(creator)))
@@ -612,12 +754,13 @@ trait ApiControllerBase extends ControllerBase {
* *
* ref is Ref to list the statuses from. It can be a SHA, a branch name, or a tag name. * ref is Ref to list the statuses from. It can be a SHA, a branch name, or a tag name.
*/ */
val listStatusesRoute = get("/api/v3/repos/:owner/:repo/commits/:ref/statuses")(referrersOnly { repository => val listStatusesRoute = get("/api/v3/repos/:owner/:repository/commits/:ref/statuses")(referrersOnly { repository =>
(for{ (for {
ref <- params.get("ref") ref <- params.get("ref")
sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref) sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref)
} yield { } yield {
JsonFormat(getCommitStatuesWithCreator(repository.owner, repository.name, sha).map{ case(status, creator) => JsonFormat(getCommitStatuesWithCreator(repository.owner, repository.name, sha).map {
case (status, creator) =>
ApiCommitStatus(status, ApiUser(creator)) ApiCommitStatus(status, ApiUser(creator))
}) })
}) getOrElse NotFound() }) getOrElse NotFound()
@@ -628,7 +771,7 @@ trait ApiControllerBase extends ControllerBase {
* *
* legacy route * legacy route
*/ */
get("/api/v3/repos/:owner/:repo/statuses/:ref"){ get("/api/v3/repos/:owner/:repository/statuses/:ref") {
listStatusesRoute.action() listStatusesRoute.action()
} }
@@ -637,8 +780,8 @@ trait ApiControllerBase extends ControllerBase {
* *
* ref is Ref to list the statuses from. It can be a SHA, a branch name, or a tag name. * ref is Ref to list the statuses from. It can be a SHA, a branch name, or a tag name.
*/ */
get("/api/v3/repos/:owner/:repo/commits/:ref/status")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/commits/:ref/status")(referrersOnly { repository =>
(for{ (for {
ref <- params.get("ref") ref <- params.get("ref")
owner <- getAccountByUserName(repository.owner) owner <- getAccountByUserName(repository.owner)
sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref) sha <- JGitUtil.getShaByRef(repository.owner, repository.name, ref)
@@ -651,26 +794,29 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* https://developer.github.com/v3/repos/commits/#get-a-single-commit * https://developer.github.com/v3/repos/commits/#get-a-single-commit
*/ */
get("/api/v3/repos/:owner/:repo/commits/:sha")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/commits/:sha")(referrersOnly { repository =>
val owner = repository.owner val owner = repository.owner
val name = repository.name val name = repository.name
val sha = params("sha") val sha = params("sha")
using(Git.open(getRepositoryDir(owner, name))){ git => using(Git.open(getRepositoryDir(owner, name))) {
git =>
val repo = git.getRepository val repo = git.getRepository
val objectId = repo.resolve(sha) val objectId = repo.resolve(sha)
val commitInfo = using(new RevWalk(repo)){ revWalk => val commitInfo = using(new RevWalk(repo)) { revWalk =>
new CommitInfo(revWalk.parseCommit(objectId)) new CommitInfo(revWalk.parseCommit(objectId))
} }
JsonFormat(ApiCommits( JsonFormat(
ApiCommits(
repositoryName = RepositoryName(repository), repositoryName = RepositoryName(repository),
commitInfo = commitInfo, commitInfo = commitInfo,
diffs = JGitUtil.getDiffs(git, Some(commitInfo.parents.head), commitInfo.id, false, true), diffs = JGitUtil.getDiffs(git, commitInfo.parents.headOption, commitInfo.id, false, true),
author = getAccount(commitInfo.authorName, commitInfo.authorEmailAddress), author = getAccount(commitInfo.authorName, commitInfo.authorEmailAddress),
committer = getAccount(commitInfo.committerName, commitInfo.committerEmailAddress), committer = getAccount(commitInfo.committerName, commitInfo.committerEmailAddress),
commentCount = getCommitComment(repository.owner, repository.name, sha).size commentCount = getCommitComment(repository.owner, repository.name, sha).size
)) )
)
} }
}) })
@@ -700,9 +846,9 @@ trait ApiControllerBase extends ControllerBase {
/** /**
* non-GitHub compatible API for Jenkins-Plugin * non-GitHub compatible API for Jenkins-Plugin
*/ */
get("/api/v3/repos/:owner/:repo/raw/*")(referrersOnly { repository => get("/api/v3/repos/:owner/:repository/raw/*")(referrersOnly { repository =>
val (id, path) = repository.splitPath(multiParams("splat").head) val (id, path) = repository.splitPath(multiParams("splat").head)
using(Git.open(getRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getRepositoryDir(repository.owner, repository.name))) { git =>
val revCommit = JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(id)) val revCommit = JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(id))
getPathObjectId(git, path, revCommit).map { objectId => getPathObjectId(git, path, revCommit).map { objectId =>
@@ -711,5 +857,10 @@ trait ApiControllerBase extends ControllerBase {
} }
}) })
/**
* non-GitHub compatible API for listing plugins
*/
get("/api/v3/gitbucket/plugins") {
PluginRegistry().getPlugins().map { ApiPlugin(_) }
}
} }

View File

@@ -1,8 +1,8 @@
package gitbucket.core.controller package gitbucket.core.controller
import java.io.FileInputStream import java.io.{File, FileInputStream}
import gitbucket.core.api.ApiError import gitbucket.core.api.{ApiError, JsonFormat}
import gitbucket.core.model.Account import gitbucket.core.model.Account
import gitbucket.core.service.{AccountService, RepositoryService, SystemSettingsService} import gitbucket.core.service.{AccountService, RepositoryService, SystemSettingsService}
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
@@ -31,8 +31,13 @@ import org.slf4j.LoggerFactory
/** /**
* Provides generic features for controller implementations. * Provides generic features for controller implementations.
*/ */
abstract class ControllerBase extends ScalatraFilter abstract class ControllerBase
with ValidationSupport with JacksonJsonSupport with I18nSupport with FlashMapSupport with Validations extends ScalatraFilter
with ValidationSupport
with JacksonJsonSupport
with I18nSupport
with FlashMapSupport
with Validations
with SystemSettingsService { with SystemSettingsService {
private val logger = LoggerFactory.getLogger(getClass) private val logger = LoggerFactory.getLogger(getClass)
@@ -41,25 +46,33 @@ abstract class ControllerBase extends ScalatraFilter
before("/api/v3/*") { before("/api/v3/*") {
contentType = formats("json") contentType = formats("json")
request.setAttribute(Keys.Request.APIv3, true)
} }
override def doFilter(request: ServletRequest, response: ServletResponse, chain: FilterChain): Unit = try { override def requestPath(uri: String, idx: Int): String = {
val path = super.requestPath(uri, idx)
if (path != "/" && path.endsWith("/")) {
path.substring(0, path.length - 1)
} else {
path
}
}
override def doFilter(request: ServletRequest, response: ServletResponse, chain: FilterChain): Unit =
try {
val httpRequest = request.asInstanceOf[HttpServletRequest] val httpRequest = request.asInstanceOf[HttpServletRequest]
val context = request.getServletContext.getContextPath val context = request.getServletContext.getContextPath
val path = httpRequest.getRequestURI.substring(context.length) val path = httpRequest.getRequestURI.substring(context.length)
if(path.startsWith("/git/") || path.startsWith("/git-lfs/")){ if (path.startsWith("/git/") || path.startsWith("/git-lfs/")) {
// Git repository // Git repository
chain.doFilter(request, response) chain.doFilter(request, response)
} else { } else {
if(path.startsWith("/api/v3/")){
httpRequest.setAttribute(Keys.Request.APIv3, true)
}
// Scalatra actions // Scalatra actions
super.doFilter(request, response, chain) super.doFilter(request, response, chain)
} }
} finally { } finally {
contextCache.remove(); contextCache.remove()
} }
private val contextCache = new java.lang.ThreadLocal[Context]() private val contextCache = new java.lang.ThreadLocal[Context]()
@@ -78,44 +91,45 @@ abstract class ControllerBase extends ScalatraFilter
} }
} }
private def LoginAccount: Option[Account] = request.getAs[Account](Keys.Session.LoginAccount).orElse(session.getAs[Account](Keys.Session.LoginAccount)) private def LoginAccount: Option[Account] =
request.getAs[Account](Keys.Session.LoginAccount).orElse(session.getAs[Account](Keys.Session.LoginAccount))
def ajaxGet(path : String)(action : => Any) : Route = def ajaxGet(path: String)(action: => Any): Route =
super.get(path){ super.get(path) {
request.setAttribute(Keys.Request.Ajax, "true") request.setAttribute(Keys.Request.Ajax, "true")
action action
} }
override def ajaxGet[T](path : String, form : ValueType[T])(action : T => Any) : Route = override def ajaxGet[T](path: String, form: ValueType[T])(action: T => Any): Route =
super.ajaxGet(path, form){ form => super.ajaxGet(path, form) { form =>
request.setAttribute(Keys.Request.Ajax, "true") request.setAttribute(Keys.Request.Ajax, "true")
action(form) action(form)
} }
def ajaxPost(path : String)(action : => Any) : Route = def ajaxPost(path: String)(action: => Any): Route =
super.post(path){ super.post(path) {
request.setAttribute(Keys.Request.Ajax, "true") request.setAttribute(Keys.Request.Ajax, "true")
action action
} }
override def ajaxPost[T](path : String, form : ValueType[T])(action : T => Any) : Route = override def ajaxPost[T](path: String, form: ValueType[T])(action: T => Any): Route =
super.ajaxPost(path, form){ form => super.ajaxPost(path, form) { form =>
request.setAttribute(Keys.Request.Ajax, "true") request.setAttribute(Keys.Request.Ajax, "true")
action(form) action(form)
} }
protected def NotFound() = protected def NotFound() =
if(request.hasAttribute(Keys.Request.Ajax)){ if (request.hasAttribute(Keys.Request.Ajax)) {
org.scalatra.NotFound() org.scalatra.NotFound()
} else if(request.hasAttribute(Keys.Request.APIv3)){ } else if (request.hasAttribute(Keys.Request.APIv3)) {
contentType = formats("json") contentType = formats("json")
org.scalatra.NotFound(ApiError("Not Found")) org.scalatra.NotFound(JsonFormat(ApiError("Not Found")))
} else { } else {
org.scalatra.NotFound(gitbucket.core.html.error("Not Found")) org.scalatra.NotFound(gitbucket.core.html.error("Not Found"))
} }
private def isBrowser(userAgent: String): Boolean = { private def isBrowser(userAgent: String): Boolean = {
if(userAgent == null || userAgent.isEmpty){ if (userAgent == null || userAgent.isEmpty) {
false false
} else { } else {
val data = Classifier.parse(userAgent) val data = Classifier.parse(userAgent)
@@ -125,67 +139,82 @@ abstract class ControllerBase extends ScalatraFilter
} }
protected def Unauthorized()(implicit context: Context) = protected def Unauthorized()(implicit context: Context) =
if(request.hasAttribute(Keys.Request.Ajax)){ if (request.hasAttribute(Keys.Request.Ajax)) {
org.scalatra.Unauthorized() org.scalatra.Unauthorized()
} else if(request.hasAttribute(Keys.Request.APIv3)){ } else if (request.hasAttribute(Keys.Request.APIv3)) {
contentType = formats("json") contentType = formats("json")
org.scalatra.Unauthorized(ApiError("Requires authentication")) org.scalatra.Unauthorized(JsonFormat(ApiError("Requires authentication")))
} else if(!isBrowser(request.getHeader("USER-AGENT"))){ } else if (!isBrowser(request.getHeader("USER-AGENT"))) {
org.scalatra.Unauthorized() org.scalatra.Unauthorized()
} else { } else {
if(context.loginAccount.isDefined){ if (context.loginAccount.isDefined) {
org.scalatra.Unauthorized(redirect("/")) org.scalatra.Unauthorized(redirect("/"))
} else { } else {
if(request.getMethod.toUpperCase == "POST"){ if (request.getMethod.toUpperCase == "POST") {
org.scalatra.Unauthorized(redirect("/signin")) org.scalatra.Unauthorized(redirect("/signin"))
} else { } else {
org.scalatra.Unauthorized(redirect("/signin?redirect=" + StringUtil.urlEncode( org.scalatra.Unauthorized(
defining(request.getQueryString){ queryString => redirect(
request.getRequestURI.substring(request.getContextPath.length) + (if(queryString != null) "?" + queryString else "") "/signin?redirect=" + StringUtil.urlEncode(
defining(request.getQueryString) { queryString =>
request.getRequestURI.substring(request.getContextPath.length) + (if (queryString != null)
"?" + queryString
else "")
} }
))) )
)
)
} }
} }
} }
error{ error {
case e => { case e => {
logger.error(s"Catch unhandled error in request: ${request}", e) logger.error(s"Catch unhandled error in request: ${request}", e)
if(request.hasAttribute(Keys.Request.Ajax)){ if (request.hasAttribute(Keys.Request.Ajax)) {
org.scalatra.InternalServerError() org.scalatra.InternalServerError()
} else if(request.hasAttribute(Keys.Request.APIv3)){ } else if (request.hasAttribute(Keys.Request.APIv3)) {
contentType = formats("json") contentType = formats("json")
org.scalatra.InternalServerError(ApiError("Internal Server Error")) org.scalatra.InternalServerError(JsonFormat(ApiError("Internal Server Error")))
} else { } else {
org.scalatra.InternalServerError(gitbucket.core.html.error("Internal Server Error", Some(e))) org.scalatra.InternalServerError(gitbucket.core.html.error("Internal Server Error", Some(e)))
} }
} }
} }
override def url(path: String, params: Iterable[(String, Any)] = Iterable.empty, override def url(
includeContextPath: Boolean = true, includeServletPath: Boolean = true, path: String,
absolutize: Boolean = true, withSessionId: Boolean = true) params: Iterable[(String, Any)] = Iterable.empty,
(implicit request: HttpServletRequest, response: HttpServletResponse): String = includeContextPath: Boolean = true,
includeServletPath: Boolean = true,
absolutize: Boolean = true,
withSessionId: Boolean = true
)(implicit request: HttpServletRequest, response: HttpServletResponse): String =
if (path.startsWith("http")) path if (path.startsWith("http")) path
else baseUrl + super.url(path, params, false, false, false) else baseUrl + super.url(path, params, false, false, false)
/** /**
* Extends scalatra-form's trim rule to eliminate CR and LF. * Extends scalatra-form's trim rule to eliminate CR and LF.
*/ */
protected def trim2[T](valueType: SingleValueType[T]): SingleValueType[T] = new SingleValueType[T](){ protected def trim2[T](valueType: SingleValueType[T]): SingleValueType[T] = new SingleValueType[T]() {
def convert(value: String, messages: Messages): T = valueType.convert(trim(value), messages) def convert(value: String, messages: Messages): T = valueType.convert(trim(value), messages)
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] = override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Seq[(String, String)] =
valueType.validate(name, trim(value), params, messages) valueType.validate(name, trim(value), params, messages)
private def trim(value: String): String = if(value == null) null else value.replace("\r\n", "").trim private def trim(value: String): String = if (value == null) null else value.replace("\r\n", "").trim
} }
/** /**
* Use this method to response the raw data against XSS. * Use this method to response the raw data against XSS.
*/ */
protected def RawData[T](contentType: String, rawData: T): T = { protected def RawData[T](contentType: String, rawData: T): T = {
if(contentType.split(";").head.trim.toLowerCase.startsWith("text/html")){ if (contentType.split(";").head.trim.toLowerCase.startsWith("text/html")) {
this.contentType = "text/plain" this.contentType = "text/plain"
} else { } else {
this.contentType = contentType this.contentType = contentType
@@ -195,8 +224,8 @@ abstract class ControllerBase extends ScalatraFilter
} }
// jenkins send message as 'application/x-www-form-urlencoded' but scalatra already parsed as multi-part-request. // jenkins send message as 'application/x-www-form-urlencoded' but scalatra already parsed as multi-part-request.
def extractFromJsonBody[A](implicit request:HttpServletRequest, mf:Manifest[A]): Option[A] = { def extractFromJsonBody[A](implicit request: HttpServletRequest, mf: Manifest[A]): Option[A] = {
(request.contentType.map(_.split(";").head.toLowerCase) match{ (request.contentType.map(_.split(";").head.toLowerCase) match {
case Some("application/x-www-form-urlencoded") => multiParams.keys.headOption.map(parse(_)) case Some("application/x-www-form-urlencoded") => multiParams.keys.headOption.map(parse(_))
case Some("application/json") => Some(parsedBody) case Some("application/json") => Some(parsedBody)
case _ => Some(parse(request.body)) case _ => Some(parse(request.body))
@@ -206,24 +235,28 @@ abstract class ControllerBase extends ScalatraFilter
protected def getPathObjectId(git: Git, path: String, revCommit: RevCommit): Option[ObjectId] = { protected def getPathObjectId(git: Git, path: String, revCommit: RevCommit): Option[ObjectId] = {
@scala.annotation.tailrec @scala.annotation.tailrec
def _getPathObjectId(path: String, walk: TreeWalk): Option[ObjectId] = walk.next match { def _getPathObjectId(path: String, walk: TreeWalk): Option[ObjectId] = walk.next match {
case true if(walk.getPathString == path) => Some(walk.getObjectId(0)) case true if (walk.getPathString == path) => Some(walk.getObjectId(0))
case true => _getPathObjectId(path, walk) case true => _getPathObjectId(path, walk)
case false => None case false => None
} }
using(new TreeWalk(git.getRepository)){ treeWalk => using(new TreeWalk(git.getRepository)) { treeWalk =>
treeWalk.addTree(revCommit.getTree) treeWalk.addTree(revCommit.getTree)
treeWalk.setRecursive(true) treeWalk.setRecursive(true)
_getPathObjectId(path, treeWalk) _getPathObjectId(path, treeWalk)
} }
} }
protected def responseRawFile(git: Git, objectId: ObjectId, path: String, protected def responseRawFile(
repository: RepositoryService.RepositoryInfo): Unit = { git: Git,
JGitUtil.getObjectLoaderFromId(git, objectId){ loader => objectId: ObjectId,
contentType = FileUtil.getMimeType(path) path: String,
repository: RepositoryService.RepositoryInfo
): Unit = {
JGitUtil.getObjectLoaderFromId(git, objectId) { loader =>
contentType = FileUtil.getSafeMimeType(path)
if(loader.isLarge){ if (loader.isLarge) {
response.setContentLength(loader.getSize.toInt) response.setContentLength(loader.getSize.toInt)
loader.copyTo(response.outputStream) loader.copyTo(response.outputStream)
} else { } else {
@@ -231,11 +264,11 @@ abstract class ControllerBase extends ScalatraFilter
val text = new String(bytes, "UTF-8") val text = new String(bytes, "UTF-8")
val attrs = JGitUtil.getLfsObjects(text) val attrs = JGitUtil.getLfsObjects(text)
if(attrs.nonEmpty) { if (attrs.nonEmpty) {
response.setContentLength(attrs("size").toInt) response.setContentLength(attrs("size").toInt)
val oid = attrs("oid").split(":")(1) val oid = attrs("oid").split(":")(1)
using(new FileInputStream(FileUtil.getLfsFilePath(repository.owner, repository.name, oid))){ in => using(new FileInputStream(FileUtil.getLfsFilePath(repository.owner, repository.name, oid))) { in =>
IOUtils.copy(in, response.getOutputStream) IOUtils.copy(in, response.getOutputStream)
} }
} else { } else {
@@ -250,7 +283,11 @@ abstract class ControllerBase extends ScalatraFilter
/** /**
* Context object for the current request. * Context object for the current request.
*/ */
case class Context(settings: SystemSettingsService.SystemSettings, loginAccount: Option[Account], request: HttpServletRequest){ case class Context(
settings: SystemSettingsService.SystemSettings,
loginAccount: Option[Account],
request: HttpServletRequest
) {
val path = settings.baseUrl.getOrElse(request.getContextPath) val path = settings.baseUrl.getOrElse(request.getContextPath)
val currentPath = request.getRequestURI.substring(request.getContextPath.length) val currentPath = request.getRequestURI.substring(request.getContextPath.length)
val baseUrl = settings.baseUrl(request) val baseUrl = settings.baseUrl(request)
@@ -271,7 +308,7 @@ case class Context(settings: SystemSettingsService.SystemSettings, loginAccount:
* Cached object are available during a request. * Cached object are available during a request.
*/ */
def cache[A](key: String)(action: => A): A = def cache[A](key: String)(action: => A): A =
defining(Keys.Request.Cache(key)){ cacheKey => defining(Keys.Request.Cache(key)) { cacheKey =>
Option(request.getAttribute(cacheKey).asInstanceOf[A]).getOrElse { Option(request.getAttribute(cacheKey).asInstanceOf[A]).getOrElse {
val newObject = action val newObject = action
request.setAttribute(cacheKey, newObject) request.setAttribute(cacheKey, newObject)
@@ -288,41 +325,98 @@ trait AccountManagementControllerBase extends ControllerBase {
self: AccountService => self: AccountService =>
protected def updateImage(userName: String, fileId: Option[String], clearImage: Boolean): Unit = protected def updateImage(userName: String, fileId: Option[String], clearImage: Boolean): Unit =
if(clearImage){ if (clearImage) {
getAccountByUserName(userName).flatMap(_.image).map { image => getAccountByUserName(userName).flatMap(_.image).foreach { image =>
new java.io.File(getUserUploadDir(userName), image).delete() new File(getUserUploadDir(userName), FileUtil.checkFilename(image)).delete()
updateAvatarImage(userName, None) updateAvatarImage(userName, None)
} }
} else { } else {
fileId.map { fileId => fileId.foreach { fileId =>
val filename = "avatar." + FileUtil.getExtension(session.getAndRemove(Keys.Session.Upload(fileId)).get) val filename = "avatar." + FileUtil.getExtension(session.getAndRemove(Keys.Session.Upload(fileId)).get)
val uploadDir = getUserUploadDir(userName) val uploadDir = getUserUploadDir(userName)
if(!uploadDir.exists){ if (!uploadDir.exists) {
uploadDir.mkdirs() uploadDir.mkdirs()
} }
Thumbnails.of(new java.io.File(getTemporaryDir(session.getId), fileId)) Thumbnails
.of(new File(getTemporaryDir(session.getId), FileUtil.checkFilename(fileId)))
.size(324, 324) .size(324, 324)
.toFile(new java.io.File(uploadDir, filename)) .toFile(new File(uploadDir, FileUtil.checkFilename(filename)))
updateAvatarImage(userName, Some(filename)) updateAvatarImage(userName, Some(filename))
} }
} }
protected def uniqueUserName: Constraint = new Constraint(){ protected def uniqueUserName: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
getAccountByUserName(value, true).map { _ => "User already exists." } getAccountByUserNameIgnoreCase(value, true).map { _ =>
"User already exists."
}
} }
protected def uniqueMailAddress(paramName: String = ""): Constraint = new Constraint(){ protected def uniqueMailAddress(paramName: String = ""): Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = { override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
val extraMailAddresses = params.filterKeys(k => k.startsWith("extraMailAddresses"))
if (extraMailAddresses.exists {
case (k, v) =>
v.contains(value)
}) {
Some("These mail addresses are duplicated.")
} else {
getAccountByMailAddress(value, true) getAccountByMailAddress(value, true)
.filter { x => if(paramName.isEmpty) true else Some(x.userName) != params.optionValue(paramName) } .collect {
.map { _ => "Mail address is already registered." } case x if paramName.isEmpty || Some(x.userName) != params.optionValue(paramName) =>
"Mail address is already registered."
}
}
} }
} }
val allReservedNames = Set("git", "admin", "upload", "api", "assets", "plugin-assets", "signin", "signout", "register", "activities.atom", "sidebar-collapse", "groups", "new") protected def uniqueExtraMailAddress(paramName: String = ""): Constraint = new Constraint() {
protected def reservedNames(): Constraint = new Constraint(){ override def validate(
override def validate(name: String, value: String, messages: Messages): Option[String] = if(allReservedNames.contains(value)){ name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
val extraMailAddresses = params.filterKeys(k => k.startsWith("extraMailAddresses"))
if (Some(value) == params.optionValue("mailAddress") || extraMailAddresses.count {
case (k, v) =>
v.contains(value)
} > 1) {
Some("These mail addresses are duplicated.")
} else {
getAccountByMailAddress(value, true)
.collect {
case x if paramName.isEmpty || Some(x.userName) != params.optionValue(paramName) =>
"Mail address is already registered."
}
}
}
}
val allReservedNames = Set(
"git",
"admin",
"upload",
"api",
"assets",
"plugin-assets",
"signin",
"signout",
"register",
"activities.atom",
"sidebar-collapse",
"groups",
"new"
)
protected def reservedNames(): Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] =
if (allReservedNames.contains(value.toLowerCase)) {
Some(s"${value} is reserved") Some(s"${value} is reserved")
} else { } else {
None None

View File

@@ -6,13 +6,25 @@ import gitbucket.core.util.{Keys, UsersAuthenticator}
import gitbucket.core.util.Implicits._ import gitbucket.core.util.Implicits._
import gitbucket.core.service.IssuesService._ import gitbucket.core.service.IssuesService._
class DashboardController extends DashboardControllerBase class DashboardController
with IssuesService with PullRequestService with RepositoryService with AccountService with CommitsService extends DashboardControllerBase
with IssuesService
with PullRequestService
with RepositoryService
with AccountService
with CommitsService
with LabelsService
with PrioritiesService
with MilestonesService
with UsersAuthenticator with UsersAuthenticator
trait DashboardControllerBase extends ControllerBase { trait DashboardControllerBase extends ControllerBase {
self: IssuesService with PullRequestService with RepositoryService with AccountService self: IssuesService with PullRequestService with RepositoryService with AccountService with UsersAuthenticator =>
with UsersAuthenticator =>
get("/dashboard/repos")(usersOnly {
val repos = getVisibleRepositories(context.loginAccount, withoutPhysicalInfo = true)
html.repos(getGroupNames(context.loginAccount.get.userName), repos, repos)
})
get("/dashboard/issues")(usersOnly { get("/dashboard/issues")(usersOnly {
searchIssues("created_by") searchIssues("created_by")
@@ -67,7 +79,7 @@ trait DashboardControllerBase extends ControllerBase {
html.issues( html.issues(
searchIssue(condition, false, (page - 1) * IssueLimit, IssueLimit, userRepos: _*), searchIssue(condition, false, (page - 1) * IssueLimit, IssueLimit, userRepos: _*),
page, page,
countIssue(condition.copy(state = "open" ), false, userRepos: _*), countIssue(condition.copy(state = "open"), false, userRepos: _*),
countIssue(condition.copy(state = "closed"), false, userRepos: _*), countIssue(condition.copy(state = "closed"), false, userRepos: _*),
filter match { filter match {
case "assigned" => condition.copy(assigned = Some(Some(userName))) case "assigned" => condition.copy(assigned = Some(Some(userName)))
@@ -76,8 +88,8 @@ trait DashboardControllerBase extends ControllerBase {
}, },
filter, filter,
getGroupNames(userName), getGroupNames(userName),
Nil, getVisibleRepositories(context.loginAccount, withoutPhysicalInfo = true)
getUserRepositories(userName, withoutPhysicalInfo = true)) )
} }
private def searchPullRequests(filter: String) = { private def searchPullRequests(filter: String) = {
@@ -92,7 +104,7 @@ trait DashboardControllerBase extends ControllerBase {
html.pulls( html.pulls(
searchIssue(condition, true, (page - 1) * PullRequestLimit, PullRequestLimit, allRepos: _*), searchIssue(condition, true, (page - 1) * PullRequestLimit, PullRequestLimit, allRepos: _*),
page, page,
countIssue(condition.copy(state = "open" ), true, allRepos: _*), countIssue(condition.copy(state = "open"), true, allRepos: _*),
countIssue(condition.copy(state = "closed"), true, allRepos: _*), countIssue(condition.copy(state = "closed"), true, allRepos: _*),
filter match { filter match {
case "assigned" => condition.copy(assigned = Some(Some(userName))) case "assigned" => condition.copy(assigned = Some(Some(userName)))
@@ -101,9 +113,8 @@ trait DashboardControllerBase extends ControllerBase {
}, },
filter, filter,
getGroupNames(userName), getGroupNames(userName),
Nil, getVisibleRepositories(context.loginAccount, withoutPhysicalInfo = true)
getUserRepositories(userName, withoutPhysicalInfo = true)) )
} }
} }

View File

@@ -1,7 +1,9 @@
package gitbucket.core.controller package gitbucket.core.controller
import java.io.File
import gitbucket.core.model.Account import gitbucket.core.model.Account
import gitbucket.core.service.{AccountService, RepositoryService, ReleaseService} import gitbucket.core.service.{AccountService, ReleaseService, RepositoryService}
import gitbucket.core.servlet.Database import gitbucket.core.servlet.Database
import gitbucket.core.util._ import gitbucket.core.util._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
@@ -19,92 +21,135 @@ import org.apache.commons.io.{FileUtils, IOUtils}
* *
* This servlet saves uploaded file. * This servlet saves uploaded file.
*/ */
class FileUploadController extends ScalatraServlet class FileUploadController
extends ScalatraServlet
with FileUploadSupport with FileUploadSupport
with RepositoryService with RepositoryService
with AccountService with AccountService
with ReleaseService{ with ReleaseService {
configureMultipartHandling(MultipartConfig(maxFileSize = Some(FileUtil.MaxFileSize))) configureMultipartHandling(MultipartConfig(maxFileSize = Some(FileUtil.MaxFileSize)))
post("/image"){ post("/image") {
execute({ (file, fileId) => execute(
FileUtils.writeByteArrayToFile(new java.io.File(getTemporaryDir(session.getId), fileId), file.get) { (file, fileId) =>
FileUtils
.writeByteArrayToFile(new File(getTemporaryDir(session.getId), FileUtil.checkFilename(fileId)), file.get)
session += Keys.Session.Upload(fileId) -> file.name session += Keys.Session.Upload(fileId) -> file.name
}, FileUtil.isImage) },
FileUtil.isImage
)
} }
post("/tmp"){ post("/tmp") {
execute({ (file, fileId) => execute(
FileUtils.writeByteArrayToFile(new java.io.File(getTemporaryDir(session.getId), fileId), file.get) { (file, fileId) =>
FileUtils
.writeByteArrayToFile(new File(getTemporaryDir(session.getId), FileUtil.checkFilename(fileId)), file.get)
session += Keys.Session.Upload(fileId) -> file.name session += Keys.Session.Upload(fileId) -> file.name
}, _ => true) },
_ => true
)
} }
post("/file/:owner/:repository"){ post("/file/:owner/:repository") {
execute({ (file, fileId) => execute(
FileUtils.writeByteArrayToFile(new java.io.File( { (file, fileId) =>
FileUtils.writeByteArrayToFile(
new File(
getAttachedDir(params("owner"), params("repository")), getAttachedDir(params("owner"), params("repository")),
fileId + "." + FileUtil.getExtension(file.getName)), file.get) FileUtil.checkFilename(fileId + "." + FileUtil.getExtension(file.getName))
}, _ => true) ),
file.get
)
},
_ => true
)
} }
post("/wiki/:owner/:repository"){ post("/wiki/:owner/:repository") {
// Don't accept not logged-in users // Don't accept not logged-in users
session.get(Keys.Session.LoginAccount).collect { case loginAccount: Account => session.get(Keys.Session.LoginAccount).collect {
case loginAccount: Account =>
val owner = params("owner") val owner = params("owner")
val repository = params("repository") val repository = params("repository")
// Check whether logged-in user is collaborator // Check whether logged-in user is collaborator
onlyWikiEditable(owner, repository, loginAccount){ onlyWikiEditable(owner, repository, loginAccount) {
execute({ (file, fileId) => execute(
{ (file, fileId) =>
val fileName = file.getName val fileName = file.getName
LockUtil.lock(s"${owner}/${repository}/wiki") { LockUtil.lock(s"${owner}/${repository}/wiki") {
using(Git.open(Directory.getWikiRepositoryDir(owner, repository))) { git => using(Git.open(Directory.getWikiRepositoryDir(owner, repository))) {
git =>
val builder = DirCache.newInCore.builder() val builder = DirCache.newInCore.builder()
val inserter = git.getRepository.newObjectInserter() val inserter = git.getRepository.newObjectInserter()
val headId = git.getRepository.resolve(Constants.HEAD + "^{commit}") val headId = git.getRepository.resolve(Constants.HEAD + "^{commit}")
if(headId != null){ if (headId != null) {
JGitUtil.processTree(git, headId){ (path, tree) => JGitUtil.processTree(git, headId) { (path, tree) =>
if(path != fileName){ if (path != fileName) {
builder.add(JGitUtil.createDirCacheEntry(path, tree.getEntryFileMode, tree.getEntryObjectId)) builder.add(JGitUtil.createDirCacheEntry(path, tree.getEntryFileMode, tree.getEntryObjectId))
} }
} }
} }
val bytes = IOUtils.toByteArray(file.getInputStream) val bytes = IOUtils.toByteArray(file.getInputStream)
builder.add(JGitUtil.createDirCacheEntry(fileName, FileMode.REGULAR_FILE, inserter.insert(Constants.OBJ_BLOB, bytes))) builder.add(
JGitUtil.createDirCacheEntry(
fileName,
FileMode.REGULAR_FILE,
inserter.insert(Constants.OBJ_BLOB, bytes)
)
)
builder.finish() builder.finish()
val newHeadId = JGitUtil.createNewCommit(git, inserter, headId, builder.getDirCache.writeTree(inserter), val newHeadId = JGitUtil.createNewCommit(
Constants.HEAD, loginAccount.fullName, loginAccount.mailAddress, s"Uploaded ${fileName}") git,
inserter,
headId,
builder.getDirCache.writeTree(inserter),
Constants.HEAD,
loginAccount.fullName,
loginAccount.mailAddress,
s"Uploaded ${fileName}"
)
fileName fileName
} }
} }
}, _ => true) },
_ => true
)
} }
} getOrElse BadRequest() } getOrElse BadRequest()
} }
post("/release/:owner/:repository/:tag"){ post("/release/:owner/:repository/:tag") {
session.get(Keys.Session.LoginAccount).collect { case _: Account => session
.get(Keys.Session.LoginAccount)
.collect {
case _: Account =>
val owner = params("owner") val owner = params("owner")
val repository = params("repository") val repository = params("repository")
val tag = params("tag") val tag = params("tag")
execute({ (file, fileId) => execute(
{ (file, fileId) =>
FileUtils.writeByteArrayToFile( FileUtils.writeByteArrayToFile(
new java.io.File(getReleaseFilesDir(owner, repository), tag + "/" + fileId), new File(getReleaseFilesDir(owner, repository), FileUtil.checkFilename(tag + "/" + fileId)),
file.get file.get
) )
}, _ => true) },
}.getOrElse(BadRequest()) _ => true
)
}
.getOrElse(BadRequest())
} }
post("/import") { post("/import") {
import JDBCUtil._ import JDBCUtil._
session.get(Keys.Session.LoginAccount).collect { case loginAccount: Account if loginAccount.isAdmin => session.get(Keys.Session.LoginAccount).collect {
case loginAccount: Account if loginAccount.isAdmin =>
execute({ (file, fileId) => execute({ (file, fileId) =>
request2Session(request).conn.importAsSQL(file.getInputStream) request2Session(request).conn.importAsSQL(file.getInputStream)
}, _ => true) }, _ => true)
@@ -115,7 +160,8 @@ class FileUploadController extends ScalatraServlet
private def onlyWikiEditable(owner: String, repository: String, loginAccount: Account)(action: => Any): Any = { private def onlyWikiEditable(owner: String, repository: String, loginAccount: Account)(action: => Any): Any = {
implicit val session = Database.getSession(request) implicit val session = Database.getSession(request)
getRepository(owner, repository) match { getRepository(owner, repository) match {
case Some(x) => x.repository.options.wikiOption match { case Some(x) =>
x.repository.options.wikiOption match {
case "ALL" if !x.repository.isPrivate => action case "ALL" if !x.repository.isPrivate => action
case "PUBLIC" if hasGuestRole(owner, repository, Some(loginAccount)) => action case "PUBLIC" if hasGuestRole(owner, repository, Some(loginAccount)) => action
case "PRIVATE" if hasDeveloperRole(owner, repository, Some(loginAccount)) => action case "PRIVATE" if hasDeveloperRole(owner, repository, Some(loginAccount)) => action
@@ -125,9 +171,10 @@ class FileUploadController extends ScalatraServlet
} }
} }
private def execute(f: (FileItem, String) => Unit , mimeTypeChcker: (String) => Boolean) = fileParams.get("file") match { private def execute(f: (FileItem, String) => Unit, mimeTypeChcker: (String) => Boolean) =
case Some(file) if(mimeTypeChcker(file.name)) => fileParams.get("file") match {
defining(FileUtil.generateFileId){ fileId => case Some(file) if (mimeTypeChcker(file.name)) =>
defining(FileUtil.generateFileId) { fileId =>
f(file, fileId) f(file, fileId)
contentType = "text/plain" contentType = "text/plain"
Ok(fileId) Ok(fileId)

View File

@@ -9,23 +9,26 @@ import gitbucket.core.model.Account
import gitbucket.core.service._ import gitbucket.core.service._
import gitbucket.core.util.Implicits._ import gitbucket.core.util.Implicits._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
import gitbucket.core.util.{Keys, LDAPUtil, ReferrerAuthenticator, UsersAuthenticator} import gitbucket.core.util._
import org.scalatra.Ok import org.scalatra.Ok
import org.scalatra.forms._ import org.scalatra.forms._
class IndexController
class IndexController extends IndexControllerBase extends IndexControllerBase
with RepositoryService with RepositoryService
with ActivityService with ActivityService
with AccountService with AccountService
with RepositorySearchService with RepositorySearchService
with IssuesService with IssuesService
with LabelsService
with MilestonesService
with PrioritiesService
with UsersAuthenticator with UsersAuthenticator
with ReferrerAuthenticator with ReferrerAuthenticator
with AccessTokenService
with AccountFederationService with AccountFederationService
with OpenIDConnectService with OpenIDConnectService
trait IndexControllerBase extends ControllerBase { trait IndexControllerBase extends ControllerBase {
self: RepositoryService self: RepositoryService
with ActivityService with ActivityService
@@ -33,6 +36,8 @@ trait IndexControllerBase extends ControllerBase {
with RepositorySearchService with RepositorySearchService
with UsersAuthenticator with UsersAuthenticator
with ReferrerAuthenticator with ReferrerAuthenticator
with AccessTokenService
with AccountFederationService
with OpenIDConnectService => with OpenIDConnectService =>
case class SignInForm(userName: String, password: String, hash: Option[String]) case class SignInForm(userName: String, password: String, hash: Option[String])
@@ -53,24 +58,36 @@ trait IndexControllerBase extends ControllerBase {
case class OidcContext(state: State, nonce: Nonce, redirectBackURI: String) case class OidcContext(state: State, nonce: Nonce, redirectBackURI: String)
get("/"){ get("/") {
context.loginAccount.map { account => context.loginAccount
.map { account =>
val visibleOwnerSet: Set[String] = Set(account.userName) ++ getGroupsByUserName(account.userName) val visibleOwnerSet: Set[String] = Set(account.userName) ++ getGroupsByUserName(account.userName)
gitbucket.core.html.index(getRecentActivitiesByOwners(visibleOwnerSet), Nil, getUserRepositories(account.userName, withoutPhysicalInfo = true)) gitbucket.core.html.index(
}.getOrElse { getRecentActivitiesByOwners(visibleOwnerSet),
gitbucket.core.html.index(getRecentActivities(), getVisibleRepositories(None, withoutPhysicalInfo = true), Nil) getVisibleRepositories(Some(account), withoutPhysicalInfo = true),
showBannerToCreatePersonalAccessToken = hasAccountFederation(account.userName) && !hasAccessToken(
account.userName
)
)
}
.getOrElse {
gitbucket.core.html.index(
getRecentActivities(),
getVisibleRepositories(None, withoutPhysicalInfo = true),
showBannerToCreatePersonalAccessToken = false
)
} }
} }
get("/signin"){ get("/signin") {
val redirect = params.get("redirect") val redirect = params.get("redirect")
if(redirect.isDefined && redirect.get.startsWith("/")){ if (redirect.isDefined && redirect.get.startsWith("/")) {
flash += Keys.Flash.Redirect -> redirect.get flash += Keys.Flash.Redirect -> redirect.get
} }
gitbucket.core.html.signin(flash.get("userName"), flash.get("password"), flash.get("error")) gitbucket.core.html.signin(flash.get("userName"), flash.get("password"), flash.get("error"))
} }
post("/signin", signinForm){ form => post("/signin", signinForm) { form =>
authenticate(context.settings, form.userName, form.password) match { authenticate(context.settings, form.userName, form.password) match {
case Some(account) => case Some(account) =>
flash.get(Keys.Flash.Redirect) match { flash.get(Keys.Flash.Redirect) match {
@@ -96,7 +113,10 @@ trait IndexControllerBase extends ControllerBase {
case Some(redirectBackURI: String) => redirectBackURI + params.getOrElse("hash", "") case Some(redirectBackURI: String) => redirectBackURI + params.getOrElse("hash", "")
case _ => "/" case _ => "/"
} }
session.setAttribute(Keys.Session.OidcContext, OidcContext(authenticationRequest.getState, authenticationRequest.getNonce, redirectBackURI)) session.setAttribute(
Keys.Session.OidcContext,
OidcContext(authenticationRequest.getState, authenticationRequest.getNonce, redirectBackURI)
)
redirect(authenticationRequest.toURI.toString) redirect(authenticationRequest.toURI.toString)
} getOrElse { } getOrElse {
NotFound() NotFound()
@@ -128,18 +148,18 @@ trait IndexControllerBase extends ControllerBase {
} }
} }
get("/signout"){ get("/signout") {
session.invalidate session.invalidate
redirect("/") redirect("/")
} }
get("/activities.atom"){ get("/activities.atom") {
contentType = "application/atom+xml; type=feed" contentType = "application/atom+xml; type=feed"
xml.feed(getRecentActivities()) xml.feed(getRecentActivities())
} }
post("/sidebar-collapse"){ post("/sidebar-collapse") {
if(params("collapse") == "true"){ if (params("collapse") == "true") {
session.setAttribute("sidebar-collapse", "true") session.setAttribute("sidebar-collapse", "true")
} else { } else {
session.setAttribute("sidebar-collapse", null) session.setAttribute("sidebar-collapse", null)
@@ -154,7 +174,7 @@ trait IndexControllerBase extends ControllerBase {
session.setAttribute(Keys.Session.LoginAccount, account) session.setAttribute(Keys.Session.LoginAccount, account)
updateLastLoginDate(account.userName) updateLastLoginDate(account.userName)
if(LDAPUtil.isDummyMailAddress(account)) { if (LDAPUtil.isDummyMailAddress(account)) {
redirect("/" + account.userName + "/_edit") redirect("/" + account.userName + "/_edit")
} }
@@ -173,20 +193,25 @@ trait IndexControllerBase extends ControllerBase {
val user = params("user").toBoolean val user = params("user").toBoolean
val group = params("group").toBoolean val group = params("group").toBoolean
org.json4s.jackson.Serialization.write( org.json4s.jackson.Serialization.write(
Map("options" -> ( Map(
"options" -> (
getAllUsers(false) getAllUsers(false)
.withFilter { t => (user, group) match { .withFilter { t =>
(user, group) match {
case (true, true) => true case (true, true) => true
case (true, false) => !t.isGroupAccount case (true, false) => !t.isGroupAccount
case (false, true) => t.isGroupAccount case (false, true) => t.isGroupAccount
case (false, false) => false case (false, false) => false
}}.map { t => }
}
.map { t =>
Map( Map(
"label" -> s"<b>@${t.userName}</b> ${t.fullName}", "label" -> s"<b>@${StringUtil.escapeHtml(t.userName)}</b> ${StringUtil.escapeHtml(t.fullName)}",
"value" -> t.userName "value" -> t.userName
) )
} }
)) )
)
) )
}) })
@@ -195,48 +220,68 @@ trait IndexControllerBase extends ControllerBase {
* Returns a single string which is any of "group", "user" or "". * Returns a single string which is any of "group", "user" or "".
*/ */
post("/_user/existence")(usersOnly { post("/_user/existence")(usersOnly {
getAccountByUserName(params("userName")).map { account => getAccountByUserNameIgnoreCase(params("userName")).map { account =>
if(account.isGroupAccount) "group" else "user" if (account.isGroupAccount) "group" else "user"
} getOrElse "" } getOrElse ""
}) })
// TODO Move to RepositoryViwerController? // TODO Move to RepositoryViwerController?
get("/:owner/:repository/search")(referrersOnly { repository => get("/:owner/:repository/search")(referrersOnly { repository =>
defining(params.getOrElse("q", "").trim, params.getOrElse("type", "code")){ case (query, target) => defining(params.getOrElse("q", "").trim, params.getOrElse("type", "code")) {
case (query, target) =>
val page = try { val page = try {
val i = params.getOrElse("page", "1").toInt val i = params.getOrElse("page", "1").toInt
if(i <= 0) 1 else i if (i <= 0) 1 else i
} catch { } catch {
case e: NumberFormatException => 1 case e: NumberFormatException => 1
} }
target.toLowerCase match { target.toLowerCase match {
case "issue" => gitbucket.core.search.html.issues( case "issues" =>
if(query.nonEmpty) searchIssues(repository.owner, repository.name, query) else Nil, gitbucket.core.search.html.issues(
query, page, repository) if (query.nonEmpty) searchIssues(repository.owner, repository.name, query, false) else Nil,
false,
query,
page,
repository
)
case "wiki" => gitbucket.core.search.html.wiki( case "pulls" =>
if(query.nonEmpty) searchWikiPages(repository.owner, repository.name, query) else Nil, gitbucket.core.search.html.issues(
query, page, repository) if (query.nonEmpty) searchIssues(repository.owner, repository.name, query, true) else Nil,
true,
query,
page,
repository
)
case _ => gitbucket.core.search.html.code( case "wiki" =>
if(query.nonEmpty) searchFiles(repository.owner, repository.name, query) else Nil, gitbucket.core.search.html.wiki(
query, page, repository) if (query.nonEmpty) searchWikiPages(repository.owner, repository.name, query) else Nil,
query,
page,
repository
)
case _ =>
gitbucket.core.search.html.code(
if (query.nonEmpty) searchFiles(repository.owner, repository.name, query) else Nil,
query,
page,
repository
)
} }
} }
}) })
get("/search"){ get("/search") {
val query = params.getOrElse("query", "").trim.toLowerCase val query = params.getOrElse("query", "").trim.toLowerCase
val visibleRepositories = getVisibleRepositories(context.loginAccount, repositoryUserName = None, withoutPhysicalInfo = true) val visibleRepositories =
getVisibleRepositories(context.loginAccount, repositoryUserName = None, withoutPhysicalInfo = true)
val repositories = visibleRepositories.filter { repository => val repositories = visibleRepositories.filter { repository =>
repository.name.toLowerCase.indexOf(query) >= 0 || repository.owner.toLowerCase.indexOf(query) >= 0 repository.name.toLowerCase.indexOf(query) >= 0 || repository.owner.toLowerCase.indexOf(query) >= 0
} }
context.loginAccount.map { account => gitbucket.core.search.html.repositories(query, repositories, visibleRepositories)
gitbucket.core.search.html.repositories(query, repositories, Nil, getUserRepositories(account.userName, withoutPhysicalInfo = true))
}.getOrElse {
gitbucket.core.search.html.repositories(query, repositories, visibleRepositories, Nil)
}
} }
} }

View File

@@ -11,8 +11,8 @@ import gitbucket.core.view.Markdown
import org.scalatra.forms._ import org.scalatra.forms._
import org.scalatra.{BadRequest, Ok} import org.scalatra.{BadRequest, Ok}
class IssuesController
class IssuesController extends IssuesControllerBase extends IssuesControllerBase
with IssuesService with IssuesService
with RepositoryService with RepositoryService
with AccountService with AccountService
@@ -45,8 +45,14 @@ trait IssuesControllerBase extends ControllerBase {
with WebHookIssueCommentService with WebHookIssueCommentService
with PrioritiesService => with PrioritiesService =>
case class IssueCreateForm(title: String, content: Option[String], case class IssueCreateForm(
assignedUserName: Option[String], milestoneId: Option[Int], priorityId: Option[Int], labelNames: Option[String]) title: String,
content: Option[String],
assignedUserName: Option[String],
milestoneId: Option[Int],
priorityId: Option[Int],
labelNames: Option[String]
)
case class CommentForm(issueId: Int, content: String) case class CommentForm(issueId: Int, content: String)
case class IssueStateForm(issueId: Int, content: Option[String]) case class IssueStateForm(issueId: Int, content: Option[String])
@@ -78,7 +84,7 @@ trait IssuesControllerBase extends ControllerBase {
get("/:owner/:repository/issues")(referrersOnly { repository => get("/:owner/:repository/issues")(referrersOnly { repository =>
val q = request.getParameter("q") val q = request.getParameter("q")
if(Option(q).exists(_.contains("is:pr"))){ if (Option(q).exists(_.contains("is:pr"))) {
redirect(s"/${repository.owner}/${repository.name}/pulls?q=${StringUtil.urlEncode(q)}") redirect(s"/${repository.owner}/${repository.name}/pulls?q=${StringUtil.urlEncode(q)}")
} else { } else {
searchIssues(repository) searchIssues(repository)
@@ -86,9 +92,11 @@ trait IssuesControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/issues/:id")(referrersOnly { repository => get("/:owner/:repository/issues/:id")(referrersOnly { repository =>
defining(repository.owner, repository.name, params("id")){ case (owner, name, issueId) => defining(repository.owner, repository.name, params("id")) {
getIssue(owner, name, issueId) map { issue => case (owner, name, issueId) =>
if(issue.isPullRequest){ getIssue(owner, name, issueId) map {
issue =>
if (issue.isPullRequest) {
redirect(s"/${repository.owner}/${repository.name}/pull/${issueId}") redirect(s"/${repository.owner}/${repository.name}/pull/${issueId}")
} else { } else {
html.issue( html.issue(
@@ -101,15 +109,17 @@ trait IssuesControllerBase extends ControllerBase {
getLabels(owner, name), getLabels(owner, name),
isIssueEditable(repository), isIssueEditable(repository),
isIssueManageable(repository), isIssueManageable(repository),
repository) repository
)
} }
} getOrElse NotFound() } getOrElse NotFound()
} }
}) })
get("/:owner/:repository/issues/new")(readableUsersOnly { repository => get("/:owner/:repository/issues/new")(readableUsersOnly { repository =>
if(isIssueEditable(repository)){ // TODO Should this check is provided by authenticator? if (isIssueEditable(repository)) { // TODO Should this check is provided by authenticator?
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
case (owner, name) =>
html.create( html.create(
getAssignableUserNames(owner, name), getAssignableUserNames(owner, name),
getMilestones(owner, name), getMilestones(owner, name),
@@ -118,13 +128,14 @@ trait IssuesControllerBase extends ControllerBase {
getLabels(owner, name), getLabels(owner, name),
isIssueManageable(repository), isIssueManageable(repository),
getContentTemplate(repository, "ISSUE_TEMPLATE"), getContentTemplate(repository, "ISSUE_TEMPLATE"),
repository) repository
)
} }
} else Unauthorized() } else Unauthorized()
}) })
post("/:owner/:repository/issues/new", issueCreateForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/issues/new", issueCreateForm)(readableUsersOnly { (form, repository) =>
if(isIssueEditable(repository)){ // TODO Should this check is provided by authenticator? if (isIssueEditable(repository)) { // TODO Should this check is provided by authenticator?
val issue = createIssue( val issue = createIssue(
repository, repository,
form.title, form.title,
@@ -133,21 +144,33 @@ trait IssuesControllerBase extends ControllerBase {
form.milestoneId, form.milestoneId,
form.priorityId, form.priorityId,
form.labelNames.toArray.flatMap(_.split(",")), form.labelNames.toArray.flatMap(_.split(",")),
context.loginAccount.get) context.loginAccount.get
)
redirect(s"/${issue.userName}/${issue.repositoryName}/issues/${issue.issueId}") redirect(s"/${issue.userName}/${issue.repositoryName}/issues/${issue.issueId}")
} else Unauthorized() } else Unauthorized()
}) })
ajaxPost("/:owner/:repository/issues/edit_title/:id", issueTitleEditForm)(readableUsersOnly { (title, repository) => ajaxPost("/:owner/:repository/issues/edit_title/:id", issueTitleEditForm)(readableUsersOnly { (title, repository) =>
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
getIssue(owner, name, params("id")).map { issue => case (owner, name) =>
if(isEditableContent(owner, name, issue.openedUserName)){ getIssue(owner, name, params("id")).map {
issue =>
if (isEditableContent(owner, name, issue.openedUserName)) {
if (issue.title != title) {
// update issue // update issue
updateIssue(owner, name, issue.issueId, title, issue.content) updateIssue(owner, name, issue.issueId, title, issue.content)
// extract references and create refer comment // extract references and create refer comment
createReferComment(owner, name, issue.copy(title = title), title, context.loginAccount.get) createReferComment(owner, name, issue.copy(title = title), title, context.loginAccount.get)
createComment(
owner,
name,
context.loginAccount.get.userName,
issue.issueId,
issue.title + "\r\n" + title,
"change_title"
)
}
redirect(s"/${owner}/${name}/issues/_data/${issue.issueId}") redirect(s"/${owner}/${name}/issues/_data/${issue.issueId}")
} else Unauthorized() } else Unauthorized()
} getOrElse NotFound() } getOrElse NotFound()
@@ -155,9 +178,10 @@ trait IssuesControllerBase extends ControllerBase {
}) })
ajaxPost("/:owner/:repository/issues/edit/:id", issueEditForm)(readableUsersOnly { (content, repository) => ajaxPost("/:owner/:repository/issues/edit/:id", issueEditForm)(readableUsersOnly { (content, repository) =>
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
case (owner, name) =>
getIssue(owner, name, params("id")).map { issue => getIssue(owner, name, params("id")).map { issue =>
if(isEditableContent(owner, name, issue.openedUserName)){ if (isEditableContent(owner, name, issue.openedUserName)) {
// update issue // update issue
updateIssue(owner, name, issue.issueId, issue.title, content) updateIssue(owner, name, issue.issueId, issue.title, content)
// extract references and create refer comment // extract references and create refer comment
@@ -171,28 +195,35 @@ trait IssuesControllerBase extends ControllerBase {
post("/:owner/:repository/issue_comments/new", commentForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/issue_comments/new", commentForm)(readableUsersOnly { (form, repository) =>
getIssue(repository.owner, repository.name, form.issueId.toString).flatMap { issue => getIssue(repository.owner, repository.name, form.issueId.toString).flatMap { issue =>
val actionOpt = params.get("action").filter(_ => isEditableContent(issue.userName, issue.repositoryName, issue.openedUserName)) val actionOpt =
handleComment(issue, Some(form.content), repository, actionOpt) map { case (issue, id) => params.get("action").filter(_ => isEditableContent(issue.userName, issue.repositoryName, issue.openedUserName))
redirect(s"/${repository.owner}/${repository.name}/${ handleComment(issue, Some(form.content), repository, actionOpt) map {
if(issue.isPullRequest) "pull" else "issues"}/${form.issueId}#comment-${id}") case (issue, id) =>
redirect(
s"/${repository.owner}/${repository.name}/${if (issue.isPullRequest) "pull" else "issues"}/${form.issueId}#comment-${id}"
)
} }
} getOrElse NotFound() } getOrElse NotFound()
}) })
post("/:owner/:repository/issue_comments/state", issueStateForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/issue_comments/state", issueStateForm)(readableUsersOnly { (form, repository) =>
getIssue(repository.owner, repository.name, form.issueId.toString).flatMap { issue => getIssue(repository.owner, repository.name, form.issueId.toString).flatMap { issue =>
val actionOpt = params.get("action").filter(_ => isEditableContent(issue.userName, issue.repositoryName, issue.openedUserName)) val actionOpt =
handleComment(issue, form.content, repository, actionOpt) map { case (issue, id) => params.get("action").filter(_ => isEditableContent(issue.userName, issue.repositoryName, issue.openedUserName))
redirect(s"/${repository.owner}/${repository.name}/${ handleComment(issue, form.content, repository, actionOpt) map {
if(issue.isPullRequest) "pull" else "issues"}/${form.issueId}#comment-${id}") case (issue, id) =>
redirect(
s"/${repository.owner}/${repository.name}/${if (issue.isPullRequest) "pull" else "issues"}/${form.issueId}#comment-${id}"
)
} }
} getOrElse NotFound() } getOrElse NotFound()
}) })
ajaxPost("/:owner/:repository/issue_comments/edit/:id", commentForm)(readableUsersOnly { (form, repository) => ajaxPost("/:owner/:repository/issue_comments/edit/:id", commentForm)(readableUsersOnly { (form, repository) =>
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
case (owner, name) =>
getComment(owner, name, params("id")).map { comment => getComment(owner, name, params("id")).map { comment =>
if(isEditableContent(owner, name, comment.commentedUserName)){ if (isEditableContent(owner, name, comment.commentedUserName)) {
updateComment(comment.issueId, comment.commentId, form.content) updateComment(comment.issueId, comment.commentId, form.content)
redirect(s"/${owner}/${name}/issue_comments/_data/${comment.commentId}") redirect(s"/${owner}/${name}/issue_comments/_data/${comment.commentId}")
} else Unauthorized() } else Unauthorized()
@@ -201,9 +232,10 @@ trait IssuesControllerBase extends ControllerBase {
}) })
ajaxPost("/:owner/:repository/issue_comments/delete/:id")(readableUsersOnly { repository => ajaxPost("/:owner/:repository/issue_comments/delete/:id")(readableUsersOnly { repository =>
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
case (owner, name) =>
getComment(owner, name, params("id")).map { comment => getComment(owner, name, params("id")).map { comment =>
if(isEditableContent(owner, name, comment.commentedUserName)){ if (isEditableContent(owner, name, comment.commentedUserName)) {
Ok(deleteComment(comment.issueId, comment.commentId)) Ok(deleteComment(comment.issueId, comment.commentId))
} else Unauthorized() } else Unauthorized()
} getOrElse NotFound() } getOrElse NotFound()
@@ -211,8 +243,9 @@ trait IssuesControllerBase extends ControllerBase {
}) })
ajaxGet("/:owner/:repository/issues/_data/:id")(readableUsersOnly { repository => ajaxGet("/:owner/:repository/issues/_data/:id")(readableUsersOnly { repository =>
getIssue(repository.owner, repository.name, params("id")) map { x => getIssue(repository.owner, repository.name, params("id")) map {
if(isEditableContent(x.userName, x.repositoryName, x.openedUserName)){ x =>
if (isEditableContent(x.userName, x.repositoryName, x.openedUserName)) {
params.get("dataType") collect { params.get("dataType") collect {
case t if t == "html" => html.editissue(x.content, x.issueId, repository) case t if t == "html" => html.editissue(x.content, x.issueId, repository)
} getOrElse { } getOrElse {
@@ -238,8 +271,9 @@ trait IssuesControllerBase extends ControllerBase {
}) })
ajaxGet("/:owner/:repository/issue_comments/_data/:id")(readableUsersOnly { repository => ajaxGet("/:owner/:repository/issue_comments/_data/:id")(readableUsersOnly { repository =>
getComment(repository.owner, repository.name, params("id")) map { x => getComment(repository.owner, repository.name, params("id")) map {
if(isEditableContent(x.userName, x.repositoryName, x.commentedUserName)){ x =>
if (isEditableContent(x.userName, x.repositoryName, x.commentedUserName)) {
params.get("dataType") collect { params.get("dataType") collect {
case t if t == "html" => html.editcomment(x.content, x.commentId, repository) case t if t == "html" => html.editcomment(x.content, x.commentId, repository)
} getOrElse { } getOrElse {
@@ -270,48 +304,60 @@ trait IssuesControllerBase extends ControllerBase {
}) })
ajaxPost("/:owner/:repository/issues/:id/label/new")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/:id/label/new")(writableUsersOnly { repository =>
defining(params("id").toInt){ issueId => defining(params("id").toInt) { issueId =>
registerIssueLabel(repository.owner, repository.name, issueId, params("labelId").toInt) registerIssueLabel(repository.owner, repository.name, issueId, params("labelId").toInt, true)
html.labellist(getIssueLabels(repository.owner, repository.name, issueId)) html.labellist(getIssueLabels(repository.owner, repository.name, issueId))
} }
}) })
ajaxPost("/:owner/:repository/issues/:id/label/delete")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/:id/label/delete")(writableUsersOnly { repository =>
defining(params("id").toInt){ issueId => defining(params("id").toInt) { issueId =>
deleteIssueLabel(repository.owner, repository.name, issueId, params("labelId").toInt) deleteIssueLabel(repository.owner, repository.name, issueId, params("labelId").toInt, true)
html.labellist(getIssueLabels(repository.owner, repository.name, issueId)) html.labellist(getIssueLabels(repository.owner, repository.name, issueId))
} }
}) })
ajaxPost("/:owner/:repository/issues/:id/assign")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/:id/assign")(writableUsersOnly { repository =>
updateAssignedUserName(repository.owner, repository.name, params("id").toInt, assignedUserName("assignedUserName")) updateAssignedUserName(
repository.owner,
repository.name,
params("id").toInt,
assignedUserName("assignedUserName"),
true
)
Ok("updated") Ok("updated")
}) })
ajaxPost("/:owner/:repository/issues/:id/milestone")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/:id/milestone")(writableUsersOnly { repository =>
updateMilestoneId(repository.owner, repository.name, params("id").toInt, milestoneId("milestoneId")) updateMilestoneId(repository.owner, repository.name, params("id").toInt, milestoneId("milestoneId"), true)
milestoneId("milestoneId").map { milestoneId => milestoneId("milestoneId").map { milestoneId =>
getMilestonesWithIssueCount(repository.owner, repository.name) getMilestonesWithIssueCount(repository.owner, repository.name)
.find(_._1.milestoneId == milestoneId).map { case (_, openCount, closeCount) => .find(_._1.milestoneId == milestoneId)
.map {
case (_, openCount, closeCount) =>
gitbucket.core.issues.milestones.html.progress(openCount + closeCount, closeCount) gitbucket.core.issues.milestones.html.progress(openCount + closeCount, closeCount)
} getOrElse NotFound() } getOrElse NotFound()
} getOrElse Ok() } getOrElse Ok()
}) })
ajaxPost("/:owner/:repository/issues/:id/priority")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/:id/priority")(writableUsersOnly { repository =>
updatePriorityId(repository.owner, repository.name, params("id").toInt, priorityId("priorityId")) val priority = priorityId("priorityId")
updatePriorityId(repository.owner, repository.name, params("id").toInt, priority, true)
Ok("updated") Ok("updated")
}) })
post("/:owner/:repository/issues/batchedit/state")(writableUsersOnly { repository => post("/:owner/:repository/issues/batchedit/state")(writableUsersOnly { repository =>
defining(params.get("value")){ action => defining(params.get("value")) {
action =>
action match { action match {
case Some("open") => executeBatch(repository) { issueId => case Some("open") =>
executeBatch(repository) { issueId =>
getIssue(repository.owner, repository.name, issueId.toString).foreach { issue => getIssue(repository.owner, repository.name, issueId.toString).foreach { issue =>
handleComment(issue, None, repository, Some("reopen")) handleComment(issue, None, repository, Some("reopen"))
} }
} }
case Some("close") => executeBatch(repository) { issueId => case Some("close") =>
executeBatch(repository) { issueId =>
getIssue(repository.owner, repository.name, issueId.toString).foreach { issue => getIssue(repository.owner, repository.name, issueId.toString).foreach { issue =>
handleComment(issue, None, repository, Some("close")) handleComment(issue, None, repository, Some("close"))
} }
@@ -322,45 +368,45 @@ trait IssuesControllerBase extends ControllerBase {
}) })
post("/:owner/:repository/issues/batchedit/label")(writableUsersOnly { repository => post("/:owner/:repository/issues/batchedit/label")(writableUsersOnly { repository =>
params("value").toIntOpt.map{ labelId => params("value").toIntOpt.map { labelId =>
executeBatch(repository) { issueId => executeBatch(repository) { issueId =>
getIssueLabel(repository.owner, repository.name, issueId, labelId) getOrElse { getIssueLabel(repository.owner, repository.name, issueId, labelId) getOrElse {
registerIssueLabel(repository.owner, repository.name, issueId, labelId) registerIssueLabel(repository.owner, repository.name, issueId, labelId, true)
} }
} }
} getOrElse NotFound() } getOrElse NotFound()
}) })
post("/:owner/:repository/issues/batchedit/assign")(writableUsersOnly { repository => post("/:owner/:repository/issues/batchedit/assign")(writableUsersOnly { repository =>
defining(assignedUserName("value")){ value => defining(assignedUserName("value")) { value =>
executeBatch(repository) { executeBatch(repository) {
updateAssignedUserName(repository.owner, repository.name, _, value) updateAssignedUserName(repository.owner, repository.name, _, value, true)
} }
} }
}) })
post("/:owner/:repository/issues/batchedit/milestone")(writableUsersOnly { repository => post("/:owner/:repository/issues/batchedit/milestone")(writableUsersOnly { repository =>
defining(milestoneId("value")){ value => defining(milestoneId("value")) { value =>
executeBatch(repository) { executeBatch(repository) {
updateMilestoneId(repository.owner, repository.name, _, value) updateMilestoneId(repository.owner, repository.name, _, value, true)
} }
} }
}) })
post("/:owner/:repository/issues/batchedit/priority")(writableUsersOnly { repository => post("/:owner/:repository/issues/batchedit/priority")(writableUsersOnly { repository =>
defining(priorityId("value")){ value => defining(priorityId("value")) { value =>
executeBatch(repository) { executeBatch(repository) {
updatePriorityId(repository.owner, repository.name, _, value) updatePriorityId(repository.owner, repository.name, _, value, true)
} }
} }
}) })
get("/:owner/:repository/_attached/:file")(referrersOnly { repository => get("/:owner/:repository/_attached/:file")(referrersOnly { repository =>
(Directory.getAttachedDir(repository.owner, repository.name) match { (Directory.getAttachedDir(repository.owner, repository.name) match {
case dir if(dir.exists && dir.isDirectory) => case dir if (dir.exists && dir.isDirectory) =>
dir.listFiles.find(_.getName.startsWith(params("file") + ".")).map { file => dir.listFiles.find(_.getName.startsWith(params("file") + ".")).map { file =>
response.setHeader("Content-Disposition", f"""inline; filename=${file.getName}""") response.setHeader("Content-Disposition", f"""inline; filename=${file.getName}""")
RawData(FileUtil.getMimeType(file.getName), file) RawData(FileUtil.getSafeMimeType(file.getName), file)
} }
case _ => None case _ => None
}) getOrElse NotFound() }) getOrElse NotFound()
@@ -371,7 +417,7 @@ trait IssuesControllerBase extends ControllerBase {
val priorityId: String => Option[Int] = (key: String) => params.get(key).flatMap(_.toIntOpt) val priorityId: String => Option[Int] = (key: String) => params.get(key).flatMap(_.toIntOpt)
private def executeBatch(repository: RepositoryService.RepositoryInfo)(execute: Int => Unit) = { private def executeBatch(repository: RepositoryService.RepositoryInfo)(execute: Int => Unit) = {
params("checked").split(',') map(_.toInt) foreach execute params("checked").split(',') map (_.toInt) foreach execute
params("from") match { params("from") match {
case "issues" => redirect(s"/${repository.owner}/${repository.name}/issues") case "issues" => redirect(s"/${repository.owner}/${repository.name}/issues")
case "pulls" => redirect(s"/${repository.owner}/${repository.name}/pulls") case "pulls" => redirect(s"/${repository.owner}/${repository.name}/pulls")
@@ -379,7 +425,8 @@ trait IssuesControllerBase extends ControllerBase {
} }
private def searchIssues(repository: RepositoryService.RepositoryInfo) = { private def searchIssues(repository: RepositoryService.RepositoryInfo) = {
defining(repository.owner, repository.name){ case (owner, repoName) => defining(repository.owner, repository.name) {
case (owner, repoName) =>
val page = IssueSearchCondition.page(request) val page = IssueSearchCondition.page(request)
// retrieve search condition // retrieve search condition
@@ -393,19 +440,22 @@ trait IssuesControllerBase extends ControllerBase {
getMilestones(owner, repoName), getMilestones(owner, repoName),
getPriorities(owner, repoName), getPriorities(owner, repoName),
getLabels(owner, repoName), getLabels(owner, repoName),
countIssue(condition.copy(state = "open" ), false, owner -> repoName), countIssue(condition.copy(state = "open"), false, owner -> repoName),
countIssue(condition.copy(state = "closed"), false, owner -> repoName), countIssue(condition.copy(state = "closed"), false, owner -> repoName),
condition, condition,
repository, repository,
isIssueEditable(repository), isIssueEditable(repository),
isIssueManageable(repository)) isIssueManageable(repository)
)
} }
} }
/** /**
* Tests whether an issue or a comment is editable by a logged-in user. * Tests whether an issue or a comment is editable by a logged-in user.
*/ */
private def isEditableContent(owner: String, repository: String, author: String)(implicit context: Context): Boolean = { private def isEditableContent(owner: String, repository: String, author: String)(
implicit context: Context
): Boolean = {
hasDeveloperRole(owner, repository, context.loginAccount) || author == context.loginAccount.get.userName hasDeveloperRole(owner, repository, context.loginAccount) || author == context.loginAccount.get.userName
} }
} }

View File

@@ -1,7 +1,14 @@
package gitbucket.core.controller package gitbucket.core.controller
import gitbucket.core.issues.labels.html import gitbucket.core.issues.labels.html
import gitbucket.core.service.{RepositoryService, AccountService, IssuesService, LabelsService} import gitbucket.core.service.{
RepositoryService,
AccountService,
IssuesService,
LabelsService,
MilestonesService,
PrioritiesService
}
import gitbucket.core.util.{ReferrerAuthenticator, WritableUsersAuthenticator} import gitbucket.core.util.{ReferrerAuthenticator, WritableUsersAuthenticator}
import gitbucket.core.util.Implicits._ import gitbucket.core.util.Implicits._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
@@ -9,13 +16,23 @@ import org.scalatra.forms._
import org.scalatra.i18n.Messages import org.scalatra.i18n.Messages
import org.scalatra.Ok import org.scalatra.Ok
class LabelsController extends LabelsControllerBase class LabelsController
with LabelsService with IssuesService with RepositoryService with AccountService extends LabelsControllerBase
with ReferrerAuthenticator with WritableUsersAuthenticator with IssuesService
with RepositoryService
with AccountService
with LabelsService
with PrioritiesService
with MilestonesService
with ReferrerAuthenticator
with WritableUsersAuthenticator
trait LabelsControllerBase extends ControllerBase { trait LabelsControllerBase extends ControllerBase {
self: LabelsService with IssuesService with RepositoryService self: LabelsService
with ReferrerAuthenticator with WritableUsersAuthenticator => with IssuesService
with RepositoryService
with ReferrerAuthenticator
with WritableUsersAuthenticator =>
case class LabelForm(labelName: String, color: String) case class LabelForm(labelName: String, color: String)
@@ -24,13 +41,13 @@ trait LabelsControllerBase extends ControllerBase {
"labelColor" -> trim(label("Color", text(required, color))) "labelColor" -> trim(label("Color", text(required, color)))
)(LabelForm.apply) )(LabelForm.apply)
get("/:owner/:repository/issues/labels")(referrersOnly { repository => get("/:owner/:repository/issues/labels")(referrersOnly { repository =>
html.list( html.list(
getLabels(repository.owner, repository.name), getLabels(repository.owner, repository.name),
countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxGet("/:owner/:repository/issues/labels/new")(writableUsersOnly { repository => ajaxGet("/:owner/:repository/issues/labels/new")(writableUsersOnly { repository =>
@@ -44,7 +61,8 @@ trait LabelsControllerBase extends ControllerBase {
// TODO futility // TODO futility
countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxGet("/:owner/:repository/issues/labels/:labelId/edit")(writableUsersOnly { repository => ajaxGet("/:owner/:repository/issues/labels/:labelId/edit")(writableUsersOnly { repository =>
@@ -60,7 +78,8 @@ trait LabelsControllerBase extends ControllerBase {
// TODO futility // TODO futility
countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByLabels(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxPost("/:owner/:repository/issues/labels/:labelId/delete")(writableUsersOnly { repository => ajaxPost("/:owner/:repository/issues/labels/:labelId/delete")(writableUsersOnly { repository =>
@@ -71,24 +90,32 @@ trait LabelsControllerBase extends ControllerBase {
/** /**
* Constraint for the identifier such as user name, repository name or page name. * Constraint for the identifier such as user name, repository name or page name.
*/ */
private def labelName: Constraint = new Constraint(){ private def labelName: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(value.contains(',')){ if (value.contains(',')) {
Some(s"${name} contains invalid character.") Some(s"${name} contains invalid character.")
} else if(value.startsWith("_") || value.startsWith("-")){ } else if (value.startsWith("_") || value.startsWith("-")) {
Some(s"${name} starts with invalid character.") Some(s"${name} starts with invalid character.")
} else { } else {
None None
} }
} }
private def uniqueLabelName: Constraint = new Constraint(){ private def uniqueLabelName: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = { override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
val owner = params.value("owner") val owner = params.value("owner")
val repository = params.value("repository") val repository = params.value("repository")
params.optionValue("labelId").map { labelId => params
.optionValue("labelId")
.map { labelId =>
getLabel(owner, repository, value).filter(_.labelId != labelId.toInt).map(_ => "Name has already been taken.") getLabel(owner, repository, value).filter(_.labelId != labelId.toInt).map(_ => "Name has already been taken.")
}.getOrElse { }
.getOrElse {
getLabel(owner, repository, value).map(_ => "Name has already been taken.") getLabel(owner, repository, value).map(_ => "Name has already been taken.")
} }
} }

View File

@@ -6,13 +6,16 @@ import gitbucket.core.util.{ReferrerAuthenticator, WritableUsersAuthenticator}
import gitbucket.core.util.Implicits._ import gitbucket.core.util.Implicits._
import org.scalatra.forms._ import org.scalatra.forms._
class MilestonesController extends MilestonesControllerBase class MilestonesController
with MilestonesService with RepositoryService with AccountService extends MilestonesControllerBase
with ReferrerAuthenticator with WritableUsersAuthenticator with MilestonesService
with RepositoryService
with AccountService
with ReferrerAuthenticator
with WritableUsersAuthenticator
trait MilestonesControllerBase extends ControllerBase { trait MilestonesControllerBase extends ControllerBase {
self: MilestonesService with RepositoryService self: MilestonesService with RepositoryService with ReferrerAuthenticator with WritableUsersAuthenticator =>
with ReferrerAuthenticator with WritableUsersAuthenticator =>
case class MilestoneForm(title: String, description: Option[String], dueDate: Option[java.util.Date]) case class MilestoneForm(title: String, description: Option[String], dueDate: Option[java.util.Date])
@@ -27,7 +30,8 @@ trait MilestonesControllerBase extends ControllerBase {
params.getOrElse("state", "open"), params.getOrElse("state", "open"),
getMilestonesWithIssueCount(repository.owner, repository.name), getMilestonesWithIssueCount(repository.owner, repository.name),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
get("/:owner/:repository/issues/milestones/new")(writableUsersOnly { get("/:owner/:repository/issues/milestones/new")(writableUsersOnly {
@@ -40,13 +44,14 @@ trait MilestonesControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/issues/milestones/:milestoneId/edit")(writableUsersOnly { repository => get("/:owner/:repository/issues/milestones/:milestoneId/edit")(writableUsersOnly { repository =>
params("milestoneId").toIntOpt.map{ milestoneId => params("milestoneId").toIntOpt.map { milestoneId =>
html.edit(getMilestone(repository.owner, repository.name, milestoneId), repository) html.edit(getMilestone(repository.owner, repository.name, milestoneId), repository)
} getOrElse NotFound() } getOrElse NotFound()
}) })
post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(writableUsersOnly { (form, repository) => post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(writableUsersOnly {
params("milestoneId").toIntOpt.flatMap{ milestoneId => (form, repository) =>
params("milestoneId").toIntOpt.flatMap { milestoneId =>
getMilestone(repository.owner, repository.name, milestoneId).map { milestone => getMilestone(repository.owner, repository.name, milestoneId).map { milestone =>
updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate)) updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate))
redirect(s"/${repository.owner}/${repository.name}/issues/milestones") redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
@@ -55,7 +60,7 @@ trait MilestonesControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/issues/milestones/:milestoneId/close")(writableUsersOnly { repository => get("/:owner/:repository/issues/milestones/:milestoneId/close")(writableUsersOnly { repository =>
params("milestoneId").toIntOpt.flatMap{ milestoneId => params("milestoneId").toIntOpt.flatMap { milestoneId =>
getMilestone(repository.owner, repository.name, milestoneId).map { milestone => getMilestone(repository.owner, repository.name, milestoneId).map { milestone =>
closeMilestone(milestone) closeMilestone(milestone)
redirect(s"/${repository.owner}/${repository.name}/issues/milestones") redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
@@ -64,7 +69,7 @@ trait MilestonesControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/issues/milestones/:milestoneId/open")(writableUsersOnly { repository => get("/:owner/:repository/issues/milestones/:milestoneId/open")(writableUsersOnly { repository =>
params("milestoneId").toIntOpt.flatMap{ milestoneId => params("milestoneId").toIntOpt.flatMap { milestoneId =>
getMilestone(repository.owner, repository.name, milestoneId).map { milestone => getMilestone(repository.owner, repository.name, milestoneId).map { milestone =>
openMilestone(milestone) openMilestone(milestone)
redirect(s"/${repository.owner}/${repository.name}/issues/milestones") redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
@@ -73,7 +78,7 @@ trait MilestonesControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/issues/milestones/:milestoneId/delete")(writableUsersOnly { repository => get("/:owner/:repository/issues/milestones/:milestoneId/delete")(writableUsersOnly { repository =>
params("milestoneId").toIntOpt.flatMap{ milestoneId => params("milestoneId").toIntOpt.flatMap { milestoneId =>
getMilestone(repository.owner, repository.name, milestoneId).map { milestone => getMilestone(repository.owner, repository.name, milestoneId).map { milestone =>
deleteMilestone(repository.owner, repository.name, milestone.milestoneId) deleteMilestone(repository.owner, repository.name, milestone.milestoneId)
redirect(s"/${repository.owner}/${repository.name}/issues/milestones") redirect(s"/${repository.owner}/${repository.name}/issues/milestones")

View File

@@ -7,20 +7,21 @@ class PreProcessController extends PreProcessControllerBase
trait PreProcessControllerBase extends ControllerBase { trait PreProcessControllerBase extends ControllerBase {
/** /**
* Provides GitHub compatible URLs for Git client. * Provides GitHub compatible URLs (e.g. http://localhost:8080/owner/repo.git) for Git client.
*
* <ul>
* <li>git clone http://localhost:8080/owner/repo</li>
* <li>git clone http://localhost:8080/owner/repo.git</li>
* </ul>
*
* @see https://git-scm.com/book/en/v2/Git-Internals-Transfer-Protocols
*/ */
get("/*/*/info/refs") { get("/*/*/info/refs") {
val query = Option(request.getQueryString).map("?" + _).getOrElse("") val query = Option(request.getQueryString).map("?" + _).getOrElse("")
halt(MovedPermanently(baseUrl + "/git" + request.getRequestURI + query)) halt(MovedPermanently(baseUrl + "/git" + request.getRequestURI + query))
} }
/**
* Provides GitHub compatible URLs for GitLFS client.
*/
post("/*/*/info/lfs/objects/batch") {
val dispatcher = request.getRequestDispatcher("/git" + request.getRequestURI)
dispatcher.forward(request, response)
}
/** /**
* Filter requests from anonymous users. * Filter requests from anonymous users.
* *
@@ -28,7 +29,7 @@ trait PreProcessControllerBase extends ControllerBase {
* But if it's not allowed, demands authentication except some paths. * But if it's not allowed, demands authentication except some paths.
*/ */
get(!context.settings.allowAnonymousAccess, context.loginAccount.isEmpty) { get(!context.settings.allowAnonymousAccess, context.loginAccount.isEmpty) {
if(!context.currentPath.startsWith("/assets") && !context.currentPath.startsWith("/signin") && if (!context.currentPath.startsWith("/assets") && !context.currentPath.startsWith("/signin") &&
!context.currentPath.startsWith("/register") && !context.currentPath.endsWith("/info/refs")) { !context.currentPath.startsWith("/register") && !context.currentPath.endsWith("/info/refs")) {
Unauthorized() Unauthorized()
} else { } else {
@@ -36,5 +37,4 @@ trait PreProcessControllerBase extends ControllerBase {
} }
} }
} }

View File

@@ -1,7 +1,14 @@
package gitbucket.core.controller package gitbucket.core.controller
import gitbucket.core.issues.priorities.html import gitbucket.core.issues.priorities.html
import gitbucket.core.service.{RepositoryService, AccountService, IssuesService, PrioritiesService} import gitbucket.core.service.{
RepositoryService,
AccountService,
IssuesService,
LabelsService,
MilestonesService,
PrioritiesService
}
import gitbucket.core.util.{ReferrerAuthenticator, WritableUsersAuthenticator} import gitbucket.core.util.{ReferrerAuthenticator, WritableUsersAuthenticator}
import gitbucket.core.util.Implicits._ import gitbucket.core.util.Implicits._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
@@ -9,13 +16,23 @@ import org.scalatra.forms._
import org.scalatra.i18n.Messages import org.scalatra.i18n.Messages
import org.scalatra.Ok import org.scalatra.Ok
class PrioritiesController extends PrioritiesControllerBase class PrioritiesController
with PrioritiesService with IssuesService with RepositoryService with AccountService extends PrioritiesControllerBase
with ReferrerAuthenticator with WritableUsersAuthenticator with IssuesService
with RepositoryService
with AccountService
with LabelsService
with PrioritiesService
with MilestonesService
with ReferrerAuthenticator
with WritableUsersAuthenticator
trait PrioritiesControllerBase extends ControllerBase { trait PrioritiesControllerBase extends ControllerBase {
self: PrioritiesService with IssuesService with RepositoryService self: PrioritiesService
with ReferrerAuthenticator with WritableUsersAuthenticator => with IssuesService
with RepositoryService
with ReferrerAuthenticator
with WritableUsersAuthenticator =>
case class PriorityForm(priorityName: String, description: Option[String], color: String) case class PriorityForm(priorityName: String, description: Option[String], color: String)
@@ -25,13 +42,13 @@ trait PrioritiesControllerBase extends ControllerBase {
"priorityColor" -> trim(label("Color", text(required, color))) "priorityColor" -> trim(label("Color", text(required, color)))
)(PriorityForm.apply) )(PriorityForm.apply)
get("/:owner/:repository/issues/priorities")(referrersOnly { repository => get("/:owner/:repository/issues/priorities")(referrersOnly { repository =>
html.list( html.list(
getPriorities(repository.owner, repository.name), getPriorities(repository.owner, repository.name),
countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxGet("/:owner/:repository/issues/priorities/new")(writableUsersOnly { repository => ajaxGet("/:owner/:repository/issues/priorities/new")(writableUsersOnly { repository =>
@@ -39,12 +56,14 @@ trait PrioritiesControllerBase extends ControllerBase {
}) })
ajaxPost("/:owner/:repository/issues/priorities/new", priorityForm)(writableUsersOnly { (form, repository) => ajaxPost("/:owner/:repository/issues/priorities/new", priorityForm)(writableUsersOnly { (form, repository) =>
val priorityId = createPriority(repository.owner, repository.name, form.priorityName, form.description, form.color.substring(1)) val priorityId =
createPriority(repository.owner, repository.name, form.priorityName, form.description, form.color.substring(1))
html.priority( html.priority(
getPriority(repository.owner, repository.name, priorityId).get, getPriority(repository.owner, repository.name, priorityId).get,
countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxGet("/:owner/:repository/issues/priorities/:priorityId/edit")(writableUsersOnly { repository => ajaxGet("/:owner/:repository/issues/priorities/:priorityId/edit")(writableUsersOnly { repository =>
@@ -53,21 +72,34 @@ trait PrioritiesControllerBase extends ControllerBase {
} getOrElse NotFound() } getOrElse NotFound()
}) })
ajaxPost("/:owner/:repository/issues/priorities/:priorityId/edit", priorityForm)(writableUsersOnly { (form, repository) => ajaxPost("/:owner/:repository/issues/priorities/:priorityId/edit", priorityForm)(writableUsersOnly {
updatePriority(repository.owner, repository.name, params("priorityId").toInt, form.priorityName, form.description, form.color.substring(1)) (form, repository) =>
updatePriority(
repository.owner,
repository.name,
params("priorityId").toInt,
form.priorityName,
form.description,
form.color.substring(1)
)
html.priority( html.priority(
getPriority(repository.owner, repository.name, params("priorityId").toInt).get, getPriority(repository.owner, repository.name, params("priorityId").toInt).get,
countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty), countIssueGroupByPriorities(repository.owner, repository.name, IssuesService.IssueSearchCondition(), Map.empty),
repository, repository,
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
ajaxPost("/:owner/:repository/issues/priorities/reorder")(writableUsersOnly { (repository) => ajaxPost("/:owner/:repository/issues/priorities/reorder")(writableUsersOnly { (repository) =>
reorderPriorities(repository.owner, repository.name, params("order") reorderPriorities(
repository.owner,
repository.name,
params("order")
.split(",") .split(",")
.map(id => id.toInt) .map(id => id.toInt)
.zipWithIndex .zipWithIndex
.toMap) .toMap
)
Ok() Ok()
}) })
@@ -87,24 +119,34 @@ trait PrioritiesControllerBase extends ControllerBase {
/** /**
* Constraint for the identifier such as user name, repository name or page name. * Constraint for the identifier such as user name, repository name or page name.
*/ */
private def priorityName: Constraint = new Constraint(){ private def priorityName: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(value.contains(',')){ if (value.contains(',')) {
Some(s"${name} contains invalid character.") Some(s"${name} contains invalid character.")
} else if(value.startsWith("_") || value.startsWith("-")){ } else if (value.startsWith("_") || value.startsWith("-")) {
Some(s"${name} starts with invalid character.") Some(s"${name} starts with invalid character.")
} else { } else {
None None
} }
} }
private def uniquePriorityName: Constraint = new Constraint(){ private def uniquePriorityName: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = { override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
val owner = params.value("owner") val owner = params.value("owner")
val repository = params.value("repository") val repository = params.value("repository")
params.optionValue("priorityId").map { priorityId => params
getPriority(owner, repository, value).filter(_.priorityId != priorityId.toInt).map(_ => "Name has already been taken.") .optionValue("priorityId")
}.getOrElse { .map { priorityId =>
getPriority(owner, repository, value)
.filter(_.priorityId != priorityId.toInt)
.map(_ => "Name has already been taken.")
}
.getOrElse {
getPriority(owner, repository, value).map(_ => "Name has already been taken.") getPriority(owner, repository, value).map(_ => "Name has already been taken.")
} }
} }

View File

@@ -1,6 +1,6 @@
package gitbucket.core.controller package gitbucket.core.controller
import gitbucket.core.model.WebHook import gitbucket.core.model.{CommitComment, CommitComments, IssueComment, WebHook}
import gitbucket.core.plugin.PluginRegistry import gitbucket.core.plugin.PluginRegistry
import gitbucket.core.pulls.html import gitbucket.core.pulls.html
import gitbucket.core.service.CommitStatusService import gitbucket.core.service.CommitStatusService
@@ -17,25 +17,49 @@ import org.scalatra.forms._
import org.eclipse.jgit.api.Git import org.eclipse.jgit.api.Git
import org.eclipse.jgit.lib.PersonIdent import org.eclipse.jgit.lib.PersonIdent
import org.eclipse.jgit.revwalk.RevWalk import org.eclipse.jgit.revwalk.RevWalk
import org.scalatra.BadRequest
import scala.collection.JavaConverters._ import scala.collection.JavaConverters._
class PullRequestsController
class PullRequestsController extends PullRequestsControllerBase extends PullRequestsControllerBase
with RepositoryService with AccountService with IssuesService with PullRequestService with MilestonesService with LabelsService with RepositoryService
with CommitsService with ActivityService with WebHookPullRequestService with AccountService
with ReadableUsersAuthenticator with ReferrerAuthenticator with WritableUsersAuthenticator with IssuesService
with CommitStatusService with MergeService with ProtectedBranchService with PrioritiesService with PullRequestService
with MilestonesService
with LabelsService
with CommitsService
with ActivityService
with WebHookPullRequestService
with ReadableUsersAuthenticator
with ReferrerAuthenticator
with WritableUsersAuthenticator
with CommitStatusService
with MergeService
with ProtectedBranchService
with PrioritiesService
trait PullRequestsControllerBase extends ControllerBase { trait PullRequestsControllerBase extends ControllerBase {
self: RepositoryService with AccountService with IssuesService with MilestonesService with LabelsService self: RepositoryService
with CommitsService with ActivityService with PullRequestService with WebHookPullRequestService with AccountService
with ReadableUsersAuthenticator with ReferrerAuthenticator with WritableUsersAuthenticator with IssuesService
with CommitStatusService with MergeService with ProtectedBranchService with PrioritiesService => with MilestonesService
with LabelsService
with CommitsService
with ActivityService
with PullRequestService
with WebHookPullRequestService
with ReadableUsersAuthenticator
with ReferrerAuthenticator
with WritableUsersAuthenticator
with CommitStatusService
with MergeService
with ProtectedBranchService
with PrioritiesService =>
val pullRequestForm = mapping( val pullRequestForm = mapping(
"title" -> trim(label("Title" , text(required, maxlength(100)))), "title" -> trim(label("Title", text(required, maxlength(100)))),
"content" -> trim(label("Content", optional(text()))), "content" -> trim(label("Content", optional(text()))),
"targetUserName" -> trim(text(required, maxlength(100))), "targetUserName" -> trim(text(required, maxlength(100))),
"targetBranch" -> trim(text(required, maxlength(100))), "targetBranch" -> trim(text(required, maxlength(100))),
@@ -75,7 +99,7 @@ trait PullRequestsControllerBase extends ControllerBase {
get("/:owner/:repository/pulls")(referrersOnly { repository => get("/:owner/:repository/pulls")(referrersOnly { repository =>
val q = request.getParameter("q") val q = request.getParameter("q")
if(Option(q).exists(_.contains("is:issue"))){ if (Option(q).exists(_.contains("is:issue"))) {
redirect(s"/${repository.owner}/${repository.name}/issues?q=" + StringUtil.urlEncode(q)) redirect(s"/${repository.owner}/${repository.name}/issues?q=" + StringUtil.urlEncode(q))
} else { } else {
searchPullRequests(None, repository) searchPullRequests(None, repository)
@@ -83,41 +107,110 @@ trait PullRequestsControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/pull/:id")(referrersOnly { repository => get("/:owner/:repository/pull/:id")(referrersOnly { repository =>
params("id").toIntOpt.flatMap{ issueId => params("id").toIntOpt.flatMap {
issueId =>
val owner = repository.owner val owner = repository.owner
val name = repository.name val name = repository.name
getPullRequest(owner, name, issueId) map { case(issue, pullreq) => getPullRequest(owner, name, issueId) map {
using(Git.open(getRepositoryDir(owner, name))){ git => case (issue, pullreq) =>
val (commits, diffs) = val (commits, diffs) =
getRequestCompareInfo(owner, name, pullreq.commitIdFrom, owner, name, pullreq.commitIdTo) getRequestCompareInfo(owner, name, pullreq.commitIdFrom, owner, name, pullreq.commitIdTo)
html.pullreq(
issue, pullreq, html.conversation(
(commits.flatten.map(commit => getCommitComments(owner, name, commit.id, true)).flatten.toList ::: getComments(owner, name, issueId)) issue,
.sortWith((a, b) => a.registeredDate before b.registeredDate), pullreq,
commits.flatten,
getPullRequestComments(owner, name, issue.issueId, commits.flatten),
diffs.size,
getIssueLabels(owner, name, issueId), getIssueLabels(owner, name, issueId),
getAssignableUserNames(owner, name), getAssignableUserNames(owner, name),
getMilestonesWithIssueCount(owner, name), getMilestonesWithIssueCount(owner, name),
getPriorities(owner, name), getPriorities(owner, name),
getLabels(owner, name), getLabels(owner, name),
commits,
diffs,
isEditable(repository), isEditable(repository),
isManageable(repository), isManageable(repository),
hasDeveloperRole(pullreq.requestUserName, pullreq.requestRepositoryName, context.loginAccount), hasDeveloperRole(pullreq.requestUserName, pullreq.requestRepositoryName, context.loginAccount),
repository, repository,
getRepository(pullreq.requestUserName, pullreq.requestRepositoryName), getRepository(pullreq.requestUserName, pullreq.requestRepositoryName),
flash.toMap.map(f => f._1 -> f._2.toString)) flash.toMap.map(f => f._1 -> f._2.toString)
)
// html.pullreq(
// issue,
// pullreq,
// comments,
// getIssueLabels(owner, name, issueId),
// getAssignableUserNames(owner, name),
// getMilestonesWithIssueCount(owner, name),
// getPriorities(owner, name),
// getLabels(owner, name),
// commits,
// diffs,
// isEditable(repository),
// isManageable(repository),
// hasDeveloperRole(pullreq.requestUserName, pullreq.requestRepositoryName, context.loginAccount),
// repository,
// getRepository(pullreq.requestUserName, pullreq.requestRepositoryName),
// flash.toMap.map(f => f._1 -> f._2.toString)
// )
} }
} getOrElse NotFound()
})
get("/:owner/:repository/pull/:id/commits")(referrersOnly { repository =>
params("id").toIntOpt.flatMap {
issueId =>
val owner = repository.owner
val name = repository.name
getPullRequest(owner, name, issueId) map {
case (issue, pullreq) =>
val (commits, diffs) =
getRequestCompareInfo(owner, name, pullreq.commitIdFrom, owner, name, pullreq.commitIdTo)
html.commits(
issue,
pullreq,
commits,
getPullRequestComments(owner, name, issue.issueId, commits.flatten),
diffs.size,
isManageable(repository),
repository
)
}
} getOrElse NotFound()
})
get("/:owner/:repository/pull/:id/files")(referrersOnly { repository =>
params("id").toIntOpt.flatMap {
issueId =>
val owner = repository.owner
val name = repository.name
getPullRequest(owner, name, issueId) map {
case (issue, pullreq) =>
val (commits, diffs) =
getRequestCompareInfo(owner, name, pullreq.commitIdFrom, owner, name, pullreq.commitIdTo)
html.files(
issue,
pullreq,
diffs,
commits.flatten,
getPullRequestComments(owner, name, issue.issueId, commits.flatten),
isManageable(repository),
repository
)
} }
} getOrElse NotFound() } getOrElse NotFound()
}) })
ajaxGet("/:owner/:repository/pull/:id/mergeguide")(referrersOnly { repository => ajaxGet("/:owner/:repository/pull/:id/mergeguide")(referrersOnly { repository =>
params("id").toIntOpt.flatMap{ issueId => params("id").toIntOpt.flatMap {
issueId =>
val owner = repository.owner val owner = repository.owner
val name = repository.name val name = repository.name
getPullRequest(owner, name, issueId) map { case(issue, pullreq) => getPullRequest(owner, name, issueId) map {
val conflictMessage = LockUtil.lock(s"${owner}/${name}"){ case (issue, pullreq) =>
val conflictMessage = LockUtil.lock(s"${owner}/${name}") {
checkConflict(owner, name, pullreq.branch, issueId) checkConflict(owner, name, pullreq.branch, issueId)
} }
val hasMergePermission = hasDeveloperRole(owner, name, context.loginAccount) val hasMergePermission = hasDeveloperRole(owner, name, context.loginAccount)
@@ -127,21 +220,35 @@ trait PullRequestsControllerBase extends ControllerBase {
commitStatues = getCommitStatues(owner, name, pullreq.commitIdTo), commitStatues = getCommitStatues(owner, name, pullreq.commitIdTo),
branchProtection = branchProtection, branchProtection = branchProtection,
branchIsOutOfDate = JGitUtil.getShaByRef(owner, name, pullreq.branch) != Some(pullreq.commitIdFrom), branchIsOutOfDate = JGitUtil.getShaByRef(owner, name, pullreq.branch) != Some(pullreq.commitIdFrom),
needStatusCheck = context.loginAccount.map{ u => needStatusCheck = context.loginAccount
.map { u =>
branchProtection.needStatusCheck(u.userName) branchProtection.needStatusCheck(u.userName)
}.getOrElse(true), }
hasUpdatePermission = hasDeveloperRole(pullreq.requestUserName, pullreq.requestRepositoryName, context.loginAccount) && .getOrElse(true),
context.loginAccount.map{ u => hasUpdatePermission = hasDeveloperRole(
!getProtectedBranchInfo(pullreq.requestUserName, pullreq.requestRepositoryName, pullreq.requestBranch).needStatusCheck(u.userName) pullreq.requestUserName,
}.getOrElse(false), pullreq.requestRepositoryName,
context.loginAccount
) &&
context.loginAccount
.map { u =>
!getProtectedBranchInfo(
pullreq.requestUserName,
pullreq.requestRepositoryName,
pullreq.requestBranch
).needStatusCheck(u.userName)
}
.getOrElse(false),
hasMergePermission = hasMergePermission, hasMergePermission = hasMergePermission,
commitIdTo = pullreq.commitIdTo) commitIdTo = pullreq.commitIdTo
)
html.mergeguide( html.mergeguide(
mergeStatus, mergeStatus,
issue, issue,
pullreq, pullreq,
repository, repository,
getRepository(pullreq.requestUserName, pullreq.requestRepositoryName).get) getRepository(pullreq.requestUserName, pullreq.requestRepositoryName).get
)
} }
} getOrElse NotFound() } getOrElse NotFound()
}) })
@@ -157,16 +264,23 @@ trait PullRequestsControllerBase extends ControllerBase {
} yield { } yield {
val repository = getRepository(owner, name).get val repository = getRepository(owner, name).get
val branchProtection = getProtectedBranchInfo(owner, name, pullreq.requestBranch) val branchProtection = getProtectedBranchInfo(owner, name, pullreq.requestBranch)
if(branchProtection.enabled){ if (branchProtection.enabled) {
flash += "error" -> s"branch ${pullreq.requestBranch} is protected." flash += "error" -> s"branch ${pullreq.requestBranch} is protected."
} else { } else {
if(repository.repository.defaultBranch != pullreq.requestBranch){ if (repository.repository.defaultBranch != pullreq.requestBranch) {
val userName = context.loginAccount.get.userName val userName = context.loginAccount.get.userName
using(Git.open(getRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getRepositoryDir(repository.owner, repository.name))) { git =>
git.branchDelete().setForce(true).setBranchNames(pullreq.requestBranch).call() git.branchDelete().setForce(true).setBranchNames(pullreq.requestBranch).call()
recordDeleteBranchActivity(repository.owner, repository.name, userName, pullreq.requestBranch) recordDeleteBranchActivity(repository.owner, repository.name, userName, pullreq.requestBranch)
} }
createComment(baseRepository.owner, baseRepository.name, userName, issueId, pullreq.requestBranch, "delete_branch") createComment(
baseRepository.owner,
baseRepository.name,
userName,
issueId,
pullreq.requestBranch,
"delete_branch"
)
} else { } else {
flash += "error" -> s"""Can't delete the default branch "${pullreq.requestBranch}".""" flash += "error" -> s"""Can't delete the default branch "${pullreq.requestBranch}"."""
} }
@@ -186,31 +300,49 @@ trait PullRequestsControllerBase extends ControllerBase {
} yield { } yield {
val repository = getRepository(owner, name).get val repository = getRepository(owner, name).get
val branchProtection = getProtectedBranchInfo(owner, name, pullreq.requestBranch) val branchProtection = getProtectedBranchInfo(owner, name, pullreq.requestBranch)
if(branchProtection.needStatusCheck(loginAccount.userName)){ if (branchProtection.needStatusCheck(loginAccount.userName)) {
flash += "error" -> s"branch ${pullreq.requestBranch} is protected need status check." flash += "error" -> s"branch ${pullreq.requestBranch} is protected need status check."
} else { } else {
LockUtil.lock(s"${owner}/${name}"){ LockUtil.lock(s"${owner}/${name}") {
val alias = if(pullreq.repositoryName == pullreq.requestRepositoryName && pullreq.userName == pullreq.requestUserName){ val alias =
if (pullreq.repositoryName == pullreq.requestRepositoryName && pullreq.userName == pullreq.requestUserName) {
pullreq.branch pullreq.branch
} else { } else {
s"${pullreq.userName}:${pullreq.branch}" s"${pullreq.userName}:${pullreq.branch}"
} }
val existIds = using(Git.open(Directory.getRepositoryDir(owner, name))) { git => JGitUtil.getAllCommitIds(git) }.toSet val existIds = using(Git.open(Directory.getRepositoryDir(owner, name))) { git =>
pullRemote(owner, name, pullreq.requestBranch, pullreq.userName, pullreq.repositoryName, pullreq.branch, loginAccount, JGitUtil.getAllCommitIds(git)
s"Merge branch '${alias}' into ${pullreq.requestBranch}") match { }.toSet
pullRemote(
owner,
name,
pullreq.requestBranch,
pullreq.userName,
pullreq.repositoryName,
pullreq.branch,
loginAccount,
s"Merge branch '${alias}' into ${pullreq.requestBranch}"
) match {
case None => // conflict case None => // conflict
flash += "error" -> s"Can't automatic merging branch '${alias}' into ${pullreq.requestBranch}." flash += "error" -> s"Can't automatic merging branch '${alias}' into ${pullreq.requestBranch}."
case Some(oldId) => case Some(oldId) =>
// update pull request // update pull request
updatePullRequests(owner, name, pullreq.requestBranch) updatePullRequests(owner, name, pullreq.requestBranch)
using(Git.open(Directory.getRepositoryDir(owner, name))) { git => using(Git.open(Directory.getRepositoryDir(owner, name))) {
git =>
// after update branch // after update branch
val newCommitId = git.getRepository.resolve(s"refs/heads/${pullreq.requestBranch}") val newCommitId = git.getRepository.resolve(s"refs/heads/${pullreq.requestBranch}")
val commits = git.log.addRange(oldId, newCommitId).call.iterator.asScala.map(c => new JGitUtil.CommitInfo(c)).toList val commits = git.log
.addRange(oldId, newCommitId)
.call
.iterator
.asScala
.map(c => new JGitUtil.CommitInfo(c))
.toList
commits.foreach { commit => commits.foreach { commit =>
if(!existIds.contains(commit.id)){ if (!existIds.contains(commit.id)) {
createIssueComment(owner, name, commit) createIssueComment(owner, name, commit)
} }
} }
@@ -219,19 +351,43 @@ trait PullRequestsControllerBase extends ControllerBase {
recordPushActivity(owner, name, loginAccount.userName, pullreq.branch, commits) recordPushActivity(owner, name, loginAccount.userName, pullreq.branch, commits)
// close issue by commit message // close issue by commit message
if(pullreq.requestBranch == repository.repository.defaultBranch){ if (pullreq.requestBranch == repository.repository.defaultBranch) {
commits.map { commit => commits.foreach { commit =>
closeIssuesFromMessage(commit.fullMessage, loginAccount.userName, owner, name) closeIssuesFromMessage(commit.fullMessage, loginAccount.userName, owner, name).foreach {
issueId =>
getIssue(repository.owner, repository.name, issueId.toString).foreach { issue =>
callIssuesWebHook("closed", repository, issue, baseUrl, loginAccount)
PluginRegistry().getIssueHooks
.foreach(
_.closedByCommitComment(issue, repository, commit.fullMessage, loginAccount)
)
}
}
} }
} }
// call web hook // call web hook
callPullRequestWebHookByRequestBranch("synchronize", repository, pullreq.requestBranch, baseUrl, loginAccount) callPullRequestWebHookByRequestBranch(
"synchronize",
repository,
pullreq.requestBranch,
baseUrl,
loginAccount
)
callWebHookOf(owner, name, WebHook.Push) { callWebHookOf(owner, name, WebHook.Push) {
for { for {
ownerAccount <- getAccountByUserName(owner) ownerAccount <- getAccountByUserName(owner)
} yield { } yield {
WebHookService.WebHookPushPayload(git, loginAccount, pullreq.requestBranch, repository, commits, ownerAccount, oldId = oldId, newId = newCommitId) WebHookService.WebHookPushPayload(
git,
loginAccount,
pullreq.requestBranch,
repository,
commits,
ownerAccount,
oldId = oldId,
newId = newCommitId
)
} }
} }
} }
@@ -245,12 +401,16 @@ trait PullRequestsControllerBase extends ControllerBase {
}) })
post("/:owner/:repository/pull/:id/merge", mergeForm)(writableUsersOnly { (form, repository) => post("/:owner/:repository/pull/:id/merge", mergeForm)(writableUsersOnly { (form, repository) =>
params("id").toIntOpt.flatMap { issueId => params("id").toIntOpt.flatMap {
issueId =>
val owner = repository.owner val owner = repository.owner
val name = repository.name val name = repository.name
LockUtil.lock(s"${owner}/${name}"){ if (repository.repository.options.mergeOptions.split(",").contains(form.strategy)) {
getPullRequest(owner, name, issueId).map { case (issue, pullreq) => LockUtil.lock(s"${owner}/${name}") {
using(Git.open(getRepositoryDir(owner, name))) { git => getPullRequest(owner, name, issueId).map {
case (issue, pullreq) =>
using(Git.open(getRepositoryDir(owner, name))) {
git =>
// mark issue as merged and close. // mark issue as merged and close.
val loginAccount = context.loginAccount.get val loginAccount = context.loginAccount.get
val commentId = createComment(owner, name, loginAccount.userName, issueId, form.message, "merge") val commentId = createComment(owner, name, loginAccount.userName, issueId, form.message, "merge")
@@ -260,10 +420,16 @@ trait PullRequestsControllerBase extends ControllerBase {
// record activity // record activity
recordMergeActivity(owner, name, loginAccount.userName, issueId, form.message) recordMergeActivity(owner, name, loginAccount.userName, issueId, form.message)
val (commits, _) = getRequestCompareInfo(owner, name, pullreq.commitIdFrom, val (commits, _) = getRequestCompareInfo(
pullreq.requestUserName, pullreq.requestRepositoryName, pullreq.commitIdTo) owner,
name,
pullreq.commitIdFrom,
pullreq.requestUserName,
pullreq.requestRepositoryName,
pullreq.commitIdTo
)
val revCommits = using(new RevWalk( git.getRepository )){ revWalk => val revCommits = using(new RevWalk(git.getRepository)) { revWalk =>
commits.flatten.map { commit => commits.flatten.map { commit =>
revWalk.parseCommit(git.getRepository.resolve(commit.id)) revWalk.parseCommit(git.getRepository.resolve(commit.id))
} }
@@ -272,26 +438,64 @@ trait PullRequestsControllerBase extends ControllerBase {
// merge git repository // merge git repository
form.strategy match { form.strategy match {
case "merge-commit" => case "merge-commit" =>
mergePullRequest(git, pullreq.branch, issueId, mergePullRequest(
git,
pullreq.branch,
issueId,
s"Merge pull request #${issueId} from ${pullreq.requestUserName}/${pullreq.requestBranch}\n\n" + form.message, s"Merge pull request #${issueId} from ${pullreq.requestUserName}/${pullreq.requestBranch}\n\n" + form.message,
new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)) new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)
)
case "rebase" => case "rebase" =>
rebasePullRequest(git, pullreq.branch, issueId, revCommits, rebasePullRequest(
new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)) git,
pullreq.branch,
issueId,
revCommits,
new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)
)
case "squash" => case "squash" =>
squashPullRequest(git, pullreq.branch, issueId, squashPullRequest(
git,
pullreq.branch,
issueId,
s"${issue.title} (#${issueId})\n\n" + form.message, s"${issue.title} (#${issueId})\n\n" + form.message,
new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)) new PersonIdent(loginAccount.fullName, loginAccount.mailAddress)
)
} }
// close issue by content of pull request // close issue by content of pull request
val defaultBranch = getRepository(owner, name).get.repository.defaultBranch val defaultBranch = getRepository(owner, name).get.repository.defaultBranch
if(pullreq.branch == defaultBranch){ if (pullreq.branch == defaultBranch) {
commits.flatten.foreach { commit => commits.flatten.foreach { commit =>
closeIssuesFromMessage(commit.fullMessage, loginAccount.userName, owner, name) closeIssuesFromMessage(commit.fullMessage, loginAccount.userName, owner, name).foreach {
issueId =>
getIssue(owner, name, issueId.toString).foreach { issue =>
callIssuesWebHook("closed", repository, issue, baseUrl, loginAccount)
PluginRegistry().getIssueHooks
.foreach(_.closedByCommitComment(issue, repository, commit.fullMessage, loginAccount))
}
}
}
val issueContent = issue.title + " " + issue.content.getOrElse("")
closeIssuesFromMessage(
issueContent,
loginAccount.userName,
owner,
name
).foreach { issueId =>
getIssue(owner, name, issueId.toString).foreach { issue =>
callIssuesWebHook("closed", repository, issue, baseUrl, loginAccount)
PluginRegistry().getIssueHooks
.foreach(_.closedByCommitComment(issue, repository, issueContent, loginAccount))
}
}
closeIssuesFromMessage(form.message, loginAccount.userName, owner, name).foreach { issueId =>
getIssue(owner, name, issueId.toString).foreach { issue =>
callIssuesWebHook("closed", repository, issue, baseUrl, loginAccount)
PluginRegistry().getIssueHooks
.foreach(_.closedByCommitComment(issue, repository, issueContent, loginAccount))
}
} }
closeIssuesFromMessage(issue.title + " " + issue.content.getOrElse(""), loginAccount.userName, owner, name)
closeIssuesFromMessage(form.message, loginAccount.userName, owner, name)
} }
updatePullRequests(owner, name, pullreq.branch) updatePullRequests(owner, name, pullreq.branch)
@@ -300,7 +504,7 @@ trait PullRequestsControllerBase extends ControllerBase {
callPullRequestWebHook("closed", repository, issueId, context.baseUrl, context.loginAccount.get) callPullRequestWebHook("closed", repository, issueId, context.baseUrl, context.loginAccount.get)
// call hooks // call hooks
PluginRegistry().getPullRequestHooks.foreach{ h => PluginRegistry().getPullRequestHooks.foreach { h =>
h.addedComment(commentId, form.message, issue, repository) h.addedComment(commentId, form.message, issue, repository)
h.merged(issue, repository) h.merged(issue, repository)
} }
@@ -309,29 +513,38 @@ trait PullRequestsControllerBase extends ControllerBase {
} }
} }
} }
} else Some(BadRequest())
} getOrElse NotFound() } getOrElse NotFound()
}) })
get("/:owner/:repository/compare")(referrersOnly { forkedRepository => get("/:owner/:repository/compare")(referrersOnly { forkedRepository =>
val headBranch:Option[String] = params.get("head") val headBranch: Option[String] = params.get("head")
(forkedRepository.repository.originUserName, forkedRepository.repository.originRepositoryName) match { (forkedRepository.repository.originUserName, forkedRepository.repository.originRepositoryName) match {
case (Some(originUserName), Some(originRepositoryName)) => { case (Some(originUserName), Some(originRepositoryName)) => {
getRepository(originUserName, originRepositoryName).map { originRepository => getRepository(originUserName, originRepositoryName).map {
originRepository =>
using( using(
Git.open(getRepositoryDir(originUserName, originRepositoryName)), Git.open(getRepositoryDir(originUserName, originRepositoryName)),
Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name)) Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name))
){ (oldGit, newGit) => ) { (oldGit, newGit) =>
val newBranch = headBranch.getOrElse(JGitUtil.getDefaultBranch(newGit, forkedRepository).get._2) val newBranch = headBranch.getOrElse(JGitUtil.getDefaultBranch(newGit, forkedRepository).get._2)
val oldBranch = originRepository.branchList.find( _ == newBranch).getOrElse(JGitUtil.getDefaultBranch(oldGit, originRepository).get._2) val oldBranch = originRepository.branchList
.find(_ == newBranch)
.getOrElse(JGitUtil.getDefaultBranch(oldGit, originRepository).get._2)
redirect(s"/${forkedRepository.owner}/${forkedRepository.name}/compare/${originUserName}:${oldBranch}...${newBranch}") redirect(
s"/${forkedRepository.owner}/${forkedRepository.name}/compare/${originUserName}:${oldBranch}...${newBranch}"
)
} }
} getOrElse NotFound() } getOrElse NotFound()
} }
case _ => { case _ => {
using(Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name))){ git => using(Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name))) { git =>
JGitUtil.getDefaultBranch(git, forkedRepository).map { case (_, defaultBranch) => JGitUtil.getDefaultBranch(git, forkedRepository).map {
redirect(s"/${forkedRepository.owner}/${forkedRepository.name}/compare/${defaultBranch}...${headBranch.getOrElse(defaultBranch)}") case (_, defaultBranch) =>
redirect(
s"/${forkedRepository.owner}/${forkedRepository.name}/compare/${defaultBranch}...${headBranch.getOrElse(defaultBranch)}"
)
} getOrElse { } getOrElse {
redirect(s"/${forkedRepository.owner}/${forkedRepository.name}") redirect(s"/${forkedRepository.owner}/${forkedRepository.name}")
} }
@@ -345,49 +558,71 @@ trait PullRequestsControllerBase extends ControllerBase {
val (originOwner, originId) = parseCompareIdentifier(origin, forkedRepository.owner) val (originOwner, originId) = parseCompareIdentifier(origin, forkedRepository.owner)
val (forkedOwner, forkedId) = parseCompareIdentifier(forked, forkedRepository.owner) val (forkedOwner, forkedId) = parseCompareIdentifier(forked, forkedRepository.owner)
(for( (for (originRepositoryName <- if (originOwner == forkedOwner) {
originRepositoryName <- if(originOwner == forkedOwner) {
// Self repository // Self repository
Some(forkedRepository.name) Some(forkedRepository.name)
} else if(forkedRepository.repository.originUserName.isEmpty){ } else if (forkedRepository.repository.originUserName.isEmpty) {
// when ForkedRepository is the original repository // when ForkedRepository is the original repository
getForkedRepositories(forkedRepository.owner, forkedRepository.name).find(_.userName == originOwner).map(_.repositoryName) getForkedRepositories(forkedRepository.owner, forkedRepository.name)
} else if(Some(originOwner) == forkedRepository.repository.originUserName){ .find(_.userName == originOwner)
.map(_.repositoryName)
} else if (Some(originOwner) == forkedRepository.repository.originUserName) {
// Original repository // Original repository
forkedRepository.repository.originRepositoryName forkedRepository.repository.originRepositoryName
} else { } else {
// Sibling repository // Sibling repository
getUserRepositories(originOwner).find { x => getUserRepositories(originOwner)
.find { x =>
x.repository.originUserName == forkedRepository.repository.originUserName && x.repository.originUserName == forkedRepository.repository.originUserName &&
x.repository.originRepositoryName == forkedRepository.repository.originRepositoryName x.repository.originRepositoryName == forkedRepository.repository.originRepositoryName
}.map(_.repository.repositoryName) }
.map(_.repository.repositoryName)
}; };
originRepository <- getRepository(originOwner, originRepositoryName) originRepository <- getRepository(originOwner, originRepositoryName)) yield {
) yield {
using( using(
Git.open(getRepositoryDir(originRepository.owner, originRepository.name)), Git.open(getRepositoryDir(originRepository.owner, originRepository.name)),
Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name)) Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name))
){ case (oldGit, newGit) => ) {
case (oldGit, newGit) =>
val (oldId, newId) = val (oldId, newId) =
if(originRepository.branchList.contains(originId) && forkedRepository.branchList.contains(forkedId)){ if (originRepository.branchList.contains(originId)) {
// Branch name val forkedId2 =
val rootId = JGitUtil.getForkedCommitId(oldGit, newGit, forkedRepository.tags.collectFirst { case x if x.name == forkedId => x.id }.getOrElse(forkedId)
originRepository.owner, originRepository.name, originId,
forkedRepository.owner, forkedRepository.name, forkedId) val originId2 = JGitUtil.getForkedCommitId(
oldGit,
newGit,
originRepository.owner,
originRepository.name,
originId,
forkedRepository.owner,
forkedRepository.name,
forkedId2
)
(Option(oldGit.getRepository.resolve(originId2)), Option(newGit.getRepository.resolve(forkedId2)))
(Option(oldGit.getRepository.resolve(rootId)), Option(newGit.getRepository.resolve(forkedId)))
} else { } else {
// Commit id val originId2 =
(Option(oldGit.getRepository.resolve(originId)), Option(newGit.getRepository.resolve(forkedId))) originRepository.tags.collectFirst { case x if x.name == originId => x.id }.getOrElse(originId)
val forkedId2 =
forkedRepository.tags.collectFirst { case x if x.name == forkedId => x.id }.getOrElse(forkedId)
(Option(oldGit.getRepository.resolve(originId2)), Option(newGit.getRepository.resolve(forkedId2)))
} }
(oldId, newId) match { (oldId, newId) match {
case (Some(oldId), Some(newId)) => { case (Some(oldId), Some(newId)) => {
val (commits, diffs) = getRequestCompareInfo( val (commits, diffs) = getRequestCompareInfo(
originRepository.owner, originRepository.name, oldId.getName, originRepository.owner,
forkedRepository.owner, forkedRepository.name, newId.getName) originRepository.name,
oldId.getName,
forkedRepository.owner,
forkedRepository.name,
newId.getName
)
val title = if(commits.flatten.length == 1){ val title = if (commits.flatten.length == 1) {
commits.flatten.head.shortMessage commits.flatten.head.shortMessage
} else { } else {
val text = forkedId.replaceAll("[\\-_]", " ") val text = forkedId.replaceAll("[\\-_]", " ")
@@ -399,13 +634,20 @@ trait PullRequestsControllerBase extends ControllerBase {
commits, commits,
diffs, diffs,
((forkedRepository.repository.originUserName, forkedRepository.repository.originRepositoryName) match { ((forkedRepository.repository.originUserName, forkedRepository.repository.originRepositoryName) match {
case (Some(userName), Some(repositoryName)) => getRepository(userName, repositoryName) match { case (Some(userName), Some(repositoryName)) =>
getRepository(userName, repositoryName) match {
case Some(x) => x.repository :: getForkedRepositories(userName, repositoryName) case Some(x) => x.repository :: getForkedRepositories(userName, repositoryName)
case None => getForkedRepositories(userName, repositoryName) case None => getForkedRepositories(userName, repositoryName)
} }
case _ => forkedRepository.repository :: getForkedRepositories(forkedRepository.owner, forkedRepository.name) case _ =>
}).map { repository => (repository.userName, repository.repositoryName) }, forkedRepository.repository :: getForkedRepositories(forkedRepository.owner, forkedRepository.name)
commits.flatten.map(commit => getCommitComments(forkedRepository.owner, forkedRepository.name, commit.id, false)).flatten.toList, }).map { repository =>
(repository.userName, repository.repositoryName)
},
commits.flatten
.map(commit => getCommitComments(forkedRepository.owner, forkedRepository.name, commit.id, false))
.flatten
.toList,
originId, originId,
forkedId, forkedId,
oldId.getName, oldId.getName,
@@ -422,9 +664,11 @@ trait PullRequestsControllerBase extends ControllerBase {
) )
} }
case (oldId, newId) => case (oldId, newId) =>
redirect(s"/${forkedRepository.owner}/${forkedRepository.name}/compare/" + redirect(
s"/${forkedRepository.owner}/${forkedRepository.name}/compare/" +
s"${originOwner}:${oldId.map(_ => originId).getOrElse(originRepository.repository.defaultBranch)}..." + s"${originOwner}:${oldId.map(_ => originId).getOrElse(originRepository.repository.defaultBranch)}..." +
s"${forkedOwner}:${newId.map(_ => forkedId).getOrElse(forkedRepository.repository.defaultBranch)}") s"${forkedOwner}:${newId.map(_ => forkedId).getOrElse(forkedRepository.repository.defaultBranch)}"
)
} }
} }
}) getOrElse NotFound() }) getOrElse NotFound()
@@ -435,25 +679,32 @@ trait PullRequestsControllerBase extends ControllerBase {
val (originOwner, tmpOriginBranch) = parseCompareIdentifier(origin, forkedRepository.owner) val (originOwner, tmpOriginBranch) = parseCompareIdentifier(origin, forkedRepository.owner)
val (forkedOwner, tmpForkedBranch) = parseCompareIdentifier(forked, forkedRepository.owner) val (forkedOwner, tmpForkedBranch) = parseCompareIdentifier(forked, forkedRepository.owner)
(for( (for (originRepositoryName <- if (originOwner == forkedOwner) {
originRepositoryName <- if(originOwner == forkedOwner){
Some(forkedRepository.name) Some(forkedRepository.name)
} else { } else {
forkedRepository.repository.originRepositoryName.orElse { forkedRepository.repository.originRepositoryName.orElse {
getForkedRepositories(forkedRepository.owner, forkedRepository.name).find(_.userName == originOwner).map(_.repositoryName) getForkedRepositories(forkedRepository.owner, forkedRepository.name)
.find(_.userName == originOwner)
.map(_.repositoryName)
} }
}; };
originRepository <- getRepository(originOwner, originRepositoryName) originRepository <- getRepository(originOwner, originRepositoryName)) yield {
) yield {
using( using(
Git.open(getRepositoryDir(originRepository.owner, originRepository.name)), Git.open(getRepositoryDir(originRepository.owner, originRepository.name)),
Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name)) Git.open(getRepositoryDir(forkedRepository.owner, forkedRepository.name))
){ case (oldGit, newGit) => ) {
case (oldGit, newGit) =>
val originBranch = JGitUtil.getDefaultBranch(oldGit, originRepository, tmpOriginBranch).get._2 val originBranch = JGitUtil.getDefaultBranch(oldGit, originRepository, tmpOriginBranch).get._2
val forkedBranch = JGitUtil.getDefaultBranch(newGit, forkedRepository, tmpForkedBranch).get._2 val forkedBranch = JGitUtil.getDefaultBranch(newGit, forkedRepository, tmpForkedBranch).get._2
val conflict = LockUtil.lock(s"${originRepository.owner}/${originRepository.name}"){ val conflict = LockUtil.lock(s"${originRepository.owner}/${originRepository.name}") {
checkConflict(originRepository.owner, originRepository.name, originBranch, checkConflict(
forkedRepository.owner, forkedRepository.name, forkedBranch) originRepository.owner,
originRepository.name,
originBranch,
forkedRepository.owner,
forkedRepository.name,
forkedBranch
)
} }
html.mergecheck(conflict.isDefined) html.mergecheck(conflict.isDefined)
} }
@@ -461,7 +712,8 @@ trait PullRequestsControllerBase extends ControllerBase {
}) })
post("/:owner/:repository/pulls/new", pullRequestForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/pulls/new", pullRequestForm)(readableUsersOnly { (form, repository) =>
defining(repository.owner, repository.name){ case (owner, name) => defining(repository.owner, repository.name) {
case (owner, name) =>
val manageable = isManageable(repository) val manageable = isManageable(repository)
val loginUserName = context.loginAccount.get.userName val loginUserName = context.loginAccount.get.userName
@@ -474,7 +726,8 @@ trait PullRequestsControllerBase extends ControllerBase {
assignedUserName = if (manageable) form.assignedUserName else None, assignedUserName = if (manageable) form.assignedUserName else None,
milestoneId = if (manageable) form.milestoneId else None, milestoneId = if (manageable) form.milestoneId else None,
priorityId = if (manageable) form.priorityId else None, priorityId = if (manageable) form.priorityId else None,
isPullRequest = true) isPullRequest = true
)
createPullRequest( createPullRequest(
originUserName = repository.owner, originUserName = repository.owner,
@@ -485,11 +738,12 @@ trait PullRequestsControllerBase extends ControllerBase {
requestRepositoryName = form.requestRepositoryName, requestRepositoryName = form.requestRepositoryName,
requestBranch = form.requestBranch, requestBranch = form.requestBranch,
commitIdFrom = form.commitIdFrom, commitIdFrom = form.commitIdFrom,
commitIdTo = form.commitIdTo) commitIdTo = form.commitIdTo
)
// insert labels // insert labels
if (manageable) { if (manageable) {
form.labelNames.map { value => form.labelNames.foreach { value =>
val labels = getLabels(owner, name) val labels = getLabels(owner, name)
value.split(",").foreach { labelName => value.split(",").foreach { labelName =>
labels.find(_.labelName == labelName).map { label => labels.find(_.labelName == labelName).map { label =>
@@ -510,7 +764,13 @@ trait PullRequestsControllerBase extends ControllerBase {
getIssue(owner, name, issueId.toString) foreach { issue => getIssue(owner, name, issueId.toString) foreach { issue =>
// extract references and create refer comment // extract references and create refer comment
createReferComment(owner, name, issue, form.title + " " + form.content.getOrElse(""), context.loginAccount.get) createReferComment(
owner,
name,
issue,
form.title + " " + form.content.getOrElse(""),
context.loginAccount.get
)
// call hooks // call hooks
PluginRegistry().getPullRequestHooks.foreach(_.created(issue, repository)) PluginRegistry().getPullRequestHooks.foreach(_.created(issue, repository))
@@ -521,14 +781,25 @@ trait PullRequestsControllerBase extends ControllerBase {
}) })
ajaxGet("/:owner/:repository/pulls/proposals")(readableUsersOnly { repository => ajaxGet("/:owner/:repository/pulls/proposals")(readableUsersOnly { repository =>
val branches = JGitUtil.getBranches( val thresholdTime = System.currentTimeMillis() - (1000 * 60 * 60)
val mailAddresses =
context.loginAccount.map(x => Seq(x.mailAddress) ++ getAccountExtraMailAddresses(x.userName)).getOrElse(Nil)
val branches = JGitUtil
.getBranches(
owner = repository.owner, owner = repository.owner,
name = repository.name, name = repository.name,
defaultBranch = repository.repository.defaultBranch, defaultBranch = repository.repository.defaultBranch,
origin = repository.repository.originUserName.isEmpty origin = repository.repository.originUserName.isEmpty
) )
.filter(x => x.mergeInfo.map(_.ahead).getOrElse(0) > 0 && x.mergeInfo.map(_.behind).getOrElse(0) == 0) .filter { x =>
.sortBy(br => (br.mergeInfo.isEmpty, br.commitTime)) x.mergeInfo.map(_.ahead).getOrElse(0) > 0 && x.mergeInfo.map(_.behind).getOrElse(0) == 0 &&
x.commitTime.getTime > thresholdTime &&
mailAddresses.contains(x.committerEmailAddress)
}
.sortBy { br =>
(br.mergeInfo.isEmpty, br.commitTime)
}
.map(_.name) .map(_.name)
.reverse .reverse
@@ -556,7 +827,7 @@ trait PullRequestsControllerBase extends ControllerBase {
* - "branch" to ("defaultOwner", "branch") * - "branch" to ("defaultOwner", "branch")
*/ */
private def parseCompareIdentifier(value: String, defaultOwner: String): (String, String) = private def parseCompareIdentifier(value: String, defaultOwner: String): (String, String) =
if(value.contains(':')){ if (value.contains(':')) {
val array = value.split(":") val array = value.split(":")
(array(0), array(1)) (array(0), array(1))
} else { } else {
@@ -564,7 +835,8 @@ trait PullRequestsControllerBase extends ControllerBase {
} }
private def searchPullRequests(userName: Option[String], repository: RepositoryService.RepositoryInfo) = private def searchPullRequests(userName: Option[String], repository: RepositoryService.RepositoryInfo) =
defining(repository.owner, repository.name){ case (owner, repoName) => defining(repository.owner, repository.name) {
case (owner, repoName) =>
val page = IssueSearchCondition.page(request) val page = IssueSearchCondition.page(request)
// retrieve search condition // retrieve search condition
@@ -578,12 +850,13 @@ trait PullRequestsControllerBase extends ControllerBase {
getMilestones(owner, repoName), getMilestones(owner, repoName),
getPriorities(owner, repoName), getPriorities(owner, repoName),
getLabels(owner, repoName), getLabels(owner, repoName),
countIssue(condition.copy(state = "open" ), true, owner -> repoName), countIssue(condition.copy(state = "open"), true, owner -> repoName),
countIssue(condition.copy(state = "closed"), true, owner -> repoName), countIssue(condition.copy(state = "closed"), true, owner -> repoName),
condition, condition,
repository, repository,
isEditable(repository), isEditable(repository),
isManageable(repository)) isManageable(repository)
)
} }
/** /**

View File

@@ -11,7 +11,8 @@ import gitbucket.core.releases.html
import org.apache.commons.io.FileUtils import org.apache.commons.io.FileUtils
import scala.collection.JavaConverters._ import scala.collection.JavaConverters._
class ReleaseController extends ReleaseControllerBase class ReleaseController
extends ReleaseControllerBase
with RepositoryService with RepositoryService
with AccountService with AccountService
with ReleaseService with ReleaseService
@@ -39,112 +40,152 @@ trait ReleaseControllerBase extends ControllerBase {
"content" -> trim(optional(text())) "content" -> trim(optional(text()))
)(ReleaseForm.apply) )(ReleaseForm.apply)
get("/:owner/:repository/releases")(referrersOnly {repository => get("/:owner/:repository/releases")(referrersOnly { repository =>
val releases = getReleases(repository.owner, repository.name) val releases = getReleases(repository.owner, repository.name)
val assets = getReleaseAssetsMap(repository.owner, repository.name) val assets = getReleaseAssetsMap(repository.owner, repository.name)
html.list( html.list(
repository, repository,
repository.tags.reverse.map { tag => repository.tags.reverse.map { tag =>
(tag, releases.find(_.tag == tag.name).map { release => (release, assets(release)) }) (tag, releases.find(_.tag == tag.name).map { release =>
(release, assets(release))
})
}, },
hasDeveloperRole(repository.owner, repository.name, context.loginAccount)) hasDeveloperRole(repository.owner, repository.name, context.loginAccount)
)
}) })
get("/:owner/:repository/releases/:tag")(referrersOnly { repository => get("/:owner/:repository/releases/:tag")(referrersOnly { repository =>
val tag = params("tag") val tagName = params("tag")
getRelease(repository.owner, repository.name, tag).map { release => getRelease(repository.owner, repository.name, tagName)
html.release(release, getReleaseAssets(repository.owner, repository.name, tag), hasDeveloperRole(repository.owner, repository.name, context.loginAccount), repository) .map { release =>
}.getOrElse(NotFound()) html.release(
release,
getReleaseAssets(repository.owner, repository.name, tagName),
hasDeveloperRole(repository.owner, repository.name, context.loginAccount),
repository
)
}
.getOrElse(NotFound())
}) })
get("/:owner/:repository/releases/:tag/assets/:fileId")(referrersOnly {repository => get("/:owner/:repository/releases/:tag/assets/:fileId")(referrersOnly { repository =>
val tag = params("tag") val tagName = params("tag")
val fileId = params("fileId") val fileId = params("fileId")
(for { (for {
_ <- repository.tags.find(_.name == tag) _ <- repository.tags.find(_.name == tagName)
_ <- getRelease(repository.owner, repository.name, tag) _ <- getRelease(repository.owner, repository.name, tagName)
asset <- getReleaseAsset(repository.owner, repository.name, tag, fileId) asset <- getReleaseAsset(repository.owner, repository.name, tagName, fileId)
} yield { } yield {
response.setHeader("Content-Disposition", s"attachment; filename=${asset.label}") response.setHeader("Content-Disposition", s"attachment; filename=${asset.label}")
RawData( RawData(
FileUtil.getMimeType(asset.label), FileUtil.getSafeMimeType(asset.label),
new File(getReleaseFilesDir(repository.owner, repository.name), tag + "/" + fileId) new File(getReleaseFilesDir(repository.owner, repository.name), FileUtil.checkFilename(tagName + "/" + fileId))
) )
}).getOrElse(NotFound()) }).getOrElse(NotFound())
}) })
get("/:owner/:repository/releases/:tag/create")(writableUsersOnly {repository => get("/:owner/:repository/releases/:tag/create")(writableUsersOnly { repository =>
html.form(repository, params("tag"), None) val tagName = params("tag")
repository.tags
.find(_.name == tagName)
.map { tag =>
html.form(repository, tag, None)
}
.getOrElse(NotFound())
}) })
post("/:owner/:repository/releases/:tag/create", releaseForm)(writableUsersOnly { (form, repository) => post("/:owner/:repository/releases/:tag/create", releaseForm)(writableUsersOnly { (form, repository) =>
val tag = params("tag") val tagName = params("tag")
val loginAccount = context.loginAccount.get val loginAccount = context.loginAccount.get
// Insert into RELEASE // Insert into RELEASE
createRelease(repository.owner, repository.name, form.name, form.content, tag, loginAccount) createRelease(repository.owner, repository.name, form.name, form.content, tagName, loginAccount)
// Insert into RELEASE_ASSET // Insert into RELEASE_ASSET
request.getParameterNames.asScala.filter(_.startsWith("file:")).foreach { paramName => val files = params.collect {
val Array(_, fileId) = paramName.split(":") case (name, value) if name.startsWith("file:") =>
val fileName = params(paramName) val Array(_, fileId) = name.split(":")
val size = new java.io.File(getReleaseFilesDir(repository.owner, repository.name), tag + "/" + fileId).length (fileId, value)
}
createReleaseAsset(repository.owner, repository.name, tag, fileId, fileName, size, loginAccount) files.foreach {
case (fileId, fileName) =>
val size =
new File(
getReleaseFilesDir(repository.owner, repository.name),
FileUtil.checkFilename(tagName + "/" + fileId)
).length
createReleaseAsset(repository.owner, repository.name, tagName, fileId, fileName, size, loginAccount)
} }
recordReleaseActivity(repository.owner, repository.name, loginAccount.userName, form.name) recordReleaseActivity(repository.owner, repository.name, loginAccount.userName, form.name)
redirect(s"/${repository.owner}/${repository.name}/releases/${tag}") redirect(s"/${repository.owner}/${repository.name}/releases/${tagName}")
}) })
get("/:owner/:repository/releases/:tag/edit")(writableUsersOnly {repository => get("/:owner/:repository/releases/:tag/edit")(writableUsersOnly { repository =>
val tag = params("tag") val tagName = params("tag")
getRelease(repository.owner, repository.name, tag).map { release => (for {
html.form(repository, release.tag, Some(release, getReleaseAssets(repository.owner, repository.name, tag))) release <- getRelease(repository.owner, repository.name, tagName)
}.getOrElse(NotFound()) tag <- repository.tags.find(_.name == tagName)
} yield {
html.form(repository, tag, Some(release, getReleaseAssets(repository.owner, repository.name, tagName)))
}).getOrElse(NotFound())
}) })
post("/:owner/:repository/releases/:tag/edit", releaseForm)(writableUsersOnly { (form, repository) => post("/:owner/:repository/releases/:tag/edit", releaseForm)(writableUsersOnly {
val tag = params("tag") (form, repository) =>
val tagName = params("tag")
val loginAccount = context.loginAccount.get val loginAccount = context.loginAccount.get
getRelease(repository.owner, repository.name, tag).map { release => getRelease(repository.owner, repository.name, tagName)
.map { release =>
// Update RELEASE // Update RELEASE
updateRelease(repository.owner, repository.name, tag, form.name, form.content) updateRelease(repository.owner, repository.name, tagName, form.name, form.content)
// Delete and Insert RELEASE_ASSET // Delete and Insert RELEASE_ASSET
val assets = getReleaseAssets(repository.owner, repository.name, tag) val assets = getReleaseAssets(repository.owner, repository.name, tagName)
deleteReleaseAssets(repository.owner, repository.name, tag) deleteReleaseAssets(repository.owner, repository.name, tagName)
val fileIds = request.getParameterNames.asScala.filter(_.startsWith("file:")).map { paramName => val files = params.collect {
val Array(_, fileId) = paramName.split(":") case (name, value) if name.startsWith("file:") =>
val fileName = params(paramName) val Array(_, fileId) = name.split(":")
val size = new java.io.File(getReleaseFilesDir(repository.owner, repository.name), release.tag + "/" + fileId).length (fileId, value)
}
createReleaseAsset(repository.owner, repository.name, tag, fileId, fileName, size, loginAccount) files.foreach {
fileId case (fileId, fileName) =>
val size =
new File(
getReleaseFilesDir(repository.owner, repository.name),
FileUtil.checkFilename(tagName + "/" + fileId)
).length
createReleaseAsset(repository.owner, repository.name, tagName, fileId, fileName, size, loginAccount)
} }
assets.foreach { asset => assets.foreach { asset =>
if(!fileIds.contains(asset.fileName)){ if (!files.exists { case (fileId, _) => fileId == asset.fileName }) {
val file = new java.io.File(getReleaseFilesDir(repository.owner, repository.name), release.tag + "/" + asset.fileName) val file = new File(
getReleaseFilesDir(repository.owner, repository.name),
FileUtil.checkFilename(release.tag + "/" + asset.fileName)
)
FileUtils.forceDelete(file) FileUtils.forceDelete(file)
} }
} }
redirect(s"/${release.userName}/${release.repositoryName}/releases/${tag}") redirect(s"/${release.userName}/${release.repositoryName}/releases/${tagName}")
}.getOrElse(NotFound()) }
.getOrElse(NotFound())
}) })
post("/:owner/:repository/releases/:tag/delete")(writableUsersOnly { repository => post("/:owner/:repository/releases/:tag/delete")(writableUsersOnly { repository =>
val tag = params("tag") val tagName = params("tag")
getRelease(repository.owner, repository.name, tag).foreach { release => getRelease(repository.owner, repository.name, tagName).foreach { release =>
FileUtils.deleteDirectory(new File(getReleaseFilesDir(repository.owner, repository.name), release.tag)) FileUtils.deleteDirectory(
new File(getReleaseFilesDir(repository.owner, repository.name), FileUtil.checkFilename(release.tag))
)
} }
deleteRelease(repository.owner, repository.name, tag) deleteRelease(repository.owner, repository.name, tagName)
redirect(s"/${repository.owner}/${repository.name}/releases") redirect(s"/${repository.owner}/${repository.name}/releases")
}) })

View File

@@ -21,14 +21,26 @@ import org.eclipse.jgit.lib.ObjectId
import gitbucket.core.model.WebHookContentType import gitbucket.core.model.WebHookContentType
import gitbucket.core.plugin.PluginRegistry import gitbucket.core.plugin.PluginRegistry
class RepositorySettingsController
class RepositorySettingsController extends RepositorySettingsControllerBase extends RepositorySettingsControllerBase
with RepositoryService with AccountService with WebHookService with ProtectedBranchService with CommitStatusService with DeployKeyService with RepositoryService
with OwnerAuthenticator with UsersAuthenticator with AccountService
with WebHookService
with ProtectedBranchService
with CommitStatusService
with DeployKeyService
with OwnerAuthenticator
with UsersAuthenticator
trait RepositorySettingsControllerBase extends ControllerBase { trait RepositorySettingsControllerBase extends ControllerBase {
self: RepositoryService with AccountService with WebHookService with ProtectedBranchService with CommitStatusService with DeployKeyService self: RepositoryService
with OwnerAuthenticator with UsersAuthenticator => with AccountService
with WebHookService
with ProtectedBranchService
with CommitStatusService
with DeployKeyService
with OwnerAuthenticator
with UsersAuthenticator =>
// for repository options // for repository options
case class OptionsForm( case class OptionsForm(
@@ -39,41 +51,51 @@ trait RepositorySettingsControllerBase extends ControllerBase {
externalIssuesUrl: Option[String], externalIssuesUrl: Option[String],
wikiOption: String, wikiOption: String,
externalWikiUrl: Option[String], externalWikiUrl: Option[String],
allowFork: Boolean allowFork: Boolean,
mergeOptions: Seq[String],
defaultMergeOption: String
) )
val optionsForm = mapping( val optionsForm = mapping(
"repositoryName" -> trim(label("Repository Name" , text(required, maxlength(100), repository, renameRepositoryName))), "repositoryName" -> trim(
"description" -> trim(label("Description" , optional(text()))), label("Repository Name", text(required, maxlength(100), repository, renameRepositoryName))
"isPrivate" -> trim(label("Repository Type" , boolean())), ),
"issuesOption" -> trim(label("Issues Option" , text(required, featureOption))), "description" -> trim(label("Description", optional(text()))),
"isPrivate" -> trim(label("Repository Type", boolean())),
"issuesOption" -> trim(label("Issues Option", text(required, featureOption))),
"externalIssuesUrl" -> trim(label("External Issues URL", optional(text(maxlength(200))))), "externalIssuesUrl" -> trim(label("External Issues URL", optional(text(maxlength(200))))),
"wikiOption" -> trim(label("Wiki Option" , text(required, featureOption))), "wikiOption" -> trim(label("Wiki Option", text(required, featureOption))),
"externalWikiUrl" -> trim(label("External Wiki URL" , optional(text(maxlength(200))))), "externalWikiUrl" -> trim(label("External Wiki URL", optional(text(maxlength(200))))),
"allowFork" -> trim(label("Allow Forking" , boolean())) "allowFork" -> trim(label("Allow Forking", boolean())),
)(OptionsForm.apply) "mergeOptions" -> mergeOptions,
"defaultMergeOption" -> trim(label("Default merge strategy", text(required)))
)(OptionsForm.apply).verifying { form =>
if (!form.mergeOptions.contains(form.defaultMergeOption)) {
Seq("defaultMergeOption" -> s"This merge strategy isn't enabled.")
} else Nil
}
// for default branch // for default branch
case class DefaultBranchForm(defaultBranch: String) case class DefaultBranchForm(defaultBranch: String)
val defaultBranchForm = mapping( val defaultBranchForm = mapping(
"defaultBranch" -> trim(label("Default Branch" , text(required, maxlength(100)))) "defaultBranch" -> trim(label("Default Branch", text(required, maxlength(100))))
)(DefaultBranchForm.apply) )(DefaultBranchForm.apply)
// for deploy key // for deploy key
case class DeployKeyForm(title: String, publicKey: String, allowWrite: Boolean) case class DeployKeyForm(title: String, publicKey: String, allowWrite: Boolean)
val deployKeyForm = mapping( val deployKeyForm = mapping(
"title" -> trim(label("Title", text(required, maxlength(100)))), "title" -> trim(label("Title", text(required, maxlength(100)))),
"publicKey" -> trim2(label("Key" , text(required))), // TODO duplication check in the repository? "publicKey" -> trim2(label("Key", text(required))), // TODO duplication check in the repository?
"allowWrite" -> trim(label("Key" , boolean())) "allowWrite" -> trim(label("Key", boolean()))
)(DeployKeyForm.apply) )(DeployKeyForm.apply)
// for web hook url addition // for web hook url addition
case class WebHookForm(url: String, events: Set[WebHook.Event], ctype: WebHookContentType, token: Option[String]) case class WebHookForm(url: String, events: Set[WebHook.Event], ctype: WebHookContentType, token: Option[String])
def webHookForm(update:Boolean) = mapping( def webHookForm(update: Boolean) =
mapping(
"url" -> trim(label("url", text(required, webHook(update)))), "url" -> trim(label("url", text(required, webHook(update)))),
"events" -> webhookEvents, "events" -> webhookEvents,
"ctype" -> label("ctype", text()), "ctype" -> label("ctype", text()),
@@ -118,27 +140,29 @@ trait RepositorySettingsControllerBase extends ControllerBase {
form.externalIssuesUrl, form.externalIssuesUrl,
form.wikiOption, form.wikiOption,
form.externalWikiUrl, form.externalWikiUrl,
form.allowFork form.allowFork,
form.mergeOptions,
form.defaultMergeOption
) )
// Change repository name // Change repository name
if(repository.name != form.repositoryName){ if (repository.name != form.repositoryName) {
// Update database // Update database
renameRepository(repository.owner, repository.name, repository.owner, form.repositoryName) renameRepository(repository.owner, repository.name, repository.owner, form.repositoryName)
// Move git repository // Move git repository
defining(getRepositoryDir(repository.owner, repository.name)){ dir => defining(getRepositoryDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory){ if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getRepositoryDir(repository.owner, form.repositoryName)) FileUtils.moveDirectory(dir, getRepositoryDir(repository.owner, form.repositoryName))
} }
} }
// Move wiki repository // Move wiki repository
defining(getWikiRepositoryDir(repository.owner, repository.name)){ dir => defining(getWikiRepositoryDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory) { if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getWikiRepositoryDir(repository.owner, form.repositoryName)) FileUtils.moveDirectory(dir, getWikiRepositoryDir(repository.owner, form.repositoryName))
} }
} }
// Move files directory // Move files directory
defining(getRepositoryFilesDir(repository.owner, repository.name)){ dir => defining(getRepositoryFilesDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory) { if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getRepositoryFilesDir(repository.owner, form.repositoryName)) FileUtils.moveDirectory(dir, getRepositoryFilesDir(repository.owner, form.repositoryName))
} }
} }
@@ -160,7 +184,7 @@ trait RepositorySettingsControllerBase extends ControllerBase {
/** Update default branch */ /** Update default branch */
post("/:owner/:repository/settings/update_default_branch", defaultBranchForm)(ownerOnly { (form, repository) => post("/:owner/:repository/settings/update_default_branch", defaultBranchForm)(ownerOnly { (form, repository) =>
if(!repository.branchList.contains(form.defaultBranch)){ if (!repository.branchList.contains(form.defaultBranch)) {
redirect(s"/${repository.owner}/${repository.name}/settings/options") redirect(s"/${repository.owner}/${repository.name}/settings/options")
} else { } else {
saveRepositoryDefaultBranch(repository.owner, repository.name, form.defaultBranch) saveRepositoryDefaultBranch(repository.owner, repository.name, form.defaultBranch)
@@ -177,12 +201,15 @@ trait RepositorySettingsControllerBase extends ControllerBase {
get("/:owner/:repository/settings/branches/:branch")(ownerOnly { repository => get("/:owner/:repository/settings/branches/:branch")(ownerOnly { repository =>
import gitbucket.core.api._ import gitbucket.core.api._
val branch = params("branch") val branch = params("branch")
if(!repository.branchList.contains(branch)){ if (!repository.branchList.contains(branch)) {
redirect(s"/${repository.owner}/${repository.name}/settings/branches") redirect(s"/${repository.owner}/${repository.name}/settings/branches")
} else { } else {
val protection = ApiBranchProtection(getProtectedBranchInfo(repository.owner, repository.name, branch)) val protection = ApiBranchProtection(getProtectedBranchInfo(repository.owner, repository.name, branch))
val lastWeeks = getRecentStatuesContexts(repository.owner, repository.name, val lastWeeks = getRecentStatuesContexts(
Date.from(LocalDateTime.now.minusWeeks(1).toInstant(ZoneOffset.UTC))).toSet repository.owner,
repository.name,
Date.from(LocalDateTime.now.minusWeeks(1).toInstant(ZoneOffset.UTC))
).toSet
val knownContexts = (lastWeeks ++ protection.status.contexts).toSeq.sortBy(identity) val knownContexts = (lastWeeks ++ protection.status.contexts).toSeq.sortBy(identity)
html.branchprotection(repository, branch, protection, knownContexts, flash.get("info")) html.branchprotection(repository, branch, protection, knownContexts, flash.get("info"))
} }
@@ -195,13 +222,14 @@ trait RepositorySettingsControllerBase extends ControllerBase {
html.collaborators( html.collaborators(
getCollaborators(repository.owner, repository.name), getCollaborators(repository.owner, repository.name),
getAccountByUserName(repository.owner).get.isGroupAccount, getAccountByUserName(repository.owner).get.isGroupAccount,
repository) repository
)
}) })
post("/:owner/:repository/settings/collaborators")(ownerOnly { repository => post("/:owner/:repository/settings/collaborators")(ownerOnly { repository =>
val collaborators = params("collaborators") val collaborators = params("collaborators")
removeCollaborators(repository.owner, repository.name) removeCollaborators(repository.owner, repository.name)
collaborators.split(",").withFilter(_.nonEmpty).map { collaborator => collaborators.split(",").withFilter(_.nonEmpty).foreach { collaborator =>
val userName :: role :: Nil = collaborator.split(":").toList val userName :: role :: Nil = collaborator.split(":").toList
addCollaborator(repository.owner, repository.name, userName, role) addCollaborator(repository.owner, repository.name, userName, role)
} }
@@ -245,9 +273,12 @@ trait RepositorySettingsControllerBase extends ControllerBase {
* Send the test request to registered web hook URLs. * Send the test request to registered web hook URLs.
*/ */
ajaxPost("/:owner/:repository/settings/hooks/test")(ownerOnly { repository => ajaxPost("/:owner/:repository/settings/hooks/test")(ownerOnly { repository =>
def _headers(h: Array[org.apache.http.Header]): Array[Array[String]] = h.map { h => Array(h.getName, h.getValue) } def _headers(h: Array[org.apache.http.Header]): Array[Array[String]] = h.map { h =>
Array(h.getName, h.getValue)
}
using(Git.open(getRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getRepositoryDir(repository.owner, repository.name))) {
git =>
import scala.collection.JavaConverters._ import scala.collection.JavaConverters._
import scala.concurrent.duration._ import scala.concurrent.duration._
import scala.concurrent._ import scala.concurrent._
@@ -261,10 +292,17 @@ trait RepositorySettingsControllerBase extends ControllerBase {
val dummyWebHookInfo = RepositoryWebHook(repository.owner, repository.name, url, ctype, token) val dummyWebHookInfo = RepositoryWebHook(repository.owner, repository.name, url, ctype, token)
val dummyPayload = { val dummyPayload = {
val ownerAccount = getAccountByUserName(repository.owner).get val ownerAccount = getAccountByUserName(repository.owner).get
val commits = if(JGitUtil.isEmpty(git)) List.empty else git.log val commits =
if (JGitUtil.isEmpty(git)) List.empty
else
git.log
.add(git.getRepository.resolve(repository.repository.defaultBranch)) .add(git.getRepository.resolve(repository.repository.defaultBranch))
.setMaxCount(4) .setMaxCount(4)
.call.iterator.asScala.map(new CommitInfo(_)).toList .call
.iterator
.asScala
.map(new CommitInfo(_))
.toList
val pushedCommit = commits.drop(1) val pushedCommit = commits.drop(1)
WebHookPushPayload( WebHookPushPayload(
@@ -281,26 +319,44 @@ trait RepositorySettingsControllerBase extends ControllerBase {
val (webHook, json, reqFuture, resFuture) = callWebHook(WebHook.Push, List(dummyWebHookInfo), dummyPayload).head val (webHook, json, reqFuture, resFuture) = callWebHook(WebHook.Push, List(dummyWebHookInfo), dummyPayload).head
val toErrorMap: PartialFunction[Throwable, Map[String,String]] = { val toErrorMap: PartialFunction[Throwable, Map[String, String]] = {
case e: java.net.UnknownHostException => Map("error"-> ("Unknown host " + e.getMessage)) case e: java.net.UnknownHostException => Map("error" -> ("Unknown host " + e.getMessage))
case e: java.lang.IllegalArgumentException => Map("error"-> ("invalid url")) case e: java.lang.IllegalArgumentException => Map("error" -> ("invalid url"))
case e: org.apache.http.client.ClientProtocolException => Map("error"-> ("invalid url")) case e: org.apache.http.client.ClientProtocolException => Map("error" -> ("invalid url"))
case NonFatal(e) => Map("error"-> (e.getClass + " "+ e.getMessage)) case NonFatal(e) => Map("error" -> (e.getClass + " " + e.getMessage))
} }
contentType = formats("json") contentType = formats("json")
org.json4s.jackson.Serialization.write(Map( org.json4s.jackson.Serialization.write(
Map(
"url" -> url, "url" -> url,
"request" -> Await.result(reqFuture.map(req => Map( "request" -> Await.result(
reqFuture
.map(
req =>
Map(
"headers" -> _headers(req.getAllHeaders), "headers" -> _headers(req.getAllHeaders),
"payload" -> json "payload" -> json
)).recover(toErrorMap), 20 seconds), )
"response" -> Await.result(resFuture.map(res => Map( )
.recover(toErrorMap),
20 seconds
),
"response" -> Await.result(
resFuture
.map(
res =>
Map(
"status" -> res.getStatusLine(), "status" -> res.getStatusLine(),
"body" -> EntityUtils.toString(res.getEntity()), "body" -> EntityUtils.toString(res.getEntity()),
"headers" -> _headers(res.getAllHeaders()) "headers" -> _headers(res.getAllHeaders())
)).recover(toErrorMap), 20 seconds) )
)) )
.recover(toErrorMap),
20 seconds
)
)
)
} }
}) })
@@ -308,7 +364,8 @@ trait RepositorySettingsControllerBase extends ControllerBase {
* Display the web hook edit page. * Display the web hook edit page.
*/ */
get("/:owner/:repository/settings/hooks/edit")(ownerOnly { repository => get("/:owner/:repository/settings/hooks/edit")(ownerOnly { repository =>
getWebHook(repository.owner, repository.name, params("url")).map{ case (webhook, events) => getWebHook(repository.owner, repository.name, params("url")).map {
case (webhook, events) =>
html.edithook(webhook, events, repository, false) html.edithook(webhook, events, repository, false)
} getOrElse NotFound() } getOrElse NotFound()
}) })
@@ -334,25 +391,25 @@ trait RepositorySettingsControllerBase extends ControllerBase {
*/ */
post("/:owner/:repository/settings/transfer", transferForm)(ownerOnly { (form, repository) => post("/:owner/:repository/settings/transfer", transferForm)(ownerOnly { (form, repository) =>
// Change repository owner // Change repository owner
if(repository.owner != form.newOwner){ if (repository.owner != form.newOwner) {
LockUtil.lock(s"${repository.owner}/${repository.name}"){ LockUtil.lock(s"${repository.owner}/${repository.name}") {
// Update database // Update database
renameRepository(repository.owner, repository.name, form.newOwner, repository.name) renameRepository(repository.owner, repository.name, form.newOwner, repository.name)
// Move git repository // Move git repository
defining(getRepositoryDir(repository.owner, repository.name)){ dir => defining(getRepositoryDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory){ if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getRepositoryDir(form.newOwner, repository.name)) FileUtils.moveDirectory(dir, getRepositoryDir(form.newOwner, repository.name))
} }
} }
// Move wiki repository // Move wiki repository
defining(getWikiRepositoryDir(repository.owner, repository.name)){ dir => defining(getWikiRepositoryDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory) { if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getWikiRepositoryDir(form.newOwner, repository.name)) FileUtils.moveDirectory(dir, getWikiRepositoryDir(form.newOwner, repository.name))
} }
} }
// Move files directory // Move files directory
defining(getRepositoryFilesDir(repository.owner, repository.name)){ dir => defining(getRepositoryFilesDir(repository.owner, repository.name)) { dir =>
if(dir.isDirectory) { if (dir.isDirectory) {
FileUtils.moveDirectory(dir, getRepositoryFilesDir(form.newOwner, repository.name)) FileUtils.moveDirectory(dir, getRepositoryFilesDir(form.newOwner, repository.name))
} }
} }
@@ -368,7 +425,7 @@ trait RepositorySettingsControllerBase extends ControllerBase {
* Delete the repository. * Delete the repository.
*/ */
post("/:owner/:repository/settings/delete")(ownerOnly { repository => post("/:owner/:repository/settings/delete")(ownerOnly { repository =>
LockUtil.lock(s"${repository.owner}/${repository.name}"){ LockUtil.lock(s"${repository.owner}/${repository.name}") {
// Delete the repository and related files // Delete the repository and related files
deleteRepository(repository.owner, repository.name) deleteRepository(repository.owner, repository.name)
@@ -418,10 +475,10 @@ trait RepositorySettingsControllerBase extends ControllerBase {
/** /**
* Provides duplication check for web hook url. * Provides duplication check for web hook url.
*/ */
private def webHook(needExists: Boolean): Constraint = new Constraint(){ private def webHook(needExists: Boolean): Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(getWebHook(params("owner"), params("repository"), value).isDefined != needExists){ if (getWebHook(params("owner"), params("repository"), value).isDefined != needExists) {
Some(if(needExists){ Some(if (needExists) {
"URL had not been registered yet." "URL had not been registered yet."
} else { } else {
"URL had been registered already." "URL had been registered already."
@@ -431,13 +488,14 @@ trait RepositorySettingsControllerBase extends ControllerBase {
} }
} }
private def webhookEvents = new ValueType[Set[WebHook.Event]]{ private def webhookEvents = new ValueType[Set[WebHook.Event]] {
def convert(name: String, params: Map[String, Seq[String]], messages: Messages): Set[WebHook.Event] = { def convert(name: String, params: Map[String, Seq[String]], messages: Messages): Set[WebHook.Event] = {
WebHook.Event.values.flatMap { t => WebHook.Event.values.flatMap { t =>
params.get(name + "." + t.name).map(_ => t) params.get(name + "." + t.name).map(_ => t)
}.toSet }.toSet
} }
def validate(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] = if(convert(name,params,messages).isEmpty){ def validate(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] =
if (convert(name, params, messages).isEmpty) {
Seq(name -> messages("error.required").format(name)) Seq(name -> messages("error.required").format(name))
} else { } else {
Nil Nil
@@ -462,8 +520,13 @@ trait RepositorySettingsControllerBase extends ControllerBase {
/** /**
* Duplicate check for the rename repository name. * Duplicate check for the rename repository name.
*/ */
private def renameRepositoryName: Constraint = new Constraint(){ private def renameRepositoryName: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = { override def validate(
name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] = {
for { for {
repoName <- params.optionValue("repository") if repoName != value repoName <- params.optionValue("repository") if repoName != value
userName <- params.optionValue("owner") userName <- params.optionValue("owner")
@@ -477,26 +540,50 @@ trait RepositorySettingsControllerBase extends ControllerBase {
/** /**
* *
*/ */
private def featureOption: Constraint = new Constraint(){ private def featureOption: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = override def validate(
if(Seq("DISABLE", "PRIVATE", "PUBLIC", "ALL").contains(value)) None else Some("Option is invalid.") name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] =
if (Seq("DISABLE", "PRIVATE", "PUBLIC", "ALL").contains(value)) None else Some("Option is invalid.")
} }
/** /**
* Provides Constraint to validate the repository transfer user. * Provides Constraint to validate the repository transfer user.
*/ */
private def transferUser: Constraint = new Constraint(){ private def transferUser: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
getAccountByUserName(value) match { getAccountByUserName(value) match {
case None => Some("User does not exist.") case None => Some("User does not exist.")
case Some(x) => if(x.userName == params("owner")){ case Some(x) =>
if (x.userName == params("owner")) {
Some("This is current repository owner.") Some("This is current repository owner.")
} else { } else {
params.get("repository").flatMap { repositoryName => params.get("repository").flatMap { repositoryName =>
getRepositoryNamesOfUser(x.userName).find(_ == repositoryName).map{ _ => "User already has same repository." } getRepositoryNamesOfUser(x.userName).find(_ == repositoryName).map { _ =>
"User already has same repository."
} }
} }
} }
} }
}
private def mergeOptions = new ValueType[Seq[String]] {
override def convert(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[String] = {
params.get("mergeOptions").getOrElse(Nil)
}
override def validate(name: String, params: Map[String, Seq[String]], messages: Messages): Seq[(String, String)] = {
val mergeOptions = params.get("mergeOptions").getOrElse(Nil)
if (mergeOptions.isEmpty) {
Seq("mergeOptions" -> "At least one option must be enabled.")
} else if (!mergeOptions.forall(x => Seq("merge-commit", "squash", "rebase").contains(x))) {
Seq("mergeOptions" -> "mergeOptions are invalid.")
} else {
Nil
}
}
}
} }

View File

@@ -15,6 +15,7 @@ import gitbucket.core.util.StringUtil._
import gitbucket.core.util.SyntaxSugars._ import gitbucket.core.util.SyntaxSugars._
import gitbucket.core.util.{AdminAuthenticator, Mailer} import gitbucket.core.util.{AdminAuthenticator, Mailer}
import org.apache.commons.io.IOUtils import org.apache.commons.io.IOUtils
import org.apache.commons.mail.EmailException
import org.json4s.jackson.Serialization import org.json4s.jackson.Serialization
import org.scalatra._ import org.scalatra._
import org.scalatra.forms._ import org.scalatra.forms._
@@ -23,8 +24,11 @@ import org.scalatra.i18n.Messages
import scala.collection.JavaConverters._ import scala.collection.JavaConverters._
import scala.collection.mutable.ListBuffer import scala.collection.mutable.ListBuffer
class SystemSettingsController extends SystemSettingsControllerBase class SystemSettingsController
with AccountService with RepositoryService with AdminAuthenticator extends SystemSettingsControllerBase
with AccountService
with RepositoryService
with AdminAuthenticator
case class Table(name: String, columns: Seq[Column]) case class Table(name: String, columns: Seq[Column])
case class Column(name: String, primaryKey: Boolean) case class Column(name: String, primaryKey: Boolean)
@@ -41,11 +45,15 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
"gravatar" -> trim(label("Gravatar", boolean())), "gravatar" -> trim(label("Gravatar", boolean())),
"notification" -> trim(label("Notification", boolean())), "notification" -> trim(label("Notification", boolean())),
"activityLogLimit" -> trim(label("Limit of activity logs", optional(number()))), "activityLogLimit" -> trim(label("Limit of activity logs", optional(number()))),
"ssh" -> trim(label("SSH access", boolean())), "ssh" -> mapping(
"sshHost" -> trim(label("SSH host", optional(text()))), "enabled" -> trim(label("SSH access", boolean())),
"sshPort" -> trim(label("SSH port", optional(number()))), "host" -> trim(label("SSH host", optional(text()))),
"port" -> trim(label("SSH port", optional(number()))),
)(Ssh.apply),
"useSMTP" -> trim(label("SMTP", boolean())), "useSMTP" -> trim(label("SMTP", boolean())),
"smtp" -> optionalIfNotChecked("useSMTP", mapping( "smtp" -> optionalIfNotChecked(
"useSMTP",
mapping(
"host" -> trim(label("SMTP Host", text(required))), "host" -> trim(label("SMTP Host", text(required))),
"port" -> trim(label("SMTP Port", optional(number()))), "port" -> trim(label("SMTP Port", optional(number()))),
"user" -> trim(label("SMTP User", optional(text()))), "user" -> trim(label("SMTP User", optional(text()))),
@@ -54,36 +62,47 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
"starttls" -> trim(label("Enable STARTTLS", optional(boolean()))), "starttls" -> trim(label("Enable STARTTLS", optional(boolean()))),
"fromAddress" -> trim(label("FROM Address", optional(text()))), "fromAddress" -> trim(label("FROM Address", optional(text()))),
"fromName" -> trim(label("FROM Name", optional(text()))) "fromName" -> trim(label("FROM Name", optional(text())))
)(Smtp.apply)), )(Smtp.apply)
),
"ldapAuthentication" -> trim(label("LDAP", boolean())), "ldapAuthentication" -> trim(label("LDAP", boolean())),
"ldap" -> optionalIfNotChecked("ldapAuthentication", mapping( "ldap" -> optionalIfNotChecked(
"ldapAuthentication",
mapping(
"host" -> trim(label("LDAP host", text(required))), "host" -> trim(label("LDAP host", text(required))),
"port" -> trim(label("LDAP port", optional(number()))), "port" -> trim(label("LDAP port", optional(number()))),
"bindDN" -> trim(label("Bind DN", optional(text()))), "bindDN" -> trim(label("Bind DN", optional(text()))),
"bindPassword" -> trim(label("Bind Password", optional(text()))), "bindPassword" -> trim(label("Bind Password", optional(text()))),
"baseDN" -> trim(label("Base DN", text(required))), "baseDN" -> trim(label("Base DN", text(required))),
"userNameAttribute" -> trim(label("User name attribute", text(required))), "userNameAttribute" -> trim(label("User name attribute", text(required))),
"additionalFilterCondition"-> trim(label("Additional filter condition", optional(text()))), "additionalFilterCondition" -> trim(label("Additional filter condition", optional(text()))),
"fullNameAttribute" -> trim(label("Full name attribute", optional(text()))), "fullNameAttribute" -> trim(label("Full name attribute", optional(text()))),
"mailAttribute" -> trim(label("Mail address attribute", optional(text()))), "mailAttribute" -> trim(label("Mail address attribute", optional(text()))),
"tls" -> trim(label("Enable TLS", optional(boolean()))), "tls" -> trim(label("Enable TLS", optional(boolean()))),
"ssl" -> trim(label("Enable SSL", optional(boolean()))), "ssl" -> trim(label("Enable SSL", optional(boolean()))),
"keystore" -> trim(label("Keystore", optional(text()))) "keystore" -> trim(label("Keystore", optional(text())))
)(Ldap.apply)), )(Ldap.apply)
),
"oidcAuthentication" -> trim(label("OIDC", boolean())), "oidcAuthentication" -> trim(label("OIDC", boolean())),
"oidc" -> optionalIfNotChecked("oidcAuthentication", mapping( "oidc" -> optionalIfNotChecked(
"oidcAuthentication",
mapping(
"issuer" -> trim(label("Issuer", text(required))), "issuer" -> trim(label("Issuer", text(required))),
"clientID" -> trim(label("Client ID", text(required))), "clientID" -> trim(label("Client ID", text(required))),
"clientSecret" -> trim(label("Client secret", text(required))), "clientSecret" -> trim(label("Client secret", text(required))),
"jwsAlgorithm" -> trim(label("Signature algorithm", optional(text()))) "jwsAlgorithm" -> trim(label("Signature algorithm", optional(text())))
)(OIDC.apply)), )(OIDC.apply)
"skinName" -> trim(label("AdminLTE skin name", text(required))) ),
"skinName" -> trim(label("AdminLTE skin name", text(required))),
"showMailAddress" -> trim(label("Show mail address", boolean())),
"pluginNetworkInstall" -> new SingleValueType[Boolean] {
override def convert(value: String, messages: Messages): Boolean = context.settings.pluginNetworkInstall
}
)(SystemSettings.apply).verifying { settings => )(SystemSettings.apply).verifying { settings =>
Vector( Vector(
if(settings.ssh && settings.baseUrl.isEmpty){ if (settings.ssh.enabled && settings.baseUrl.isEmpty) {
Some("baseUrl" -> "Base URL is required if SSH access is enabled.") Some("baseUrl" -> "Base URL is required if SSH access is enabled.")
} else None, } else None,
if(settings.ssh && settings.sshHost.isEmpty){ if (settings.ssh.enabled && settings.ssh.sshHost.isEmpty) {
Some("sshHost" -> "SSH host is required if SSH access is enabled.") Some("sshHost" -> "SSH host is required if SSH access is enabled.")
} else None } else None
).flatten ).flatten
@@ -107,82 +126,117 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
case class DataExportForm(tableNames: List[String]) case class DataExportForm(tableNames: List[String])
case class NewUserForm(userName: String, password: String, fullName: String, case class NewUserForm(
mailAddress: String, isAdmin: Boolean, userName: String,
description: Option[String], url: Option[String], fileId: Option[String]) password: String,
fullName: String,
mailAddress: String,
extraMailAddresses: List[String],
isAdmin: Boolean,
description: Option[String],
url: Option[String],
fileId: Option[String]
)
case class EditUserForm(userName: String, password: Option[String], fullName: String, case class EditUserForm(
mailAddress: String, isAdmin: Boolean, description: Option[String], url: Option[String], userName: String,
fileId: Option[String], clearImage: Boolean, isRemoved: Boolean) password: Option[String],
fullName: String,
mailAddress: String,
extraMailAddresses: List[String],
isAdmin: Boolean,
description: Option[String],
url: Option[String],
fileId: Option[String],
clearImage: Boolean,
isRemoved: Boolean
)
case class NewGroupForm(groupName: String, description: Option[String], url: Option[String], fileId: Option[String], case class NewGroupForm(
members: String) groupName: String,
description: Option[String],
case class EditGroupForm(groupName: String, description: Option[String], url: Option[String], fileId: Option[String], url: Option[String],
members: String, clearImage: Boolean, isRemoved: Boolean) fileId: Option[String],
members: String
)
case class EditGroupForm(
groupName: String,
description: Option[String],
url: Option[String],
fileId: Option[String],
members: String,
clearImage: Boolean,
isRemoved: Boolean
)
val newUserForm = mapping( val newUserForm = mapping(
"userName" -> trim(label("Username" ,text(required, maxlength(100), identifier, uniqueUserName, reservedNames))), "userName" -> trim(label("Username", text(required, maxlength(100), identifier, uniqueUserName, reservedNames))),
"password" -> trim(label("Password" ,text(required, maxlength(20), password))), "password" -> trim(label("Password", text(required, maxlength(20), password))),
"fullName" -> trim(label("Full Name" ,text(required, maxlength(100)))), "fullName" -> trim(label("Full Name", text(required, maxlength(100)))),
"mailAddress" -> trim(label("Mail Address" ,text(required, maxlength(100), uniqueMailAddress()))), "mailAddress" -> trim(label("Mail Address", text(required, maxlength(100), uniqueMailAddress()))),
"isAdmin" -> trim(label("User Type" ,boolean())), "extraMailAddresses" -> list(
"description" -> trim(label("bio" ,optional(text()))), trim(label("Additional Mail Address", text(maxlength(100), uniqueExtraMailAddress("userName"))))
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), ),
"fileId" -> trim(label("File ID" ,optional(text()))) "isAdmin" -> trim(label("User Type", boolean())),
"description" -> trim(label("bio", optional(text()))),
"url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID", optional(text())))
)(NewUserForm.apply) )(NewUserForm.apply)
val editUserForm = mapping( val editUserForm = mapping(
"userName" -> trim(label("Username" ,text(required, maxlength(100), identifier))), "userName" -> trim(label("Username", text(required, maxlength(100), identifier))),
"password" -> trim(label("Password" ,optional(text(maxlength(20), password)))), "password" -> trim(label("Password", optional(text(maxlength(20), password)))),
"fullName" -> trim(label("Full Name" ,text(required, maxlength(100)))), "fullName" -> trim(label("Full Name", text(required, maxlength(100)))),
"mailAddress" -> trim(label("Mail Address" ,text(required, maxlength(100), uniqueMailAddress("userName")))), "mailAddress" -> trim(label("Mail Address", text(required, maxlength(100), uniqueMailAddress("userName")))),
"isAdmin" -> trim(label("User Type" ,boolean())), "extraMailAddresses" -> list(
"description" -> trim(label("bio" ,optional(text()))), trim(label("Additional Mail Address", text(maxlength(100), uniqueExtraMailAddress("userName"))))
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), ),
"fileId" -> trim(label("File ID" ,optional(text()))), "isAdmin" -> trim(label("User Type", boolean())),
"clearImage" -> trim(label("Clear image" ,boolean())), "description" -> trim(label("bio", optional(text()))),
"removed" -> trim(label("Disable" ,boolean(disableByNotYourself("userName")))) "url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID", optional(text()))),
"clearImage" -> trim(label("Clear image", boolean())),
"removed" -> trim(label("Disable", boolean(disableByNotYourself("userName"))))
)(EditUserForm.apply) )(EditUserForm.apply)
val newGroupForm = mapping( val newGroupForm = mapping(
"groupName" -> trim(label("Group name" ,text(required, maxlength(100), identifier, uniqueUserName, reservedNames))), "groupName" -> trim(label("Group name", text(required, maxlength(100), identifier, uniqueUserName, reservedNames))),
"description" -> trim(label("Group description", optional(text()))), "description" -> trim(label("Group description", optional(text()))),
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), "url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID" ,optional(text()))), "fileId" -> trim(label("File ID", optional(text()))),
"members" -> trim(label("Members" ,text(required, members))) "members" -> trim(label("Members", text(required, members)))
)(NewGroupForm.apply) )(NewGroupForm.apply)
val editGroupForm = mapping( val editGroupForm = mapping(
"groupName" -> trim(label("Group name" ,text(required, maxlength(100), identifier))), "groupName" -> trim(label("Group name", text(required, maxlength(100), identifier))),
"description" -> trim(label("Group description", optional(text()))), "description" -> trim(label("Group description", optional(text()))),
"url" -> trim(label("URL" ,optional(text(maxlength(200))))), "url" -> trim(label("URL", optional(text(maxlength(200))))),
"fileId" -> trim(label("File ID" ,optional(text()))), "fileId" -> trim(label("File ID", optional(text()))),
"members" -> trim(label("Members" ,text(required, members))), "members" -> trim(label("Members", text(required, members))),
"clearImage" -> trim(label("Clear image" ,boolean())), "clearImage" -> trim(label("Clear image", boolean())),
"removed" -> trim(label("Disable" ,boolean())) "removed" -> trim(label("Disable", boolean()))
)(EditGroupForm.apply) )(EditGroupForm.apply)
get("/admin/dbviewer")(adminOnly { get("/admin/dbviewer")(adminOnly {
val conn = request2Session(request).conn val conn = request2Session(request).conn
val meta = conn.getMetaData val meta = conn.getMetaData
val tables = ListBuffer[Table]() val tables = ListBuffer[Table]()
using(meta.getTables(null, "%", "%", Array("TABLE", "VIEW"))){ rs => using(meta.getTables(null, "%", "%", Array("TABLE", "VIEW"))) {
while(rs.next()){ rs =>
while (rs.next()) {
val tableName = rs.getString("TABLE_NAME") val tableName = rs.getString("TABLE_NAME")
val pkColumns = ListBuffer[String]() val pkColumns = ListBuffer[String]()
using(meta.getPrimaryKeys(null, null, tableName)){ rs => using(meta.getPrimaryKeys(null, null, tableName)) { rs =>
while(rs.next()){ while (rs.next()) {
pkColumns += rs.getString("COLUMN_NAME").toUpperCase pkColumns += rs.getString("COLUMN_NAME").toUpperCase
} }
} }
val columns = ListBuffer[Column]() val columns = ListBuffer[Column]()
using(meta.getColumns(null, "%", tableName, "%")){ rs => using(meta.getColumns(null, "%", tableName, "%")) { rs =>
while(rs.next()){ while (rs.next()) {
val columnName = rs.getString("COLUMN_NAME").toUpperCase val columnName = rs.getString("COLUMN_NAME").toUpperCase
columns += Column(columnName, pkColumns.contains(columnName)) columns += Column(columnName, pkColumns.contains(columnName))
} }
@@ -196,20 +250,23 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
post("/admin/dbviewer/_query")(adminOnly { post("/admin/dbviewer/_query")(adminOnly {
contentType = formats("json") contentType = formats("json")
params.get("query").collectFirst { case query if query.trim.nonEmpty => params.get("query").collectFirst {
case query if query.trim.nonEmpty =>
val trimmedQuery = query.trim val trimmedQuery = query.trim
if(trimmedQuery.nonEmpty){ if (trimmedQuery.nonEmpty) {
try { try {
val conn = request2Session(request).conn val conn = request2Session(request).conn
using(conn.prepareStatement(query)){ stmt => using(conn.prepareStatement(query)) {
if(trimmedQuery.toUpperCase.startsWith("SELECT")){ stmt =>
using(stmt.executeQuery()){ rs => if (trimmedQuery.toUpperCase.startsWith("SELECT")) {
using(stmt.executeQuery()) {
rs =>
val meta = rs.getMetaData val meta = rs.getMetaData
val columns = for(i <- 1 to meta.getColumnCount) yield { val columns = for (i <- 1 to meta.getColumnCount) yield {
meta.getColumnName(i) meta.getColumnName(i)
} }
val result = ListBuffer[Map[String, String]]() val result = ListBuffer[Map[String, String]]()
while(rs.next()){ while (rs.next()) {
val row = columns.map { columnName => val row = columns.map { columnName =>
columnName -> Option(rs.getObject(columnName)).map(_.toString).getOrElse("<NULL>") columnName -> Option(rs.getObject(columnName)).map(_.toString).getOrElse("<NULL>")
}.toMap }.toMap
@@ -231,7 +288,7 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
}) })
get("/admin/system")(adminOnly { get("/admin/system")(adminOnly {
html.system(flash.get("info")) html.settings(flash.get("info"))
}) })
post("/admin/system", form)(adminOnly { form => post("/admin/system", form)(adminOnly { form =>
@@ -242,8 +299,7 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
for { for {
sshAddress <- form.sshAddress sshAddress <- form.sshAddress
baseUrl <- form.baseUrl baseUrl <- form.baseUrl
} } SshServer.start(sshAddress, baseUrl)
SshServer.start(sshAddress, baseUrl)
} }
flash += "info" -> "System settings has been updated." flash += "info" -> "System settings has been updated."
@@ -263,84 +319,125 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
"Test mail has been sent to: " + form.testAddress "Test mail has been sent to: " + form.testAddress
} catch { } catch {
case e: Exception => "[Error] " + e.toString case e: EmailException => s"[Error] ${e.getCause}"
case e: Exception => s"[Error] ${e.toString}"
} }
}) })
get("/admin/plugins")(adminOnly { get("/admin/plugins")(adminOnly {
// Installed plugins // Installed plugins
val enabledPlugins = PluginRegistry().getPlugins() val enabledPlugins = PluginRegistry().getPlugins()
val gitbucketVersion = GitBucketCoreModule.getVersions.asScala.last.getVersion
val gitbucketSemver = Semver.valueOf(gitbucketVersion)
val gitbucketVersion = Semver.valueOf(GitBucketCoreModule.getVersions.asScala.last.getVersion) // Plugins in the remote repository
val repositoryPlugins = if (context.settings.pluginNetworkInstall) {
// Plugins in the local repository PluginRepository
val repositoryPlugins = PluginRepository.getPlugins() .getPlugins()
.filterNot { meta => .map {
enabledPlugins.exists { plugin => plugin.pluginId == meta.id && meta =>
Semver.valueOf(plugin.pluginVersion).greaterThanOrEqualTo(Semver.valueOf(meta.latestVersion.version)) (meta, meta.versions.reverse.find {
version =>
val semver = Semver.valueOf(version.version)
gitbucketVersion == version.gitbucketVersion && !enabledPlugins.exists { plugin =>
if (plugin.pluginId == meta.id) {
Semver.valueOf(plugin.pluginVersion) match {
case x if x.greaterThan(semver) => true
case x if x.equals(semver) =>
plugin.gitbucketVersion match {
case None => true
case Some(x) => Semver.valueOf(x).greaterThanOrEqualTo(gitbucketSemver)
} }
}.map { meta => case _ => false
(meta, meta.versions.reverse.find { version => gitbucketVersion.satisfies(version.range) }) }
}.collect { case (meta, Some(version)) => } else false
}
})
}
.collect {
case (meta, Some(version)) =>
new PluginInfoBase( new PluginInfoBase(
pluginId = meta.id, pluginId = meta.id,
pluginName = meta.name, pluginName = meta.name,
pluginVersion = version.version, pluginVersion = version.version,
gitbucketVersion = Some(version.gitbucketVersion),
description = meta.description description = meta.description
) )
} }
} else Nil
// Merge // Merge
val plugins = enabledPlugins.map((_, true)) ++ repositoryPlugins.map((_, false)) val plugins = (enabledPlugins.map((_, true)) ++ repositoryPlugins.map((_, false)))
.groupBy(_._1.pluginId)
.map {
case (pluginId, plugins) =>
val (plugin, enabled) = plugins.head
(plugin, enabled, if (plugins.length > 1) plugins.last._1.pluginVersion else "")
}
.toList
html.plugins(plugins, flash.get("info")) html.plugins(plugins, flash.get("info"))
}) })
post("/admin/plugins/_reload")(adminOnly { post("/admin/plugins/_reload")(adminOnly {
// Update configuration
val pluginNetworkInstall = params.get("pluginNetworkInstall").map(_.toBoolean).getOrElse(false)
saveSystemSettings(context.settings.copy(pluginNetworkInstall = pluginNetworkInstall))
// Reload plugins
PluginRegistry.reload(request.getServletContext(), loadSystemSettings(), request2Session(request).conn) PluginRegistry.reload(request.getServletContext(), loadSystemSettings(), request2Session(request).conn)
flash += "info" -> "All plugins were reloaded." flash += "info" -> "All plugins were reloaded."
redirect("/admin/plugins") redirect("/admin/plugins")
}) })
post("/admin/plugins/:pluginId/:version/_uninstall")(adminOnly { post("/admin/plugins/:pluginId/_uninstall")(adminOnly {
val pluginId = params("pluginId") val pluginId = params("pluginId")
val version = params("version")
PluginRegistry().getPlugins() if (PluginRegistry().getPlugins().exists(_.pluginId == pluginId)) {
.collect { case plugin if (plugin.pluginId == pluginId && plugin.pluginVersion == version) => plugin } PluginRegistry
.foreach { _ => .uninstall(pluginId, request.getServletContext, loadSystemSettings(), request2Session(request).conn)
PluginRegistry.uninstall(pluginId, request.getServletContext, loadSystemSettings(), request2Session(request).conn)
flash += "info" -> s"${pluginId} was uninstalled." flash += "info" -> s"${pluginId} was uninstalled."
} }
redirect("/admin/plugins") redirect("/admin/plugins")
}) })
post("/admin/plugins/:pluginId/:version/_install")(adminOnly { post("/admin/plugins/:pluginId/:version/_install")(adminOnly {
if (context.settings.pluginNetworkInstall) {
val pluginId = params("pluginId") val pluginId = params("pluginId")
val version = params("version") val version = params("version")
/// TODO!!!! val gitbucketVersion = GitBucketCoreModule.getVersions.asScala.last.getVersion
PluginRepository.getPlugins()
.collect { case meta if meta.id == pluginId => (meta, meta.versions.find(_.version == version) )} PluginRepository
.foreach { case (meta, version) => .getPlugins()
.collectFirst {
case meta if meta.id == pluginId =>
(meta, meta.versions.find(x => x.gitbucketVersion == gitbucketVersion && x.version == version))
}
.foreach {
case (meta, version) =>
version.foreach { version => version.foreach { version =>
// TODO Install version!
PluginRegistry.install( PluginRegistry.install(
new java.io.File(PluginHome, s".repository/${version.file}"), pluginId,
new java.net.URL(version.url),
request.getServletContext, request.getServletContext,
loadSystemSettings(), loadSystemSettings(),
request2Session(request).conn request2Session(request).conn
) )
flash += "info" -> s"${pluginId} was installed." flash += "info" -> s"${pluginId}:${version.version} was installed."
} }
} }
}
redirect("/admin/plugins") redirect("/admin/plugins")
}) })
get("/admin/users")(adminOnly { get("/admin/users")(adminOnly {
val includeRemoved = params.get("includeRemoved").map(_.toBoolean).getOrElse(false) val includeRemoved = params.get("includeRemoved").map(_.toBoolean).getOrElse(false)
val includeGroups = params.get("includeGroups").map(_.toBoolean).getOrElse(false) val includeGroups = params.get("includeGroups").map(_.toBoolean).getOrElse(false)
val users = getAllUsers(includeRemoved, includeGroups) val users = getAllUsers(includeRemoved, includeGroups)
val members = users.collect { case account if(account.isGroupAccount) => val members = users.collect {
case account if (account.isGroupAccount) =>
account.userName -> getGroupMembers(account.userName).map(_.userName) account.userName -> getGroupMembers(account.userName).map(_.userName)
}.toMap }.toMap
@@ -348,28 +445,39 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
}) })
get("/admin/users/_newuser")(adminOnly { get("/admin/users/_newuser")(adminOnly {
html.user(None) html.user(None, Nil)
}) })
post("/admin/users/_newuser", newUserForm)(adminOnly { form => post("/admin/users/_newuser", newUserForm)(adminOnly { form =>
createAccount(form.userName, sha1(form.password), form.fullName, form.mailAddress, form.isAdmin, form.description, form.url) createAccount(
form.userName,
pbkdf2_sha256(form.password),
form.fullName,
form.mailAddress,
form.isAdmin,
form.description,
form.url
)
updateImage(form.userName, form.fileId, false) updateImage(form.userName, form.fileId, false)
updateAccountExtraMailAddresses(form.userName, form.extraMailAddresses.filter(_ != ""))
redirect("/admin/users") redirect("/admin/users")
}) })
get("/admin/users/:userName/_edituser")(adminOnly { get("/admin/users/:userName/_edituser")(adminOnly {
val userName = params("userName") val userName = params("userName")
html.user(getAccountByUserName(userName, true), flash.get("error")) val extraMails = getAccountExtraMailAddresses(userName)
html.user(getAccountByUserName(userName, true), extraMails, flash.get("error"))
}) })
post("/admin/users/:name/_edituser", editUserForm)(adminOnly { form => post("/admin/users/:name/_edituser", editUserForm)(adminOnly { form =>
val userName = params("userName") val userName = params("userName")
getAccountByUserName(userName, true).map { account => getAccountByUserName(userName, true).map {
if(account.isAdmin && (form.isRemoved || !form.isAdmin) && isLastAdministrator(account)){ account =>
if (account.isAdmin && (form.isRemoved || !form.isAdmin) && isLastAdministrator(account)) {
flash += "error" -> "Account can't be turned off because this is last one administrator." flash += "error" -> "Account can't be turned off because this is last one administrator."
redirect(s"/admin/users/${userName}/_edituser") redirect(s"/admin/users/${userName}/_edituser")
} else { } else {
if(form.isRemoved){ if (form.isRemoved) {
// Remove repositories // Remove repositories
// getRepositoryNamesOfUser(userName).foreach { repositoryName => // getRepositoryNamesOfUser(userName).foreach { repositoryName =>
// deleteRepository(userName, repositoryName) // deleteRepository(userName, repositoryName)
@@ -381,19 +489,23 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
removeUserRelatedData(userName) removeUserRelatedData(userName)
} }
updateAccount(account.copy( updateAccount(
password = form.password.map(sha1).getOrElse(account.password), account.copy(
password = form.password.map(pbkdf2_sha256).getOrElse(account.password),
fullName = form.fullName, fullName = form.fullName,
mailAddress = form.mailAddress, mailAddress = form.mailAddress,
isAdmin = form.isAdmin, isAdmin = form.isAdmin,
description = form.description, description = form.description,
url = form.url, url = form.url,
isRemoved = form.isRemoved)) isRemoved = form.isRemoved
)
)
updateImage(userName, form.fileId, form.clearImage) updateImage(userName, form.fileId, form.clearImage)
updateAccountExtraMailAddresses(userName, form.extraMailAddresses.filter(_ != ""))
// call hooks // call hooks
if(form.isRemoved) PluginRegistry().getAccountHooks.foreach(_.deleted(userName)) if (form.isRemoved) PluginRegistry().getAccountHooks.foreach(_.deleted(userName))
redirect("/admin/users") redirect("/admin/users")
} }
@@ -406,31 +518,45 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
post("/admin/users/_newgroup", newGroupForm)(adminOnly { form => post("/admin/users/_newgroup", newGroupForm)(adminOnly { form =>
createGroup(form.groupName, form.description, form.url) createGroup(form.groupName, form.description, form.url)
updateGroupMembers(form.groupName, form.members.split(",").map { updateGroupMembers(
form.groupName,
form.members
.split(",")
.map {
_.split(":") match { _.split(":") match {
case Array(userName, isManager) => (userName, isManager.toBoolean) case Array(userName, isManager) => (userName, isManager.toBoolean)
} }
}.toList) }
.toList
)
updateImage(form.groupName, form.fileId, false) updateImage(form.groupName, form.fileId, false)
redirect("/admin/users") redirect("/admin/users")
}) })
get("/admin/users/:groupName/_editgroup")(adminOnly { get("/admin/users/:groupName/_editgroup")(adminOnly {
defining(params("groupName")){ groupName => defining(params("groupName")) { groupName =>
html.usergroup(getAccountByUserName(groupName, true), getGroupMembers(groupName)) html.usergroup(getAccountByUserName(groupName, true), getGroupMembers(groupName))
} }
}) })
post("/admin/users/:groupName/_editgroup", editGroupForm)(adminOnly { form => post("/admin/users/:groupName/_editgroup", editGroupForm)(adminOnly { form =>
defining(params("groupName"), form.members.split(",").map { defining(
params("groupName"),
form.members
.split(",")
.map {
_.split(":") match { _.split(":") match {
case Array(userName, isManager) => (userName, isManager.toBoolean) case Array(userName, isManager) => (userName, isManager.toBoolean)
} }
}.toList){ case (groupName, members) => }
getAccountByUserName(groupName, true).map { account => .toList
) {
case (groupName, members) =>
getAccountByUserName(groupName, true).map {
account =>
updateGroup(groupName, form.description, form.url, form.isRemoved) updateGroup(groupName, form.description, form.url, form.isRemoved)
if(form.isRemoved){ if (form.isRemoved) {
// Remove from GROUP_MEMBER // Remove from GROUP_MEMBER
updateGroupMembers(form.groupName, Nil) updateGroupMembers(form.groupName, Nil)
// // Remove repositories // // Remove repositories
@@ -473,25 +599,26 @@ trait SystemSettingsControllerBase extends AccountManagementControllerBase {
response.setHeader("Content-Disposition", "attachment; filename=" + file.getName) response.setHeader("Content-Disposition", "attachment; filename=" + file.getName)
response.setContentLength(file.length.toInt) response.setContentLength(file.length.toInt)
using(new FileInputStream(file)){ in => using(new FileInputStream(file)) { in =>
IOUtils.copy(in, response.outputStream) IOUtils.copy(in, response.outputStream)
} }
() ()
}) })
private def members: Constraint = new Constraint(){ private def members: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
if(value.split(",").exists { if (value.split(",").exists {
_.split(":") match { case Array(userName, isManager) => isManager.toBoolean } _.split(":") match { case Array(userName, isManager) => isManager.toBoolean }
}) None else Some("Must select one manager at least.") }) None
else Some("Must select one manager at least.")
} }
} }
protected def disableByNotYourself(paramName: String): Constraint = new Constraint() { protected def disableByNotYourself(paramName: String): Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
params.get(paramName).flatMap { userName => params.get(paramName).flatMap { userName =>
if(userName == context.loginAccount.get.userName && params.get("removed") == Some("true")) if (userName == context.loginAccount.get.userName && params.get("removed") == Some("true"))
Some("You can't disable your account yourself") Some("You can't disable your account yourself")
else else
None None

View File

@@ -14,58 +14,58 @@ trait ValidationSupport extends FormSupport { self: ServletBase with JacksonJson
def get[T](path: String, form: ValueType[T])(action: T => Any): Route = { def get[T](path: String, form: ValueType[T])(action: T => Any): Route = {
registerValidate(path, form) registerValidate(path, form)
get(path){ get(path) {
validate(form)(errors => BadRequest(), form => action(form)) validate(form)(errors => BadRequest(), form => action(form))
} }
} }
def post[T](path: String, form: ValueType[T])(action: T => Any): Route = { def post[T](path: String, form: ValueType[T])(action: T => Any): Route = {
registerValidate(path, form) registerValidate(path, form)
post(path){ post(path) {
validate(form)(errors => BadRequest(), form => action(form)) validate(form)(errors => BadRequest(), form => action(form))
} }
} }
def put[T](path: String, form: ValueType[T])(action: T => Any): Route = { def put[T](path: String, form: ValueType[T])(action: T => Any): Route = {
registerValidate(path, form) registerValidate(path, form)
put(path){ put(path) {
validate(form)(errors => BadRequest(), form => action(form)) validate(form)(errors => BadRequest(), form => action(form))
} }
} }
def delete[T](path: String, form: ValueType[T])(action: T => Any): Route = { def delete[T](path: String, form: ValueType[T])(action: T => Any): Route = {
registerValidate(path, form) registerValidate(path, form)
delete(path){ delete(path) {
validate(form)(errors => BadRequest(), form => action(form)) validate(form)(errors => BadRequest(), form => action(form))
} }
} }
def ajaxGet[T](path: String, form: ValueType[T])(action: T => Any): Route = { def ajaxGet[T](path: String, form: ValueType[T])(action: T => Any): Route = {
get(path){ get(path) {
validate(form)(errors => ajaxError(errors), form => action(form)) validate(form)(errors => ajaxError(errors), form => action(form))
} }
} }
def ajaxPost[T](path: String, form: ValueType[T])(action: T => Any): Route = { def ajaxPost[T](path: String, form: ValueType[T])(action: T => Any): Route = {
post(path){ post(path) {
validate(form)(errors => ajaxError(errors), form => action(form)) validate(form)(errors => ajaxError(errors), form => action(form))
} }
} }
def ajaxDelete[T](path: String, form: ValueType[T])(action: T => Any): Route = { def ajaxDelete[T](path: String, form: ValueType[T])(action: T => Any): Route = {
delete(path){ delete(path) {
validate(form)(errors => ajaxError(errors), form => action(form)) validate(form)(errors => ajaxError(errors), form => action(form))
} }
} }
def ajaxPut[T](path: String, form: ValueType[T])(action: T => Any): Route = { def ajaxPut[T](path: String, form: ValueType[T])(action: T => Any): Route = {
put(path){ put(path) {
validate(form)(errors => ajaxError(errors), form => action(form)) validate(form)(errors => ajaxError(errors), form => action(form))
} }
} }
private def registerValidate[T](path: String, form: ValueType[T]) = { private def registerValidate[T](path: String, form: ValueType[T]) = {
post(path.replaceFirst("/$", "") + "/validate"){ post(path.replaceFirst("/$", "") + "/validate") {
contentType = "application/json" contentType = "application/json"
toJson(form.validate("", multiParams, messages)) toJson(form.validate("", multiParams, messages))
} }
@@ -84,7 +84,8 @@ trait ValidationSupport extends FormSupport { self: ServletBase with JacksonJson
* Converts errors to JSON. * Converts errors to JSON.
*/ */
private def toJson(errors: Seq[(String, String)]): JObject = private def toJson(errors: Seq[(String, String)]): JObject =
JObject(errors.map { case (key, value) => JObject(errors.map {
case (key, value) =>
JField(key, JString(value)) JField(key, JString(value))
}.toList) }.toList)

View File

@@ -14,38 +14,60 @@ import org.scalatra.forms._
import org.eclipse.jgit.api.Git import org.eclipse.jgit.api.Git
import org.scalatra.i18n.Messages import org.scalatra.i18n.Messages
class WikiController extends WikiControllerBase class WikiController
with WikiService with RepositoryService with AccountService with ActivityService with WebHookService extends WikiControllerBase
with ReadableUsersAuthenticator with ReferrerAuthenticator with WikiService
with RepositoryService
with AccountService
with ActivityService
with WebHookService
with ReadableUsersAuthenticator
with ReferrerAuthenticator
trait WikiControllerBase extends ControllerBase { trait WikiControllerBase extends ControllerBase {
self: WikiService with RepositoryService with AccountService with ActivityService with WebHookService self: WikiService
with ReadableUsersAuthenticator with ReferrerAuthenticator => with RepositoryService
with AccountService
with ActivityService
with WebHookService
with ReadableUsersAuthenticator
with ReferrerAuthenticator =>
case class WikiPageEditForm(pageName: String, content: String, message: Option[String], currentPageName: String, id: String) case class WikiPageEditForm(
pageName: String,
content: String,
message: Option[String],
currentPageName: String,
id: String
)
val newForm = mapping( val newForm = mapping(
"pageName" -> trim(label("Page name" , text(required, maxlength(40), pagename, unique))), "pageName" -> trim(label("Page name", text(required, maxlength(40), pagename, unique))),
"content" -> trim(label("Content" , text(required, conflictForNew))), "content" -> trim(label("Content", text(required, conflictForNew))),
"message" -> trim(label("Message" , optional(text()))), "message" -> trim(label("Message", optional(text()))),
"currentPageName" -> trim(label("Current page name" , text())), "currentPageName" -> trim(label("Current page name", text())),
"id" -> trim(label("Latest commit id" , text())) "id" -> trim(label("Latest commit id", text()))
)(WikiPageEditForm.apply) )(WikiPageEditForm.apply)
val editForm = mapping( val editForm = mapping(
"pageName" -> trim(label("Page name" , text(required, maxlength(40), pagename))), "pageName" -> trim(label("Page name", text(required, maxlength(40), pagename))),
"content" -> trim(label("Content" , text(required, conflictForEdit))), "content" -> trim(label("Content", text(required, conflictForEdit))),
"message" -> trim(label("Message" , optional(text()))), "message" -> trim(label("Message", optional(text()))),
"currentPageName" -> trim(label("Current page name" , text(required))), "currentPageName" -> trim(label("Current page name", text(required))),
"id" -> trim(label("Latest commit id" , text(required))) "id" -> trim(label("Latest commit id", text(required)))
)(WikiPageEditForm.apply) )(WikiPageEditForm.apply)
get("/:owner/:repository/wiki")(referrersOnly { repository => get("/:owner/:repository/wiki")(referrersOnly { repository =>
getWikiPage(repository.owner, repository.name, "Home").map { page => getWikiPage(repository.owner, repository.name, "Home").map { page =>
html.page("Home", page, getWikiPageList(repository.owner, repository.name), html.page(
repository, isEditable(repository), "Home",
page,
getWikiPageList(repository.owner, repository.name),
repository,
isEditable(repository),
getWikiPage(repository.owner, repository.name, "_Sidebar"), getWikiPage(repository.owner, repository.name, "_Sidebar"),
getWikiPage(repository.owner, repository.name, "_Footer")) getWikiPage(repository.owner, repository.name, "_Footer")
)
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/Home/_edit") } getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/Home/_edit")
}) })
@@ -53,17 +75,22 @@ trait WikiControllerBase extends ControllerBase {
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
getWikiPage(repository.owner, repository.name, pageName).map { page => getWikiPage(repository.owner, repository.name, pageName).map { page =>
html.page(pageName, page, getWikiPageList(repository.owner, repository.name), html.page(
repository, isEditable(repository), pageName,
page,
getWikiPageList(repository.owner, repository.name),
repository,
isEditable(repository),
getWikiPage(repository.owner, repository.name, "_Sidebar"), getWikiPage(repository.owner, repository.name, "_Sidebar"),
getWikiPage(repository.owner, repository.name, "_Footer")) getWikiPage(repository.owner, repository.name, "_Footer")
)
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}/_edit") } getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}/_edit")
}) })
get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository => get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository =>
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))) { git =>
JGitUtil.getCommitLog(git, "master", path = pageName + ".md") match { JGitUtil.getCommitLog(git, "master", path = pageName + ".md") match {
case Right((logs, hasNext)) => html.history(Some(pageName), logs, repository, isEditable(repository)) case Right((logs, hasNext)) => html.history(Some(pageName), logs, repository, isEditable(repository))
case Left(_) => NotFound() case Left(_) => NotFound()
@@ -75,41 +102,57 @@ trait WikiControllerBase extends ControllerBase {
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
val Array(from, to) = params("commitId").split("\\.\\.\\.") val Array(from, to) = params("commitId").split("\\.\\.\\.")
using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))) { git =>
html.compare(Some(pageName), from, to, JGitUtil.getDiffs(git, Some(from), to, true, false).filter(_.newPath == pageName + ".md"), repository, html.compare(
isEditable(repository), flash.get("info")) Some(pageName),
from,
to,
JGitUtil.getDiffs(git, Some(from), to, true, false).filter(_.newPath == pageName + ".md"),
repository,
isEditable(repository),
flash.get("info")
)
} }
}) })
get("/:owner/:repository/wiki/_compare/:commitId")(referrersOnly { repository => get("/:owner/:repository/wiki/_compare/:commitId")(referrersOnly { repository =>
val Array(from, to) = params("commitId").split("\\.\\.\\.") val Array(from, to) = params("commitId").split("\\.\\.\\.")
using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))) { git =>
html.compare(None, from, to, JGitUtil.getDiffs(git, Some(from), to, true, false), repository, html.compare(
isEditable(repository), flash.get("info")) None,
from,
to,
JGitUtil.getDiffs(git, Some(from), to, true, false),
repository,
isEditable(repository),
flash.get("info")
)
} }
}) })
get("/:owner/:repository/wiki/:page/_revert/:commitId")(readableUsersOnly { repository => get("/:owner/:repository/wiki/:page/_revert/:commitId")(readableUsersOnly { repository =>
if(isEditable(repository)){ if (isEditable(repository)) {
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
val Array(from, to) = params("commitId").split("\\.\\.\\.") val Array(from, to) = params("commitId").split("\\.\\.\\.")
if(revertWikiPage(repository.owner, repository.name, from, to, context.loginAccount.get, Some(pageName))){ if (revertWikiPage(repository.owner, repository.name, from, to, context.loginAccount.get, Some(pageName))) {
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}") redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}")
} else { } else {
flash += "info" -> "This patch was not able to be reversed." flash += "info" -> "This patch was not able to be reversed."
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}/_compare/${from}...${to}") redirect(
s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(pageName)}/_compare/${from}...${to}"
)
} }
} else Unauthorized() } else Unauthorized()
}) })
get("/:owner/:repository/wiki/_revert/:commitId")(readableUsersOnly { repository => get("/:owner/:repository/wiki/_revert/:commitId")(readableUsersOnly { repository =>
if(isEditable(repository)){ if (isEditable(repository)) {
val Array(from, to) = params("commitId").split("\\.\\.\\.") val Array(from, to) = params("commitId").split("\\.\\.\\.")
if(revertWikiPage(repository.owner, repository.name, from, to, context.loginAccount.get, None)){ if (revertWikiPage(repository.owner, repository.name, from, to, context.loginAccount.get, None)) {
redirect(s"/${repository.owner}/${repository.name}/wiki/") redirect(s"/${repository.owner}/${repository.name}/wiki")
} else { } else {
flash += "info" -> "This patch was not able to be reversed." flash += "info" -> "This patch was not able to be reversed."
redirect(s"/${repository.owner}/${repository.name}/wiki/_compare/${from}...${to}") redirect(s"/${repository.owner}/${repository.name}/wiki/_compare/${from}...${to}")
@@ -118,15 +161,16 @@ trait WikiControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/wiki/:page/_edit")(readableUsersOnly { repository => get("/:owner/:repository/wiki/:page/_edit")(readableUsersOnly { repository =>
if(isEditable(repository)){ if (isEditable(repository)) {
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
html.edit(pageName, getWikiPage(repository.owner, repository.name, pageName), repository) html.edit(pageName, getWikiPage(repository.owner, repository.name, pageName), repository)
} else Unauthorized() } else Unauthorized()
}) })
post("/:owner/:repository/wiki/_edit", editForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/wiki/_edit", editForm)(readableUsersOnly { (form, repository) =>
if(isEditable(repository)){ if (isEditable(repository)) {
defining(context.loginAccount.get){ loginAccount => defining(context.loginAccount.get) {
loginAccount =>
saveWikiPage( saveWikiPage(
repository.owner, repository.owner,
repository.name, repository.name,
@@ -136,16 +180,23 @@ trait WikiControllerBase extends ControllerBase {
loginAccount, loginAccount,
form.message.getOrElse(""), form.message.getOrElse(""),
Some(form.id) Some(form.id)
).map { commitId => ).foreach {
commitId =>
updateLastActivityDate(repository.owner, repository.name) updateLastActivityDate(repository.owner, repository.name)
recordEditWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName, commitId) recordEditWikiPageActivity(
callWebHookOf(repository.owner, repository.name, WebHook.Gollum){ repository.owner,
repository.name,
loginAccount.userName,
form.pageName,
commitId
)
callWebHookOf(repository.owner, repository.name, WebHook.Gollum) {
getAccountByUserName(repository.owner).map { repositoryUser => getAccountByUserName(repository.owner).map { repositoryUser =>
WebHookGollumPayload("edited", form.pageName, commitId, repository, repositoryUser, loginAccount) WebHookGollumPayload("edited", form.pageName, commitId, repository, repositoryUser, loginAccount)
} }
} }
} }
if(notReservedPageName(form.pageName)) { if (notReservedPageName(form.pageName)) {
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}") redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}")
} else { } else {
redirect(s"/${repository.owner}/${repository.name}/wiki") redirect(s"/${repository.owner}/${repository.name}/wiki")
@@ -155,14 +206,15 @@ trait WikiControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/wiki/_new")(readableUsersOnly { repository => get("/:owner/:repository/wiki/_new")(readableUsersOnly { repository =>
if(isEditable(repository)){ if (isEditable(repository)) {
html.edit("", None, repository) html.edit("", None, repository)
} else Unauthorized() } else Unauthorized()
}) })
post("/:owner/:repository/wiki/_new", newForm)(readableUsersOnly { (form, repository) => post("/:owner/:repository/wiki/_new", newForm)(readableUsersOnly { (form, repository) =>
if(isEditable(repository)){ if (isEditable(repository)) {
defining(context.loginAccount.get){ loginAccount => defining(context.loginAccount.get) {
loginAccount =>
saveWikiPage( saveWikiPage(
repository.owner, repository.owner,
repository.name, repository.name,
@@ -172,17 +224,18 @@ trait WikiControllerBase extends ControllerBase {
loginAccount, loginAccount,
form.message.getOrElse(""), form.message.getOrElse(""),
None None
).map { commitId => ).foreach {
commitId =>
updateLastActivityDate(repository.owner, repository.name) updateLastActivityDate(repository.owner, repository.name)
recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName) recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
callWebHookOf(repository.owner, repository.name, WebHook.Gollum){ callWebHookOf(repository.owner, repository.name, WebHook.Gollum) {
getAccountByUserName(repository.owner).map { repositoryUser => getAccountByUserName(repository.owner).map { repositoryUser =>
WebHookGollumPayload("created", form.pageName, commitId, repository, repositoryUser, loginAccount) WebHookGollumPayload("created", form.pageName, commitId, repository, repositoryUser, loginAccount)
} }
} }
} }
if(notReservedPageName(form.pageName)) { if (notReservedPageName(form.pageName)) {
redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}") redirect(s"/${repository.owner}/${repository.name}/wiki/${StringUtil.urlEncode(form.pageName)}")
} else { } else {
redirect(s"/${repository.owner}/${repository.name}/wiki") redirect(s"/${repository.owner}/${repository.name}/wiki")
@@ -192,11 +245,18 @@ trait WikiControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/wiki/:page/_delete")(readableUsersOnly { repository => get("/:owner/:repository/wiki/:page/_delete")(readableUsersOnly { repository =>
if(isEditable(repository)){ if (isEditable(repository)) {
val pageName = StringUtil.urlDecode(params("page")) val pageName = StringUtil.urlDecode(params("page"))
defining(context.loginAccount.get){ loginAccount => defining(context.loginAccount.get) { loginAccount =>
deleteWikiPage(repository.owner, repository.name, pageName, loginAccount.fullName, loginAccount.mailAddress, s"Destroyed ${pageName}") deleteWikiPage(
repository.owner,
repository.name,
pageName,
loginAccount.fullName,
loginAccount.mailAddress,
s"Destroyed ${pageName}"
)
updateLastActivityDate(repository.owner, repository.name) updateLastActivityDate(repository.owner, repository.name)
redirect(s"/${repository.owner}/${repository.name}/wiki") redirect(s"/${repository.owner}/${repository.name}/wiki")
@@ -209,7 +269,7 @@ trait WikiControllerBase extends ControllerBase {
}) })
get("/:owner/:repository/wiki/_history")(referrersOnly { repository => get("/:owner/:repository/wiki/_history")(referrersOnly { repository =>
using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))) { git =>
JGitUtil.getCommitLog(git, "master") match { JGitUtil.getCommitLog(git, "master") match {
case Right((logs, hasNext)) => html.history(None, logs, repository, isEditable(repository)) case Right((logs, hasNext)) => html.history(None, logs, repository, isEditable(repository))
case Left(_) => NotFound() case Left(_) => NotFound()
@@ -219,7 +279,7 @@ trait WikiControllerBase extends ControllerBase {
get("/:owner/:repository/wiki/_blob/*")(referrersOnly { repository => get("/:owner/:repository/wiki/_blob/*")(referrersOnly { repository =>
val path = multiParams("splat").head val path = multiParams("splat").head
using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))){ git => using(Git.open(getWikiRepositoryDir(repository.owner, repository.name))) { git =>
val revCommit = JGitUtil.getRevCommitFromId(git, git.getRepository.resolve("master")) val revCommit = JGitUtil.getRevCommitFromId(git, git.getRepository.resolve("master"))
getPathObjectId(git, path, revCommit).map { objectId => getPathObjectId(git, path, revCommit).map { objectId =>
@@ -228,25 +288,32 @@ trait WikiControllerBase extends ControllerBase {
} }
}) })
private def unique: Constraint = new Constraint(){ private def unique: Constraint = new Constraint() {
override def validate(name: String, value: String, params: Map[String, Seq[String]], messages: Messages): Option[String] = override def validate(
getWikiPageList(params.value("owner"), params.value("repository")).find(_ == value).map(_ => "Page already exists.") name: String,
value: String,
params: Map[String, Seq[String]],
messages: Messages
): Option[String] =
getWikiPageList(params.value("owner"), params.value("repository"))
.find(_ == value)
.map(_ => "Page already exists.")
} }
private def pagename: Constraint = new Constraint(){ private def pagename: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = override def validate(name: String, value: String, messages: Messages): Option[String] =
if(value.exists("\\/:*?\"<>|".contains(_))){ if (value.exists("\\/:*?\"<>|".contains(_))) {
Some(s"${name} contains invalid character.") Some(s"${name} contains invalid character.")
} else if(notReservedPageName(value) && (value.startsWith("_") || value.startsWith("-"))){ } else if (notReservedPageName(value) && (value.startsWith("_") || value.startsWith("-"))) {
Some(s"${name} starts with invalid character.") Some(s"${name} starts with invalid character.")
} else { } else {
None None
} }
} }
private def notReservedPageName(value: String) = ! (Array[String]("_Sidebar","_Footer") contains value) private def notReservedPageName(value: String) = !(Array[String]("_Sidebar", "_Footer") contains value)
private def conflictForNew: Constraint = new Constraint(){ private def conflictForNew: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
targetWikiPage.map { _ => targetWikiPage.map { _ =>
"Someone has created the wiki since you started. Please reload this page and re-apply your changes." "Someone has created the wiki since you started. Please reload this page and re-apply your changes."
@@ -254,9 +321,9 @@ trait WikiControllerBase extends ControllerBase {
} }
} }
private def conflictForEdit: Constraint = new Constraint(){ private def conflictForEdit: Constraint = new Constraint() {
override def validate(name: String, value: String, messages: Messages): Option[String] = { override def validate(name: String, value: String, messages: Messages): Option[String] = {
targetWikiPage.filter(_.id != params("id")).map{ _ => targetWikiPage.filter(_.id != params("id")).map { _ =>
"Someone has edited the wiki since you started. Please reload this page and re-apply your changes." "Someone has edited the wiki since you started. Please reload this page and re-apply your changes."
} }
} }

View File

@@ -1,6 +1,5 @@
package gitbucket.core.model package gitbucket.core.model
trait AccessTokenComponent { self: Profile => trait AccessTokenComponent { self: Profile =>
import profile.api._ import profile.api._

View File

@@ -20,7 +20,22 @@ trait AccountComponent { self: Profile =>
val groupAccount = column[Boolean]("GROUP_ACCOUNT") val groupAccount = column[Boolean]("GROUP_ACCOUNT")
val removed = column[Boolean]("REMOVED") val removed = column[Boolean]("REMOVED")
val description = column[String]("DESCRIPTION") val description = column[String]("DESCRIPTION")
def * = (userName, fullName, mailAddress, password, isAdmin, url.?, registeredDate, updatedDate, lastLoginDate.?, image.?, groupAccount, removed, description.?) <> (Account.tupled, Account.unapply) def * =
(
userName,
fullName,
mailAddress,
password,
isAdmin,
url.?,
registeredDate,
updatedDate,
lastLoginDate.?,
image.?,
groupAccount,
removed,
description.?
) <> (Account.tupled, Account.unapply)
} }
} }

View File

@@ -0,0 +1,19 @@
package gitbucket.core.model
trait AccountExtraMailAddressComponent { self: Profile =>
import profile.api._
lazy val AccountExtraMailAddresses = TableQuery[AccountExtraMailAddresses]
class AccountExtraMailAddresses(tag: Tag) extends Table[AccountExtraMailAddress](tag, "ACCOUNT_EXTRA_MAIL_ADDRESS") {
val userName = column[String]("USER_NAME", O PrimaryKey)
val extraMailAddress = column[String]("EXTRA_MAIL_ADDRESS", O PrimaryKey)
def * =
(userName, extraMailAddress) <> (AccountExtraMailAddress.tupled, AccountExtraMailAddress.unapply)
}
}
case class AccountExtraMailAddress(
userName: String,
extraMailAddress: String
)

View File

@@ -3,7 +3,8 @@ package gitbucket.core.model
trait AccountWebHookComponent extends TemplateComponent { self: Profile => trait AccountWebHookComponent extends TemplateComponent { self: Profile =>
import profile.api._ import profile.api._
private implicit val whContentTypeColumnType = MappedColumnType.base[WebHookContentType, String](whct => whct.code , code => WebHookContentType.valueOf(code)) private implicit val whContentTypeColumnType =
MappedColumnType.base[WebHookContentType, String](whct => whct.code, code => WebHookContentType.valueOf(code))
lazy val AccountWebHooks = TableQuery[AccountWebHooks] lazy val AccountWebHooks = TableQuery[AccountWebHooks]

View File

@@ -8,7 +8,9 @@ trait AccountWebHookEventComponent extends TemplateComponent {
lazy val AccountWebHookEvents = TableQuery[AccountWebHookEvents] lazy val AccountWebHookEvents = TableQuery[AccountWebHookEvents]
class AccountWebHookEvents(tag: Tag) extends Table[AccountWebHookEvent](tag, "ACCOUNT_WEB_HOOK_EVENT") with BasicTemplate { class AccountWebHookEvents(tag: Tag)
extends Table[AccountWebHookEvent](tag, "ACCOUNT_WEB_HOOK_EVENT")
with BasicTemplate {
val url = column[String]("URL") val url = column[String]("URL")
val event = column[WebHook.Event]("EVENT") val event = column[WebHook.Event]("EVENT")
@@ -31,4 +33,4 @@ case class AccountWebHookEvent(
userName: String, userName: String,
url: String, url: String,
event: WebHook.Event event: WebHook.Event
) )

View File

@@ -13,7 +13,8 @@ trait ActivityComponent extends TemplateComponent { self: Profile =>
val message = column[String]("MESSAGE") val message = column[String]("MESSAGE")
val additionalInfo = column[String]("ADDITIONAL_INFO") val additionalInfo = column[String]("ADDITIONAL_INFO")
val activityDate = column[java.util.Date]("ACTIVITY_DATE") val activityDate = column[java.util.Date]("ACTIVITY_DATE")
def * = (userName, repositoryName, activityUserName, activityType, message, additionalInfo.?, activityDate, activityId) <> (Activity.tupled, Activity.unapply) def * =
(userName, repositoryName, activityUserName, activityType, message, additionalInfo.?, activityDate, activityId) <> (Activity.tupled, Activity.unapply)
} }
} }

View File

@@ -76,9 +76,11 @@ protected[model] trait TemplateComponent { self: Profile =>
byRepository(userName, repositoryName) && (this.commitId === commitId) byRepository(userName, repositoryName) && (this.commitId === commitId)
} }
trait BranchTemplate extends BasicTemplate{ self: Table[_] => trait BranchTemplate extends BasicTemplate { self: Table[_] =>
val branch = column[String]("BRANCH") val branch = column[String]("BRANCH")
def byBranch(owner: String, repository: String, branchName: String) = byRepository(owner, repository) && (branch === branchName.bind) def byBranch(owner: String, repository: String, branchName: String) =
def byBranch(owner: Rep[String], repository: Rep[String], branchName: Rep[String]) = byRepository(owner, repository) && (this.branch === branchName) byRepository(owner, repository) && (branch === branchName.bind)
def byBranch(owner: Rep[String], repository: Rep[String], branchName: Rep[String]) =
byRepository(owner, repository) && (this.branch === branchName)
} }
} }

View File

@@ -1,6 +1,7 @@
package gitbucket.core.model package gitbucket.core.model
import java.util.Date
trait Comment { sealed trait Comment {
val commentedUserName: String val commentedUserName: String
val registeredDate: java.util.Date val registeredDate: java.util.Date
} }
@@ -18,13 +19,14 @@ trait IssueCommentComponent extends TemplateComponent { self: Profile =>
val content = column[String]("CONTENT") val content = column[String]("CONTENT")
val registeredDate = column[java.util.Date]("REGISTERED_DATE") val registeredDate = column[java.util.Date]("REGISTERED_DATE")
val updatedDate = column[java.util.Date]("UPDATED_DATE") val updatedDate = column[java.util.Date]("UPDATED_DATE")
def * = (userName, repositoryName, issueId, commentId, action, commentedUserName, content, registeredDate, updatedDate) <> (IssueComment.tupled, IssueComment.unapply) def * =
(userName, repositoryName, issueId, commentId, action, commentedUserName, content, registeredDate, updatedDate) <> (IssueComment.tupled, IssueComment.unapply)
def byPrimaryKey(commentId: Int) = this.commentId === commentId.bind def byPrimaryKey(commentId: Int) = this.commentId === commentId.bind
} }
} }
case class IssueComment ( case class IssueComment(
userName: String, userName: String,
repositoryName: String, repositoryName: String,
issueId: Int, issueId: Int,
@@ -52,7 +54,27 @@ trait CommitCommentComponent extends TemplateComponent { self: Profile =>
val registeredDate = column[java.util.Date]("REGISTERED_DATE") val registeredDate = column[java.util.Date]("REGISTERED_DATE")
val updatedDate = column[java.util.Date]("UPDATED_DATE") val updatedDate = column[java.util.Date]("UPDATED_DATE")
val issueId = column[Option[Int]]("ISSUE_ID") val issueId = column[Option[Int]]("ISSUE_ID")
def * = (userName, repositoryName, commitId, commentId, commentedUserName, content, fileName, oldLine, newLine, registeredDate, updatedDate, issueId) <> (CommitComment.tupled, CommitComment.unapply) val originalCommitId = column[String]("ORIGINAL_COMMIT_ID")
val originalOldLine = column[Option[Int]]("ORIGINAL_OLD_LINE")
val originalNewLine = column[Option[Int]]("ORIGINAL_NEW_LINE")
def * =
(
userName,
repositoryName,
commitId,
commentId,
commentedUserName,
content,
fileName,
oldLine,
newLine,
registeredDate,
updatedDate,
issueId,
originalCommitId,
originalOldLine,
originalNewLine
) <> (CommitComment.tupled, CommitComment.unapply)
def byPrimaryKey(commentId: Int) = this.commentId === commentId.bind def byPrimaryKey(commentId: Int) = this.commentId === commentId.bind
} }
@@ -70,5 +92,16 @@ case class CommitComment(
newLine: Option[Int], newLine: Option[Int],
registeredDate: java.util.Date, registeredDate: java.util.Date,
updatedDate: java.util.Date, updatedDate: java.util.Date,
issueId: Option[Int] issueId: Option[Int],
) extends Comment originalCommitId: String,
originalOldLine: Option[Int],
originalNewLine: Option[Int]
) extends Comment
case class CommitComments(
fileName: String,
commentedUserName: String,
registeredDate: Date,
comments: Seq[CommitComment],
diff: Option[String]
) extends Comment

View File

@@ -4,7 +4,7 @@ trait CommitStatusComponent extends TemplateComponent { self: Profile =>
import profile.api._ import profile.api._
import self._ import self._
implicit val commitStateColumnType = MappedColumnType.base[CommitState, String](b => b.name , i => CommitState(i)) implicit val commitStateColumnType = MappedColumnType.base[CommitState, String](b => b.name, i => CommitState(i))
lazy val CommitStatuses = TableQuery[CommitStatuses] lazy val CommitStatuses = TableQuery[CommitStatuses]
class CommitStatuses(tag: Tag) extends Table[CommitStatus](tag, "COMMIT_STATUS") with CommitTemplate { class CommitStatuses(tag: Tag) extends Table[CommitStatus](tag, "COMMIT_STATUS") with CommitTemplate {
@@ -16,12 +16,24 @@ trait CommitStatusComponent extends TemplateComponent { self: Profile =>
val creator = column[String]("CREATOR") val creator = column[String]("CREATOR")
val registeredDate = column[java.util.Date]("REGISTERED_DATE") val registeredDate = column[java.util.Date]("REGISTERED_DATE")
val updatedDate = column[java.util.Date]("UPDATED_DATE") val updatedDate = column[java.util.Date]("UPDATED_DATE")
def * = (commitStatusId, userName, repositoryName, commitId, context, state, targetUrl, description, creator, registeredDate, updatedDate) <> ((CommitStatus.apply _).tupled, CommitStatus.unapply) def * =
(
commitStatusId,
userName,
repositoryName,
commitId,
context,
state,
targetUrl,
description,
creator,
registeredDate,
updatedDate
) <> ((CommitStatus.apply _).tupled, CommitStatus.unapply)
def byPrimaryKey(id: Int) = commitStatusId === id.bind def byPrimaryKey(id: Int) = commitStatusId === id.bind
} }
} }
case class CommitStatus( case class CommitStatus(
commitStatusId: Int = 0, commitStatusId: Int = 0,
userName: String, userName: String,
@@ -36,7 +48,8 @@ case class CommitStatus(
updatedDate: java.util.Date updatedDate: java.util.Date
) )
object CommitStatus { object CommitStatus {
def pending(owner: String, repository: String, context: String) = CommitStatus( def pending(owner: String, repository: String, context: String) =
CommitStatus(
commitStatusId = 0, commitStatusId = 0,
userName = owner, userName = owner,
repositoryName = repository, repositoryName = repository,
@@ -47,12 +60,12 @@ object CommitStatus {
description = Some("Waiting for status to be reported"), description = Some("Waiting for status to be reported"),
creator = "", creator = "",
registeredDate = new java.util.Date(), registeredDate = new java.util.Date(),
updatedDate = new java.util.Date()) updatedDate = new java.util.Date()
)
} }
sealed abstract class CommitState(val name: String) sealed abstract class CommitState(val name: String)
object CommitState { object CommitState {
object ERROR extends CommitState("error") object ERROR extends CommitState("error")
@@ -76,11 +89,11 @@ object CommitState {
* success if the latest status for all contexts is success * success if the latest status for all contexts is success
*/ */
def combine(statuses: Set[CommitState]): CommitState = { def combine(statuses: Set[CommitState]): CommitState = {
if(statuses.isEmpty){ if (statuses.isEmpty) {
PENDING PENDING
} else if(statuses.contains(CommitState.ERROR) || statuses.contains(CommitState.FAILURE)) { } else if (statuses.contains(CommitState.ERROR) || statuses.contains(CommitState.FAILURE)) {
FAILURE FAILURE
} else if(statuses.contains(CommitState.PENDING)) { } else if (statuses.contains(CommitState.PENDING)) {
PENDING PENDING
} else { } else {
SUCCESS SUCCESS
@@ -88,4 +101,3 @@ object CommitState {
} }
} }

View File

@@ -10,7 +10,8 @@ trait DeployKeyComponent extends TemplateComponent { self: Profile =>
val title = column[String]("TITLE") val title = column[String]("TITLE")
val publicKey = column[String]("PUBLIC_KEY") val publicKey = column[String]("PUBLIC_KEY")
val allowWrite = column[Boolean]("ALLOW_WRITE") val allowWrite = column[Boolean]("ALLOW_WRITE")
def * = (userName, repositoryName, deployKeyId, title, publicKey, allowWrite) <> (DeployKey.tupled, DeployKey.unapply) def * =
(userName, repositoryName, deployKeyId, title, publicKey, allowWrite) <> (DeployKey.tupled, DeployKey.unapply)
def byPrimaryKey(userName: String, repositoryName: String, deployKeyId: Int) = def byPrimaryKey(userName: String, repositoryName: String, deployKeyId: Int) =
(this.userName === userName.bind) && (this.repositoryName === repositoryName.bind) && (this.deployKeyId === deployKeyId.bind) (this.userName === userName.bind) && (this.repositoryName === repositoryName.bind) && (this.deployKeyId === deployKeyId.bind)

View File

@@ -13,13 +13,19 @@ trait IssueComponent extends TemplateComponent { self: Profile =>
def byPrimaryKey(owner: String, repository: String) = byRepository(owner, repository) def byPrimaryKey(owner: String, repository: String) = byRepository(owner, repository)
} }
class IssueOutline(tag: Tag) extends Table[(String, String, Int, Int, Int)](tag, "ISSUE_OUTLINE_VIEW") with IssueTemplate { class IssueOutline(tag: Tag)
extends Table[(String, String, Int, Int, Int)](tag, "ISSUE_OUTLINE_VIEW")
with IssueTemplate {
val commentCount = column[Int]("COMMENT_COUNT") val commentCount = column[Int]("COMMENT_COUNT")
val priority = column[Int]("PRIORITY") val priority = column[Int]("PRIORITY")
def * = (userName, repositoryName, issueId, commentCount, priority) def * = (userName, repositoryName, issueId, commentCount, priority)
} }
class Issues(tag: Tag) extends Table[Issue](tag, "ISSUE") with IssueTemplate with MilestoneTemplate with PriorityTemplate { class Issues(tag: Tag)
extends Table[Issue](tag, "ISSUE")
with IssueTemplate
with MilestoneTemplate
with PriorityTemplate {
val openedUserName = column[String]("OPENED_USER_NAME") val openedUserName = column[String]("OPENED_USER_NAME")
val assignedUserName = column[String]("ASSIGNED_USER_NAME") val assignedUserName = column[String]("ASSIGNED_USER_NAME")
val title = column[String]("TITLE") val title = column[String]("TITLE")
@@ -28,7 +34,22 @@ trait IssueComponent extends TemplateComponent { self: Profile =>
val registeredDate = column[java.util.Date]("REGISTERED_DATE") val registeredDate = column[java.util.Date]("REGISTERED_DATE")
val updatedDate = column[java.util.Date]("UPDATED_DATE") val updatedDate = column[java.util.Date]("UPDATED_DATE")
val pullRequest = column[Boolean]("PULL_REQUEST") val pullRequest = column[Boolean]("PULL_REQUEST")
def * = (userName, repositoryName, issueId, openedUserName, milestoneId.?, priorityId.?, assignedUserName.?, title, content.?, closed, registeredDate, updatedDate, pullRequest) <> (Issue.tupled, Issue.unapply) def * =
(
userName,
repositoryName,
issueId,
openedUserName,
milestoneId.?,
priorityId.?,
assignedUserName.?,
title,
content.?,
closed,
registeredDate,
updatedDate,
pullRequest
) <> (Issue.tupled, Issue.unapply)
def byPrimaryKey(owner: String, repository: String, issueId: Int) = byIssue(owner, repository, issueId) def byPrimaryKey(owner: String, repository: String, issueId: Int) = byIssue(owner, repository, issueId)
} }

View File

@@ -12,23 +12,19 @@ trait LabelComponent extends TemplateComponent { self: Profile =>
def * = (userName, repositoryName, labelId, labelName, color) <> (Label.tupled, Label.unapply) def * = (userName, repositoryName, labelId, labelName, color) <> (Label.tupled, Label.unapply)
def byPrimaryKey(owner: String, repository: String, labelId: Int) = byLabel(owner, repository, labelId) def byPrimaryKey(owner: String, repository: String, labelId: Int) = byLabel(owner, repository, labelId)
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], labelId: Rep[Int]) = byLabel(userName, repositoryName, labelId) def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], labelId: Rep[Int]) =
byLabel(userName, repositoryName, labelId)
} }
} }
case class Label( case class Label(userName: String, repositoryName: String, labelId: Int = 0, labelName: String, color: String) {
userName: String,
repositoryName: String,
labelId: Int = 0,
labelName: String,
color: String){
val fontColor = { val fontColor = {
val r = color.substring(0, 2) val r = color.substring(0, 2)
val g = color.substring(2, 4) val g = color.substring(2, 4)
val b = color.substring(4, 6) val b = color.substring(4, 6)
if(Integer.parseInt(r, 16) + Integer.parseInt(g, 16) + Integer.parseInt(b, 16) > 408){ if (Integer.parseInt(r, 16) + Integer.parseInt(g, 16) + Integer.parseInt(b, 16) > 408) {
"000000" "000000"
} else { } else {
"ffffff" "ffffff"

View File

@@ -12,10 +12,12 @@ trait MilestoneComponent extends TemplateComponent { self: Profile =>
val description = column[Option[String]]("DESCRIPTION") val description = column[Option[String]]("DESCRIPTION")
val dueDate = column[Option[java.util.Date]]("DUE_DATE") val dueDate = column[Option[java.util.Date]]("DUE_DATE")
val closedDate = column[Option[java.util.Date]]("CLOSED_DATE") val closedDate = column[Option[java.util.Date]]("CLOSED_DATE")
def * = (userName, repositoryName, milestoneId, title, description, dueDate, closedDate) <> (Milestone.tupled, Milestone.unapply) def * =
(userName, repositoryName, milestoneId, title, description, dueDate, closedDate) <> (Milestone.tupled, Milestone.unapply)
def byPrimaryKey(owner: String, repository: String, milestoneId: Int) = byMilestone(owner, repository, milestoneId) def byPrimaryKey(owner: String, repository: String, milestoneId: Int) = byMilestone(owner, repository, milestoneId)
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], milestoneId: Rep[Int]) = byMilestone(userName, repositoryName, milestoneId) def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], milestoneId: Rep[Int]) =
byMilestone(userName, repositoryName, milestoneId)
} }
} }

View File

@@ -12,14 +12,16 @@ trait PriorityComponent extends TemplateComponent { self: Profile =>
val ordering = column[Int]("ORDERING") val ordering = column[Int]("ORDERING")
val isDefault = column[Boolean]("IS_DEFAULT") val isDefault = column[Boolean]("IS_DEFAULT")
val color = column[String]("COLOR") val color = column[String]("COLOR")
def * = (userName, repositoryName, priorityId, priorityName, description.?, isDefault, ordering, color) <> (Priority.tupled, Priority.unapply) def * =
(userName, repositoryName, priorityId, priorityName, description.?, isDefault, ordering, color) <> (Priority.tupled, Priority.unapply)
def byPrimaryKey(owner: String, repository: String, priorityId: Int) = byPriority(owner, repository, priorityId) def byPrimaryKey(owner: String, repository: String, priorityId: Int) = byPriority(owner, repository, priorityId)
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], priorityId: Rep[Int]) = byPriority(userName, repositoryName, priorityId) def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], priorityId: Rep[Int]) =
byPriority(userName, repositoryName, priorityId)
} }
} }
case class Priority ( case class Priority(
userName: String, userName: String,
repositoryName: String, repositoryName: String,
priorityId: Int = 0, priorityId: Int = 0,
@@ -27,14 +29,15 @@ case class Priority (
description: Option[String], description: Option[String],
isDefault: Boolean, isDefault: Boolean,
ordering: Int = 0, ordering: Int = 0,
color: String){ color: String
) {
val fontColor = { val fontColor = {
val r = color.substring(0, 2) val r = color.substring(0, 2)
val g = color.substring(2, 4) val g = color.substring(2, 4)
val b = color.substring(4, 6) val b = color.substring(4, 6)
if(Integer.parseInt(r, 16) + Integer.parseInt(g, 16) + Integer.parseInt(b, 16) > 408){ if (Integer.parseInt(r, 16) + Integer.parseInt(g, 16) + Integer.parseInt(b, 16) > 408) {
"000000" "000000"
} else { } else {
"ffffff" "ffffff"

View File

@@ -23,8 +23,8 @@ trait Profile {
/** /**
* Extends Column to add conditional condition * Extends Column to add conditional condition
*/ */
implicit class RichColumn(c1: Rep[Boolean]){ implicit class RichColumn(c1: Rep[Boolean]) {
def &&(c2: => Rep[Boolean], guard: => Boolean): Rep[Boolean] = if(guard) c1 && c2 else c1 def &&(c2: => Rep[Boolean], guard: => Boolean): Rep[Boolean] = if (guard) c1 && c2 else c1
} }
/** /**
@@ -40,7 +40,9 @@ trait ProfileProvider { self: Profile =>
} }
trait CoreProfile extends ProfileProvider with Profile trait CoreProfile
extends ProfileProvider
with Profile
with AccessTokenComponent with AccessTokenComponent
with AccountComponent with AccountComponent
with ActivityComponent with ActivityComponent
@@ -64,7 +66,8 @@ trait CoreProfile extends ProfileProvider with Profile
with AccountFederationComponent with AccountFederationComponent
with ProtectedBranchComponent with ProtectedBranchComponent
with DeployKeyComponent with DeployKeyComponent
with ReleaseComponent with ReleaseTagComponent
with ReleaseAssetComponent with ReleaseAssetComponent
with AccountExtraMailAddressComponent
object Profile extends CoreProfile object Profile extends CoreProfile

View File

@@ -8,27 +8,22 @@ trait ProtectedBranchComponent extends TemplateComponent { self: Profile =>
class ProtectedBranches(tag: Tag) extends Table[ProtectedBranch](tag, "PROTECTED_BRANCH") with BranchTemplate { class ProtectedBranches(tag: Tag) extends Table[ProtectedBranch](tag, "PROTECTED_BRANCH") with BranchTemplate {
val statusCheckAdmin = column[Boolean]("STATUS_CHECK_ADMIN") val statusCheckAdmin = column[Boolean]("STATUS_CHECK_ADMIN")
def * = (userName, repositoryName, branch, statusCheckAdmin) <> (ProtectedBranch.tupled, ProtectedBranch.unapply) def * = (userName, repositoryName, branch, statusCheckAdmin) <> (ProtectedBranch.tupled, ProtectedBranch.unapply)
def byPrimaryKey(userName: String, repositoryName: String, branch: String) = byBranch(userName, repositoryName, branch) def byPrimaryKey(userName: String, repositoryName: String, branch: String) =
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], branch: Rep[String]) = byBranch(userName, repositoryName, branch) byBranch(userName, repositoryName, branch)
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], branch: Rep[String]) =
byBranch(userName, repositoryName, branch)
} }
lazy val ProtectedBranchContexts = TableQuery[ProtectedBranchContexts] lazy val ProtectedBranchContexts = TableQuery[ProtectedBranchContexts]
class ProtectedBranchContexts(tag: Tag) extends Table[ProtectedBranchContext](tag, "PROTECTED_BRANCH_REQUIRE_CONTEXT") with BranchTemplate { class ProtectedBranchContexts(tag: Tag)
extends Table[ProtectedBranchContext](tag, "PROTECTED_BRANCH_REQUIRE_CONTEXT")
with BranchTemplate {
val context = column[String]("CONTEXT") val context = column[String]("CONTEXT")
def * = (userName, repositoryName, branch, context) <> (ProtectedBranchContext.tupled, ProtectedBranchContext.unapply) def * =
(userName, repositoryName, branch, context) <> (ProtectedBranchContext.tupled, ProtectedBranchContext.unapply)
} }
} }
case class ProtectedBranch(userName: String, repositoryName: String, branch: String, statusCheckAdmin: Boolean)
case class ProtectedBranch( case class ProtectedBranchContext(userName: String, repositoryName: String, branch: String, context: String)
userName: String,
repositoryName: String,
branch: String,
statusCheckAdmin: Boolean)
case class ProtectedBranchContext(
userName: String,
repositoryName: String,
branch: String,
context: String)

View File

@@ -12,10 +12,23 @@ trait PullRequestComponent extends TemplateComponent { self: Profile =>
val requestBranch = column[String]("REQUEST_BRANCH") val requestBranch = column[String]("REQUEST_BRANCH")
val commitIdFrom = column[String]("COMMIT_ID_FROM") val commitIdFrom = column[String]("COMMIT_ID_FROM")
val commitIdTo = column[String]("COMMIT_ID_TO") val commitIdTo = column[String]("COMMIT_ID_TO")
def * = (userName, repositoryName, issueId, branch, requestUserName, requestRepositoryName, requestBranch, commitIdFrom, commitIdTo) <> (PullRequest.tupled, PullRequest.unapply) def * =
(
userName,
repositoryName,
issueId,
branch,
requestUserName,
requestRepositoryName,
requestBranch,
commitIdFrom,
commitIdTo
) <> (PullRequest.tupled, PullRequest.unapply)
def byPrimaryKey(userName: String, repositoryName: String, issueId: Int) = byIssue(userName, repositoryName, issueId) def byPrimaryKey(userName: String, repositoryName: String, issueId: Int) =
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], issueId: Rep[Int]) = byIssue(userName, repositoryName, issueId) byIssue(userName, repositoryName, issueId)
def byPrimaryKey(userName: Rep[String], repositoryName: Rep[String], issueId: Rep[Int]) =
byIssue(userName, repositoryName, issueId)
} }
} }

View File

@@ -20,9 +20,12 @@ trait ReleaseAssetComponent extends TemplateComponent {
val registeredDate = column[Date]("REGISTERED_DATE") val registeredDate = column[Date]("REGISTERED_DATE")
val updatedDate = column[Date]("UPDATED_DATE") val updatedDate = column[Date]("UPDATED_DATE")
def * = (userName, repositoryName, tag, releaseAssetId, fileName, label, size, uploader, registeredDate, updatedDate) <> (ReleaseAsset.tupled, ReleaseAsset.unapply) def * =
def byPrimaryKey(owner: String, repository: String, tag: String, fileName: String) = byTag(owner, repository, tag) && (this.fileName === fileName.bind) (userName, repositoryName, tag, releaseAssetId, fileName, label, size, uploader, registeredDate, updatedDate) <> (ReleaseAsset.tupled, ReleaseAsset.unapply)
def byTag(owner: String, repository: String, tag: String) = byRepository(owner, repository) && (this.tag === tag.bind) def byPrimaryKey(owner: String, repository: String, tag: String, fileName: String) =
byTag(owner, repository, tag) && (this.fileName === fileName.bind)
def byTag(owner: String, repository: String, tag: String) =
byRepository(owner, repository) && (this.tag === tag.bind)
} }
} }

View File

@@ -1,14 +1,14 @@
package gitbucket.core.model package gitbucket.core.model
trait ReleaseComponent extends TemplateComponent { trait ReleaseTagComponent extends TemplateComponent {
self: Profile => self: Profile =>
import profile.api._ import profile.api._
import self._ import self._
lazy val Releases = TableQuery[Releases] lazy val ReleaseTags = TableQuery[ReleaseTags]
class Releases(tag_ : Tag) extends Table[Release](tag_, "RELEASE") with BasicTemplate { class ReleaseTags(tag_ : Tag) extends Table[ReleaseTag](tag_, "RELEASE_TAG") with BasicTemplate {
val name = column[String]("NAME") val name = column[String]("NAME")
val tag = column[String]("TAG") val tag = column[String]("TAG")
val author = column[String]("AUTHOR") val author = column[String]("AUTHOR")
@@ -16,13 +16,15 @@ trait ReleaseComponent extends TemplateComponent {
val registeredDate = column[java.util.Date]("REGISTERED_DATE") val registeredDate = column[java.util.Date]("REGISTERED_DATE")
val updatedDate = column[java.util.Date]("UPDATED_DATE") val updatedDate = column[java.util.Date]("UPDATED_DATE")
def * = (userName, repositoryName, name, tag, author, content, registeredDate, updatedDate) <> (Release.tupled, Release.unapply) def * =
(userName, repositoryName, name, tag, author, content, registeredDate, updatedDate) <> (ReleaseTag.tupled, ReleaseTag.unapply)
def byPrimaryKey(owner: String, repository: String, tag: String) = byTag(owner, repository, tag) def byPrimaryKey(owner: String, repository: String, tag: String) = byTag(owner, repository, tag)
def byTag(owner: String, repository: String, tag: String) = byRepository(owner, repository) && (this.tag === tag.bind) def byTag(owner: String, repository: String, tag: String) =
byRepository(owner, repository) && (this.tag === tag.bind)
} }
} }
case class Release( case class ReleaseTag(
userName: String, userName: String,
repositoryName: String, repositoryName: String,
name: String, name: String,

View File

@@ -22,13 +22,28 @@ trait RepositoryComponent extends TemplateComponent { self: Profile =>
val wikiOption = column[String]("WIKI_OPTION") val wikiOption = column[String]("WIKI_OPTION")
val externalWikiUrl = column[String]("EXTERNAL_WIKI_URL") val externalWikiUrl = column[String]("EXTERNAL_WIKI_URL")
val allowFork = column[Boolean]("ALLOW_FORK") val allowFork = column[Boolean]("ALLOW_FORK")
val mergeOptions = column[String]("MERGE_OPTIONS")
val defaultMergeOption = column[String]("DEFAULT_MERGE_OPTION")
def * = ( def * =
(userName, repositoryName, isPrivate, description.?, defaultBranch, (
registeredDate, updatedDate, lastActivityDate, originUserName.?, originRepositoryName.?, parentUserName.?, parentRepositoryName.?), (
(issuesOption, externalIssuesUrl.?, wikiOption, externalWikiUrl.?, allowFork) userName,
).shaped <> ( repositoryName,
{ case (repository, options) => isPrivate,
description.?,
defaultBranch,
registeredDate,
updatedDate,
lastActivityDate,
originUserName.?,
originRepositoryName.?,
parentUserName.?,
parentRepositoryName.?
),
(issuesOption, externalIssuesUrl.?, wikiOption, externalWikiUrl.?, allowFork, mergeOptions, defaultMergeOption)
).shaped <> ({
case (repository, options) =>
Repository( Repository(
repository._1, repository._1,
repository._2, repository._2,
@@ -45,7 +60,9 @@ trait RepositoryComponent extends TemplateComponent { self: Profile =>
RepositoryOptions.tupled.apply(options) RepositoryOptions.tupled.apply(options)
) )
}, { (r: Repository) => }, { (r: Repository) =>
Some((( Some(
(
(
r.userName, r.userName,
r.repositoryName, r.repositoryName,
r.isPrivate, r.isPrivate,
@@ -58,9 +75,12 @@ trait RepositoryComponent extends TemplateComponent { self: Profile =>
r.originRepositoryName, r.originRepositoryName,
r.parentUserName, r.parentUserName,
r.parentRepositoryName r.parentRepositoryName
),( ),
(
RepositoryOptions.unapply(r.options).get RepositoryOptions.unapply(r.options).get
))) )
)
)
}) })
def byPrimaryKey(owner: String, repository: String) = byRepository(owner, repository) def byPrimaryKey(owner: String, repository: String) = byRepository(owner, repository)
@@ -88,5 +108,7 @@ case class RepositoryOptions(
externalIssuesUrl: Option[String], externalIssuesUrl: Option[String],
wikiOption: String, wikiOption: String,
externalWikiUrl: Option[String], externalWikiUrl: Option[String],
allowFork: Boolean allowFork: Boolean,
mergeOptions: String,
defaultMergeOption: String
) )

View File

@@ -3,7 +3,8 @@ package gitbucket.core.model
trait RepositoryWebHookComponent extends TemplateComponent { self: Profile => trait RepositoryWebHookComponent extends TemplateComponent { self: Profile =>
import profile.api._ import profile.api._
implicit val whContentTypeColumnType = MappedColumnType.base[WebHookContentType, String](whct => whct.code , code => WebHookContentType.valueOf(code)) implicit val whContentTypeColumnType =
MappedColumnType.base[WebHookContentType, String](whct => whct.code, code => WebHookContentType.valueOf(code))
lazy val RepositoryWebHooks = TableQuery[RepositoryWebHooks] lazy val RepositoryWebHooks = TableQuery[RepositoryWebHooks]
@@ -11,13 +12,14 @@ trait RepositoryWebHookComponent extends TemplateComponent { self: Profile =>
val url = column[String]("URL") val url = column[String]("URL")
val token = column[Option[String]]("TOKEN") val token = column[Option[String]]("TOKEN")
val ctype = column[WebHookContentType]("CTYPE") val ctype = column[WebHookContentType]("CTYPE")
def * = (userName, repositoryName, url, ctype, token) <> ((RepositoryWebHook.apply _).tupled, RepositoryWebHook.unapply) def * =
(userName, repositoryName, url, ctype, token) <> ((RepositoryWebHook.apply _).tupled, RepositoryWebHook.unapply)
def byPrimaryKey(owner: String, repository: String, url: String) = byRepository(owner, repository) && (this.url === url.bind) def byPrimaryKey(owner: String, repository: String, url: String) =
byRepository(owner, repository) && (this.url === url.bind)
} }
} }
case class RepositoryWebHook( case class RepositoryWebHook(
userName: String, userName: String,
repositoryName: String, repositoryName: String,

View File

@@ -6,17 +6,22 @@ trait RepositoryWebHookEventComponent extends TemplateComponent { self: Profile
lazy val RepositoryWebHookEvents = TableQuery[RepositoryWebHookEvents] lazy val RepositoryWebHookEvents = TableQuery[RepositoryWebHookEvents]
class RepositoryWebHookEvents(tag: Tag) extends Table[RepositoryWebHookEvent](tag, "WEB_HOOK_EVENT") with BasicTemplate { class RepositoryWebHookEvents(tag: Tag)
extends Table[RepositoryWebHookEvent](tag, "WEB_HOOK_EVENT")
with BasicTemplate {
val url = column[String]("URL") val url = column[String]("URL")
val event = column[WebHook.Event]("EVENT") val event = column[WebHook.Event]("EVENT")
def * = (userName, repositoryName, url, event) <> ((RepositoryWebHookEvent.apply _).tupled, RepositoryWebHookEvent.unapply) def * =
(userName, repositoryName, url, event) <> ((RepositoryWebHookEvent.apply _).tupled, RepositoryWebHookEvent.unapply)
def byRepositoryWebHook(owner: String, repository: String, url: String) = byRepository(owner, repository) && (this.url === url.bind) def byRepositoryWebHook(owner: String, repository: String, url: String) =
byRepository(owner, repository) && (this.url === url.bind)
def byRepositoryWebHook(owner: Rep[String], repository: Rep[String], url: Rep[String]) = def byRepositoryWebHook(owner: Rep[String], repository: Rep[String], url: Rep[String]) =
byRepository(userName, repositoryName) && (this.url === url) byRepository(userName, repositoryName) && (this.url === url)
def byRepositoryWebHook(webhook: RepositoryWebHooks) = def byRepositoryWebHook(webhook: RepositoryWebHooks) =
byRepository(webhook.userName, webhook.repositoryName) && (this.url === webhook.url) byRepository(webhook.userName, webhook.repositoryName) && (this.url === webhook.url)
def byPrimaryKey(owner: String, repository: String, url: String, event: WebHook.Event) = byRepositoryWebHook(owner, repository, url) && (this.event === event.bind) def byPrimaryKey(owner: String, repository: String, url: String, event: WebHook.Event) =
byRepositoryWebHook(owner, repository, url) && (this.event === event.bind)
} }
} }

View File

@@ -12,7 +12,8 @@ trait SshKeyComponent { self: Profile =>
val publicKey = column[String]("PUBLIC_KEY") val publicKey = column[String]("PUBLIC_KEY")
def * = (userName, sshKeyId, title, publicKey) <> (SshKey.tupled, SshKey.unapply) def * = (userName, sshKeyId, title, publicKey) <> (SshKey.tupled, SshKey.unapply)
def byPrimaryKey(userName: String, sshKeyId: Int) = (this.userName === userName.bind) && (this.sshKeyId === sshKeyId.bind) def byPrimaryKey(userName: String, sshKeyId: Int) =
(this.userName === userName.bind) && (this.sshKeyId === sshKeyId.bind)
} }
} }

View File

@@ -16,7 +16,7 @@ object WebHookContentType {
def valueOpt(code: String): Option[WebHookContentType] = map.get(code) def valueOpt(code: String): Option[WebHookContentType] = map.get(code)
} }
trait WebHook{ trait WebHook {
val url: String val url: String
val ctype: WebHookContentType val ctype: WebHookContentType
val token: Option[String] val token: Option[String]
@@ -45,7 +45,7 @@ object WebHook {
case object TeamAdd extends Event("team_add") case object TeamAdd extends Event("team_add")
case object Watch extends Event("watch") case object Watch extends Event("watch")
object Event{ object Event {
val values = List( val values = List(
CommitComment, CommitComment,
Create, Create,
@@ -68,7 +68,7 @@ object WebHook {
Watch Watch
) )
private val map: Map[String,Event] = values.map(e => e.name -> e).toMap private val map: Map[String, Event] = values.map(e => e.name -> e).toMap
def valueOf(name: String): Event = map(name) def valueOf(name: String): Event = map(name)
def valueOpt(name: String): Option[Event] = map.get(name) def valueOpt(name: String): Option[Event] = map.get(name)
} }

View File

@@ -10,13 +10,18 @@ import gitbucket.core.service.SystemSettingsService.SystemSettings
* @param localPath the string to assemble local file path of repository (e.g. "gist/$1/$2") * @param localPath the string to assemble local file path of repository (e.g. "gist/$1/$2")
* @param filter the filter for request to the Git repository which is defined by this routing * @param filter the filter for request to the Git repository which is defined by this routing
*/ */
case class GitRepositoryRouting(urlPattern: String, localPath: String, filter: GitRepositoryFilter){ case class GitRepositoryRouting(urlPattern: String, localPath: String, filter: GitRepositoryFilter) {
def this(urlPattern: String, localPath: String) = { def this(urlPattern: String, localPath: String) = {
this(urlPattern, localPath, new GitRepositoryFilter(){ this(
def filter(repositoryName: String, userName: Option[String], settings: SystemSettings, isUpdating: Boolean) urlPattern,
(implicit session: Session): Boolean = true localPath,
}) new GitRepositoryFilter() {
def filter(repositoryName: String, userName: Option[String], settings: SystemSettings, isUpdating: Boolean)(
implicit session: Session
): Boolean = true
}
)
} }
} }
@@ -36,7 +41,8 @@ trait GitRepositoryFilter {
* @param session the database session * @param session the database session
* @return true if allow accessing to repository, otherwise false. * @return true if allow accessing to repository, otherwise false.
*/ */
def filter(path: String, userName: Option[String], settings: SystemSettings, isUpdating: Boolean) def filter(path: String, userName: Option[String], settings: SystemSettings, isUpdating: Boolean)(
(implicit session: Session): Boolean implicit session: Session
): Boolean
} }

View File

@@ -1,7 +1,7 @@
package gitbucket.core.plugin package gitbucket.core.plugin
import gitbucket.core.controller.Context import gitbucket.core.controller.Context
import gitbucket.core.model.Issue import gitbucket.core.model.{Account, Issue}
import gitbucket.core.service.RepositoryService.RepositoryInfo import gitbucket.core.service.RepositoryService.RepositoryInfo
import gitbucket.core.model.Profile._ import gitbucket.core.model.Profile._
import profile.api._ import profile.api._
@@ -9,9 +9,25 @@ import profile.api._
trait IssueHook { trait IssueHook {
def created(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = () def created(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = ()
def addedComment(commentId: Int, content: String, issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = () def addedComment(commentId: Int, content: String, issue: Issue, repository: RepositoryInfo)(
implicit session: Session,
context: Context
): Unit = ()
def closed(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = () def closed(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = ()
def reopened(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = () def reopened(issue: Issue, repository: RepositoryInfo)(implicit session: Session, context: Context): Unit = ()
def assigned(
issue: Issue,
repository: RepositoryInfo,
assigner: Option[String],
assigned: Option[String],
oldAssigned: Option[String]
)(
implicit session: Session,
context: Context
): Unit = ()
def closedByCommitComment(issue: Issue, repository: RepositoryInfo, message: String, pusher: Account)(
implicit session: Session
): Unit = ()
} }

View File

@@ -30,7 +30,8 @@ abstract class Plugin {
/** /**
* Override to declare this plug-in provides images. * Override to declare this plug-in provides images.
*/ */
def images(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, Array[Byte])] = Nil def images(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, Array[Byte])] =
Nil
/** /**
* Override to declare this plug-in provides controllers. * Override to declare this plug-in provides controllers.
@@ -40,7 +41,11 @@ abstract class Plugin {
/** /**
* Override to declare this plug-in provides controllers. * Override to declare this plug-in provides controllers.
*/ */
def controllers(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, ControllerBase)] = Nil def controllers(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(String, ControllerBase)] = Nil
/** /**
* Override to declare this plug-in provides JavaScript. * Override to declare this plug-in provides JavaScript.
@@ -50,7 +55,8 @@ abstract class Plugin {
/** /**
* Override to declare this plug-in provides JavaScript. * Override to declare this plug-in provides JavaScript.
*/ */
def javaScripts(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, String)] = Nil def javaScripts(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, String)] =
Nil
/** /**
* Override to declare this plug-in provides renderers. * Override to declare this plug-in provides renderers.
@@ -60,7 +66,8 @@ abstract class Plugin {
/** /**
* Override to declare this plug-in provides renderers. * Override to declare this plug-in provides renderers.
*/ */
def renderers(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, Renderer)] = Nil def renderers(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, Renderer)] =
Nil
/** /**
* Override to add git repository routings. * Override to add git repository routings.
@@ -70,7 +77,11 @@ abstract class Plugin {
/** /**
* Override to add git repository routings. * Override to add git repository routings.
*/ */
def repositoryRoutings(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[GitRepositoryRouting] = Nil def repositoryRoutings(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[GitRepositoryRouting] = Nil
/** /**
* Override to add account hooks. * Override to add account hooks.
@@ -100,7 +111,11 @@ abstract class Plugin {
/** /**
* Override to add repository hooks. * Override to add repository hooks.
*/ */
def repositoryHooks(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[RepositoryHook] = Nil def repositoryHooks(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[RepositoryHook] = Nil
/** /**
* Override to add issue hooks. * Override to add issue hooks.
@@ -120,7 +135,11 @@ abstract class Plugin {
/** /**
* Override to add pull request hooks. * Override to add pull request hooks.
*/ */
def pullRequestHooks(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[PullRequestHook] = Nil def pullRequestHooks(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[PullRequestHook] = Nil
/** /**
* Override to add repository headers. * Override to add repository headers.
@@ -130,7 +149,11 @@ abstract class Plugin {
/** /**
* Override to add repository headers. * Override to add repository headers.
*/ */
def repositoryHeaders(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(RepositoryInfo, Context) => Option[Html]] = Nil def repositoryHeaders(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(RepositoryInfo, Context) => Option[Html]] = Nil
/** /**
* Override to add global menus. * Override to add global menus.
@@ -140,7 +163,11 @@ abstract class Plugin {
/** /**
* Override to add global menus. * Override to add global menus.
*/ */
def globalMenus(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Context) => Option[Link]] = Nil def globalMenus(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Context) => Option[Link]] = Nil
/** /**
* Override to add repository menus. * Override to add repository menus.
@@ -150,7 +177,11 @@ abstract class Plugin {
/** /**
* Override to add repository menus. * Override to add repository menus.
*/ */
def repositoryMenus(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(RepositoryInfo, Context) => Option[Link]] = Nil def repositoryMenus(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(RepositoryInfo, Context) => Option[Link]] = Nil
/** /**
* Override to add repository setting tabs. * Override to add repository setting tabs.
@@ -160,7 +191,11 @@ abstract class Plugin {
/** /**
* Override to add repository setting tabs. * Override to add repository setting tabs.
*/ */
def repositorySettingTabs(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(RepositoryInfo, Context) => Option[Link]] = Nil def repositorySettingTabs(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(RepositoryInfo, Context) => Option[Link]] = Nil
/** /**
* Override to add profile tabs. * Override to add profile tabs.
@@ -170,7 +205,11 @@ abstract class Plugin {
/** /**
* Override to add profile tabs. * Override to add profile tabs.
*/ */
def profileTabs(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Account, Context) => Option[Link]] = Nil def profileTabs(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Account, Context) => Option[Link]] = Nil
/** /**
* Override to add system setting menus. * Override to add system setting menus.
@@ -180,7 +219,11 @@ abstract class Plugin {
/** /**
* Override to add system setting menus. * Override to add system setting menus.
*/ */
def systemSettingMenus(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Context) => Option[Link]] = Nil def systemSettingMenus(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Context) => Option[Link]] = Nil
/** /**
* Override to add account setting menus. * Override to add account setting menus.
@@ -190,7 +233,11 @@ abstract class Plugin {
/** /**
* Override to add account setting menus. * Override to add account setting menus.
*/ */
def accountSettingMenus(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Context) => Option[Link]] = Nil def accountSettingMenus(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Context) => Option[Link]] = Nil
/** /**
* Override to add dashboard tabs. * Override to add dashboard tabs.
@@ -200,7 +247,11 @@ abstract class Plugin {
/** /**
* Override to add dashboard tabs. * Override to add dashboard tabs.
*/ */
def dashboardTabs(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Context) => Option[Link]] = Nil def dashboardTabs(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Context) => Option[Link]] = Nil
/** /**
* Override to add issue sidebars. * Override to add issue sidebars.
@@ -210,7 +261,11 @@ abstract class Plugin {
/** /**
* Override to add issue sidebars. * Override to add issue sidebars.
*/ */
def issueSidebars(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(Issue, RepositoryInfo, Context) => Option[Html]] = Nil def issueSidebars(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(Issue, RepositoryInfo, Context) => Option[Html]] = Nil
/** /**
* Override to add assets mappings. * Override to add assets mappings.
@@ -220,7 +275,11 @@ abstract class Plugin {
/** /**
* Override to add assets mappings. * Override to add assets mappings.
*/ */
def assetsMappings(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[(String, String)] = Nil def assetsMappings(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[(String, String)] = Nil
/** /**
* Override to add text decorators. * Override to add text decorators.
@@ -230,7 +289,8 @@ abstract class Plugin {
/** /**
* Override to add text decorators. * Override to add text decorators.
*/ */
def textDecorators(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[TextDecorator] = Nil def textDecorators(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[TextDecorator] =
Nil
/** /**
* Override to add suggestion provider. * Override to add suggestion provider.
@@ -240,7 +300,11 @@ abstract class Plugin {
/** /**
* Override to add suggestion provider. * Override to add suggestion provider.
*/ */
def suggestionProviders(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[SuggestionProvider] = Nil def suggestionProviders(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[SuggestionProvider] = Nil
/** /**
* Override to add ssh command providers. * Override to add ssh command providers.
@@ -250,24 +314,31 @@ abstract class Plugin {
/** /**
* Override to add ssh command providers. * Override to add ssh command providers.
*/ */
def sshCommandProviders(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Seq[PartialFunction[String, Command]] = Nil def sshCommandProviders(
registry: PluginRegistry,
context: ServletContext,
settings: SystemSettings
): Seq[PartialFunction[String, Command]] = Nil
/** /**
* This method is invoked in initialization of plugin system. * This method is invoked in initialization of plugin system.
* Register plugin functionality to PluginRegistry. * Register plugin functionality to PluginRegistry.
*/ */
def initialize(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Unit = { def initialize(registry: PluginRegistry, context: ServletContext, settings: SystemSettings): Unit = {
(images ++ images(registry, context, settings)).foreach { case (id, in) => (images ++ images(registry, context, settings)).foreach {
case (id, in) =>
registry.addImage(id, in) registry.addImage(id, in)
} }
(controllers ++ controllers(registry, context, settings)).foreach { case (path, controller) => (controllers ++ controllers(registry, context, settings)).foreach {
case (path, controller) =>
registry.addController(path, controller) registry.addController(path, controller)
} }
(javaScripts ++ javaScripts(registry, context, settings)).foreach { case (path, script) => (javaScripts ++ javaScripts(registry, context, settings)).foreach {
case (path, script) =>
registry.addJavaScript(path, script) registry.addJavaScript(path, script)
} }
(renderers ++ renderers(registry, context, settings)).foreach { case (extension, renderer) => (renderers ++ renderers(registry, context, settings)).foreach {
case (extension, renderer) =>
registry.addRenderer(extension, renderer) registry.addRenderer(extension, renderer)
} }
(repositoryRoutings ++ repositoryRoutings(registry, context, settings)).foreach { routing => (repositoryRoutings ++ repositoryRoutings(registry, context, settings)).foreach { routing =>
@@ -345,7 +416,7 @@ abstract class Plugin {
* Helper method to get a resource from classpath. * Helper method to get a resource from classpath.
*/ */
protected def fromClassPath(path: String): Array[Byte] = protected def fromClassPath(path: String): Array[Byte] =
using(getClass.getClassLoader.getResourceAsStream(path)){ in => using(getClass.getClassLoader.getResourceAsStream(path)) { in =>
val bytes = new Array[Byte](in.available) val bytes = new Array[Byte](in.available)
in.read(bytes) in.read(bytes)
bytes bytes

View File

@@ -4,11 +4,11 @@ import java.io.{File, FilenameFilter, InputStream}
import java.net.URLClassLoader import java.net.URLClassLoader
import java.nio.file.{Files, Paths, StandardWatchEventKinds} import java.nio.file.{Files, Paths, StandardWatchEventKinds}
import java.util.Base64 import java.util.Base64
import java.util.concurrent.atomic.AtomicBoolean
import java.util.concurrent.ConcurrentLinkedQueue import java.util.concurrent.ConcurrentLinkedQueue
import java.util.concurrent.ConcurrentHashMap import java.util.concurrent.ConcurrentHashMap
import javax.servlet.ServletContext import javax.servlet.ServletContext
import com.github.zafarkhaja.semver.Version
import gitbucket.core.controller.{Context, ControllerBase} import gitbucket.core.controller.{Context, ControllerBase}
import gitbucket.core.model.{Account, Issue} import gitbucket.core.model.{Account, Issue}
import gitbucket.core.service.ProtectedBranchService.ProtectedBranchReceiveHook import gitbucket.core.service.ProtectedBranchService.ProtectedBranchReceiveHook
@@ -70,7 +70,7 @@ class PluginRegistry {
@deprecated("Use addImage(id: String, bytes: Array[Byte]) instead", "3.4.0") @deprecated("Use addImage(id: String, bytes: Array[Byte]) instead", "3.4.0")
def addImage(id: String, in: InputStream): Unit = { def addImage(id: String, in: InputStream): Unit = {
val bytes = using(in){ in => val bytes = using(in) { in =>
val bytes = new Array[Byte](in.available) val bytes = new Array[Byte](in.available)
in.read(bytes) in.read(bytes)
bytes bytes
@@ -87,9 +87,11 @@ class PluginRegistry {
def getControllers(): Seq[(ControllerBase, String)] = controllers.asScala.toSeq def getControllers(): Seq[(ControllerBase, String)] = controllers.asScala.toSeq
def addJavaScript(path: String, script: String): Unit = javaScripts.add((path, script)) //javaScripts += ((path, script)) def addJavaScript(path: String, script: String): Unit =
javaScripts.add((path, script)) //javaScripts += ((path, script))
def getJavaScript(currentPath: String): List[String] = javaScripts.asScala.filter(x => currentPath.matches(x._1)).toList.map(_._2) def getJavaScript(currentPath: String): List[String] =
javaScripts.asScala.filter(x => currentPath.matches(x._1)).toList.map(_._2)
def addRenderer(extension: String, renderer: Renderer): Unit = renderers.put(extension, renderer) def addRenderer(extension: String, renderer: Renderer): Unit = renderers.put(extension, renderer)
@@ -129,7 +131,8 @@ class PluginRegistry {
def getPullRequestHooks: Seq[PullRequestHook] = pullRequestHooks.asScala.toSeq def getPullRequestHooks: Seq[PullRequestHook] = pullRequestHooks.asScala.toSeq
def addRepositoryHeader(repositoryHeader: (RepositoryInfo, Context) => Option[Html]): Unit = repositoryHeaders.add(repositoryHeader) def addRepositoryHeader(repositoryHeader: (RepositoryInfo, Context) => Option[Html]): Unit =
repositoryHeaders.add(repositoryHeader)
def getRepositoryHeaders: Seq[(RepositoryInfo, Context) => Option[Html]] = repositoryHeaders.asScala.toSeq def getRepositoryHeaders: Seq[(RepositoryInfo, Context) => Option[Html]] = repositoryHeaders.asScala.toSeq
@@ -137,11 +140,13 @@ class PluginRegistry {
def getGlobalMenus: Seq[(Context) => Option[Link]] = globalMenus.asScala.toSeq def getGlobalMenus: Seq[(Context) => Option[Link]] = globalMenus.asScala.toSeq
def addRepositoryMenu(repositoryMenu: (RepositoryInfo, Context) => Option[Link]): Unit = repositoryMenus.add(repositoryMenu) def addRepositoryMenu(repositoryMenu: (RepositoryInfo, Context) => Option[Link]): Unit =
repositoryMenus.add(repositoryMenu)
def getRepositoryMenus: Seq[(RepositoryInfo, Context) => Option[Link]] = repositoryMenus.asScala.toSeq def getRepositoryMenus: Seq[(RepositoryInfo, Context) => Option[Link]] = repositoryMenus.asScala.toSeq
def addRepositorySettingTab(repositorySettingTab: (RepositoryInfo, Context) => Option[Link]): Unit = repositorySettingTabs.add(repositorySettingTab) def addRepositorySettingTab(repositorySettingTab: (RepositoryInfo, Context) => Option[Link]): Unit =
repositorySettingTabs.add(repositorySettingTab)
def getRepositorySettingTabs: Seq[(RepositoryInfo, Context) => Option[Link]] = repositorySettingTabs.asScala.toSeq def getRepositorySettingTabs: Seq[(RepositoryInfo, Context) => Option[Link]] = repositorySettingTabs.asScala.toSeq
@@ -149,11 +154,13 @@ class PluginRegistry {
def getProfileTabs: Seq[(Account, Context) => Option[Link]] = profileTabs.asScala.toSeq def getProfileTabs: Seq[(Account, Context) => Option[Link]] = profileTabs.asScala.toSeq
def addSystemSettingMenu(systemSettingMenu: (Context) => Option[Link]): Unit = systemSettingMenus.add(systemSettingMenu) def addSystemSettingMenu(systemSettingMenu: (Context) => Option[Link]): Unit =
systemSettingMenus.add(systemSettingMenu)
def getSystemSettingMenus: Seq[(Context) => Option[Link]] = systemSettingMenus.asScala.toSeq def getSystemSettingMenus: Seq[(Context) => Option[Link]] = systemSettingMenus.asScala.toSeq
def addAccountSettingMenu(accountSettingMenu: (Context) => Option[Link]): Unit = accountSettingMenus.add(accountSettingMenu) def addAccountSettingMenu(accountSettingMenu: (Context) => Option[Link]): Unit =
accountSettingMenus.add(accountSettingMenu)
def getAccountSettingMenus: Seq[(Context) => Option[Link]] = accountSettingMenus.asScala.toSeq def getAccountSettingMenus: Seq[(Context) => Option[Link]] = accountSettingMenus.asScala.toSeq
@@ -161,7 +168,8 @@ class PluginRegistry {
def getDashboardTabs: Seq[(Context) => Option[Link]] = dashboardTabs.asScala.toSeq def getDashboardTabs: Seq[(Context) => Option[Link]] = dashboardTabs.asScala.toSeq
def addIssueSidebar(issueSidebar: (Issue, RepositoryInfo, Context) => Option[Html]): Unit = issueSidebars.add(issueSidebar) def addIssueSidebar(issueSidebar: (Issue, RepositoryInfo, Context) => Option[Html]): Unit =
issueSidebars.add(issueSidebar)
def getIssueSidebars: Seq[(Issue, RepositoryInfo, Context) => Option[Html]] = issueSidebars.asScala.toSeq def getIssueSidebars: Seq[(Issue, RepositoryInfo, Context) => Option[Html]] = issueSidebars.asScala.toSeq
@@ -177,7 +185,8 @@ class PluginRegistry {
def getSuggestionProviders: Seq[SuggestionProvider] = suggestionProviders.asScala.toSeq def getSuggestionProviders: Seq[SuggestionProvider] = suggestionProviders.asScala.toSeq
def addSshCommandProvider(sshCommandProvider: PartialFunction[String, Command]): Unit = sshCommandProviders.add(sshCommandProvider) def addSshCommandProvider(sshCommandProvider: PartialFunction[String, Command]): Unit =
sshCommandProviders.add(sshCommandProvider)
def getSshCommandProviders: Seq[PartialFunction[String, Command]] = sshCommandProviders.asScala.toSeq def getSshCommandProviders: Seq[PartialFunction[String, Command]] = sshCommandProviders.asScala.toSeq
} }
@@ -193,7 +202,6 @@ object PluginRegistry {
private var watcher: PluginWatchThread = null private var watcher: PluginWatchThread = null
private var extraWatcher: PluginWatchThread = null private var extraWatcher: PluginWatchThread = null
private val initializing = new AtomicBoolean(false)
/** /**
* Returns the PluginRegistry singleton instance. * Returns the PluginRegistry singleton instance.
@@ -212,41 +220,78 @@ object PluginRegistry {
/** /**
* Uninstall a specified plugin. * Uninstall a specified plugin.
*/ */
def uninstall(pluginId: String, context: ServletContext, settings: SystemSettings, conn: java.sql.Connection): Unit = synchronized { def uninstall(pluginId: String, context: ServletContext, settings: SystemSettings, conn: java.sql.Connection): Unit =
instance.getPlugins() synchronized {
.collect { case plugin if plugin.pluginId == pluginId => plugin }
.foreach { plugin =>
// try {
// plugin.pluginClass.uninstall(instance, context, settings)
// } catch {
// case e: Exception =>
// logger.error(s"Error during uninstalling plugin: ${plugin.pluginJar.getName}", e)
// }
shutdown(context, settings) shutdown(context, settings)
plugin.pluginJar.delete()
new File(PluginHome)
.listFiles((_: File, name: String) => {
name.startsWith(s"gitbucket-${pluginId}-plugin") && name.endsWith(".jar")
})
.foreach(_.delete())
instance = new PluginRegistry() instance = new PluginRegistry()
initialize(context, settings, conn) initialize(context, settings, conn)
} }
}
/** /**
* Install a plugin from a specified jar file. * Install a plugin from a specified jar file.
*/ */
def install(file: File, context: ServletContext, settings: SystemSettings, conn: java.sql.Connection): Unit = synchronized { def install(
pluginId: String,
url: java.net.URL,
context: ServletContext,
settings: SystemSettings,
conn: java.sql.Connection
): Unit =
synchronized {
shutdown(context, settings) shutdown(context, settings)
FileUtils.copyFile(file, new File(PluginHome, file.getName))
new File(PluginHome)
.listFiles((_: File, name: String) => {
name.startsWith(s"gitbucket-${pluginId}-plugin") && name.endsWith(".jar")
})
.foreach(_.delete())
val in = url.openStream()
FileUtils.copyToFile(in, new File(PluginHome, new File(url.getFile).getName))
instance = new PluginRegistry() instance = new PluginRegistry()
initialize(context, settings, conn) initialize(context, settings, conn)
} }
private def listPluginJars(dir: File): Seq[File] = { private def listPluginJars(dir: File): Seq[File] = {
dir.listFiles(new FilenameFilter { dir
.listFiles(new FilenameFilter {
override def accept(dir: File, name: String): Boolean = name.endsWith(".jar") override def accept(dir: File, name: String): Boolean = name.endsWith(".jar")
}).toSeq.sortBy(_.getName).reverse })
.toSeq
.sortBy(x => Version.valueOf(getPluginVersion(x.getName)))
.reverse
} }
lazy val extraPluginDir: Option[String] = Option(System.getProperty("gitbucket.pluginDir")) lazy val extraPluginDir: Option[String] = Option(System.getProperty("gitbucket.pluginDir"))
def getGitBucketVersion(pluginJarFileName: String): Option[String] = {
val regex = ".+-gitbucket\\_(\\d+\\.\\d+\\.\\d+(-SNAPSHOT)?)-.+".r
pluginJarFileName match {
case regex(all, _) => Some(all)
case _ => None
}
}
def getPluginVersion(pluginJarFileName: String): String = {
val regex = ".+-((\\d+)\\.(\\d+)(\\.(\\d+))?(-SNAPSHOT)?)\\.jar$".r
pluginJarFileName match {
case regex(all, major, minor, _, patch, modifier) => {
if (patch != null) all
else {
s"${major}.${minor}.0" + (if (modifier == null) "" else modifier)
}
}
case _ => "0.0.0"
}
}
/** /**
* Initializes all installed plugins. * Initializes all installed plugins.
*/ */
@@ -256,21 +301,26 @@ object PluginRegistry {
// Clean installed directory // Clean installed directory
val installedDir = new File(PluginHome, ".installed") val installedDir = new File(PluginHome, ".installed")
if(installedDir.exists){ if (installedDir.exists) {
FileUtils.deleteDirectory(installedDir) FileUtils.deleteDirectory(installedDir)
} }
installedDir.mkdirs() installedDir.mkdirs()
val pluginJars = listPluginJars(pluginDir) val pluginJars = listPluginJars(pluginDir)
val extraJars = extraPluginDir.map { extraDir => listPluginJars(new File(extraDir)) }.getOrElse(Nil)
val extraJars = extraPluginDir
.map { extraDir =>
listPluginJars(new File(extraDir))
}
.getOrElse(Nil)
(extraJars ++ pluginJars).foreach { pluginJar => (extraJars ++ pluginJars).foreach { pluginJar =>
val installedJar = new File(installedDir, pluginJar.getName) val installedJar = new File(installedDir, pluginJar.getName)
FileUtils.copyFile(pluginJar, installedJar) FileUtils.copyFile(pluginJar, installedJar)
logger.info(s"Initialize ${pluginJar.getName}") logger.info(s"Initialize ${pluginJar.getName}")
val classLoader = new URLClassLoader(Array(installedJar.toURI.toURL), Thread.currentThread.getContextClassLoader) val classLoader =
new URLClassLoader(Array(installedJar.toURI.toURL), Thread.currentThread.getContextClassLoader)
try { try {
val plugin = classLoader.loadClass("Plugin").getDeclaredConstructor().newInstance().asInstanceOf[Plugin] val plugin = classLoader.loadClass("Plugin").getDeclaredConstructor().newInstance().asInstanceOf[Plugin]
val pluginId = plugin.pluginId val pluginId = plugin.pluginId
@@ -283,26 +333,38 @@ object PluginRegistry {
case None => { case None => {
// Migration // Migration
val solidbase = new Solidbase() val solidbase = new Solidbase()
solidbase.migrate(conn, classLoader, DatabaseConfig.liquiDriver, new Module(plugin.pluginId, plugin.versions: _*)) solidbase
.migrate(
conn,
classLoader,
DatabaseConfig.liquiDriver,
new Module(plugin.pluginId, plugin.versions: _*)
)
conn.commit()
// Check database version // Check database version
val databaseVersion = manager.getCurrentVersion(plugin.pluginId) val databaseVersion = manager.getCurrentVersion(plugin.pluginId)
val pluginVersion = plugin.versions.last.getVersion val pluginVersion = plugin.versions.last.getVersion
if (databaseVersion != pluginVersion) { if (databaseVersion != pluginVersion) {
throw new IllegalStateException(s"Plugin version is ${pluginVersion}, but database version is ${databaseVersion}") throw new IllegalStateException(
s"Plugin version is ${pluginVersion}, but database version is ${databaseVersion}"
)
} }
// Initialize // Initialize
plugin.initialize(instance, context, settings) plugin.initialize(instance, context, settings)
instance.addPlugin(PluginInfo( instance.addPlugin(
PluginInfo(
pluginId = plugin.pluginId, pluginId = plugin.pluginId,
pluginName = plugin.pluginName, pluginName = plugin.pluginName,
pluginVersion = plugin.versions.last.getVersion, pluginVersion = plugin.versions.last.getVersion,
gitbucketVersion = getGitBucketVersion(installedJar.getName),
description = plugin.description, description = plugin.description,
pluginClass = plugin, pluginClass = plugin,
pluginJar = pluginJar, pluginJar = pluginJar,
classLoader = classLoader classLoader = classLoader
)) )
)
} }
} }
} catch { } catch {
@@ -310,13 +372,13 @@ object PluginRegistry {
} }
} }
if(watcher == null){ if (watcher == null) {
watcher = new PluginWatchThread(context, PluginHome) watcher = new PluginWatchThread(context, PluginHome)
watcher.start() watcher.start()
} }
extraPluginDir.foreach { extraDir => extraPluginDir.foreach { extraDir =>
if(extraWatcher == null){ if (extraWatcher == null) {
extraWatcher = new PluginWatchThread(context, extraDir) extraWatcher = new PluginWatchThread(context, extraDir)
extraWatcher.start() extraWatcher.start()
} }
@@ -327,11 +389,11 @@ object PluginRegistry {
instance.getPlugins().foreach { plugin => instance.getPlugins().foreach { plugin =>
try { try {
plugin.pluginClass.shutdown(instance, context, settings) plugin.pluginClass.shutdown(instance, context, settings)
if(watcher != null){ if (watcher != null) {
watcher.interrupt() watcher.interrupt()
watcher = null watcher = null
} }
if(extraWatcher != null){ if (extraWatcher != null) {
extraWatcher.interrupt() extraWatcher.interrupt()
extraWatcher = null extraWatcher = null
} }
@@ -358,6 +420,7 @@ class PluginInfoBase(
val pluginId: String, val pluginId: String,
val pluginName: String, val pluginName: String,
val pluginVersion: String, val pluginVersion: String,
val gitbucketVersion: Option[String],
val description: String val description: String
) )
@@ -365,11 +428,12 @@ case class PluginInfo(
override val pluginId: String, override val pluginId: String,
override val pluginName: String, override val pluginName: String,
override val pluginVersion: String, override val pluginVersion: String,
override val gitbucketVersion: Option[String],
override val description: String, override val description: String,
pluginClass: Plugin, pluginClass: Plugin,
pluginJar: File, pluginJar: File,
classLoader: URLClassLoader classLoader: URLClassLoader
) extends PluginInfoBase(pluginId, pluginName, pluginVersion, description) ) extends PluginInfoBase(pluginId, pluginName, pluginVersion, gitbucketVersion, description)
class PluginWatchThread(context: ServletContext, dir: String) extends Thread with SystemSettingsService { class PluginWatchThread(context: ServletContext, dir: String) extends Thread with SystemSettingsService {
import gitbucket.core.model.Profile.profile.blockingApi._ import gitbucket.core.model.Profile.profile.blockingApi._
@@ -379,17 +443,19 @@ class PluginWatchThread(context: ServletContext, dir: String) extends Thread wit
override def run(): Unit = { override def run(): Unit = {
val path = Paths.get(dir) val path = Paths.get(dir)
if(!Files.exists(path)){ if (!Files.exists(path)) {
Files.createDirectories(path) Files.createDirectories(path)
} }
val fs = path.getFileSystem val fs = path.getFileSystem
val watcher = fs.newWatchService val watcher = fs.newWatchService
val watchKey = path.register(watcher, val watchKey = path.register(
watcher,
StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_CREATE,
StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.ENTRY_MODIFY,
StandardWatchEventKinds.ENTRY_DELETE, StandardWatchEventKinds.ENTRY_DELETE,
StandardWatchEventKinds.OVERFLOW) StandardWatchEventKinds.OVERFLOW
)
logger.info("Start PluginWatchThread: " + path) logger.info("Start PluginWatchThread: " + path)
@@ -399,7 +465,7 @@ class PluginWatchThread(context: ServletContext, dir: String) extends Thread wit
val events = detectedWatchKey.pollEvents.asScala.filter { e => val events = detectedWatchKey.pollEvents.asScala.filter { e =>
e.context.toString != ".installed" && !e.context.toString.endsWith(".bak") e.context.toString != ".installed" && !e.context.toString.endsWith(".bak")
} }
if(events.nonEmpty){ if (events.nonEmpty) {
events.foreach { event => events.foreach { event =>
logger.info(event.kind + ": " + event.context) logger.info(event.kind + ": " + event.context)
} }

View File

@@ -1,23 +1,27 @@
package gitbucket.core.plugin package gitbucket.core.plugin
import org.json4s._ import org.json4s._
import gitbucket.core.util.Directory._ import org.apache.commons.io.IOUtils
import org.apache.commons.io.FileUtils import org.slf4j.LoggerFactory
object PluginRepository { object PluginRepository {
private val logger = LoggerFactory.getLogger(getClass)
implicit val formats = DefaultFormats implicit val formats = DefaultFormats
def parsePluginJson(json: String): Seq[PluginMetadata] = { def parsePluginJson(json: String): Seq[PluginMetadata] = {
org.json4s.jackson.JsonMethods.parse(json).extract[Seq[PluginMetadata]] org.json4s.jackson.JsonMethods.parse(json).extract[Seq[PluginMetadata]]
} }
lazy val LocalRepositoryDir = new java.io.File(PluginHome, ".repository")
lazy val LocalRepositoryIndexFile = new java.io.File(LocalRepositoryDir, "plugins.json")
def getPlugins(): Seq[PluginMetadata] = { def getPlugins(): Seq[PluginMetadata] = {
if(LocalRepositoryIndexFile.exists){ try {
parsePluginJson(FileUtils.readFileToString(LocalRepositoryIndexFile, "UTF-8")) val url = new java.net.URL("https://plugins.gitbucket-community.org/releases/plugins.json")
} else Nil val str = IOUtils.toString(url, "UTF-8")
parsePluginJson(str)
} catch {
case t: Throwable =>
logger.warn("Failed to access to the plugin repository: " + t.toString)
Nil
}
} }
} }
@@ -29,15 +33,12 @@ case class PluginMetadata(
description: String, description: String,
versions: Seq[VersionDef], versions: Seq[VersionDef],
default: Boolean = false default: Boolean = false
){ ) {
lazy val latestVersion: VersionDef = versions.last lazy val latestVersion: VersionDef = versions.last
} }
case class VersionDef( case class VersionDef(
version: String, version: String,
url: String, url: String,
range: String gitbucketVersion: String
){ )
lazy val file = url.substring(url.lastIndexOf("/") + 1)
}

View File

@@ -6,10 +6,12 @@ import profile.api._
trait ReceiveHook { trait ReceiveHook {
def preReceive(owner: String, repository: String, receivePack: ReceivePack, command: ReceiveCommand, pusher: String) def preReceive(owner: String, repository: String, receivePack: ReceivePack, command: ReceiveCommand, pusher: String)(
(implicit session: Session): Option[String] = None implicit session: Session
): Option[String] = None
def postReceive(owner: String, repository: String, receivePack: ReceivePack, command: ReceiveCommand, pusher: String) def postReceive(owner: String, repository: String, receivePack: ReceivePack, command: ReceiveCommand, pusher: String)(
(implicit session: Session): Unit = () implicit session: Session
): Unit = ()
} }

View File

@@ -21,14 +21,16 @@ trait Renderer {
object MarkdownRenderer extends Renderer { object MarkdownRenderer extends Renderer {
override def render(request: RenderRequest): Html = { override def render(request: RenderRequest): Html = {
import request._ import request._
Html(Markdown.toHtml( Html(
Markdown.toHtml(
markdown = fileContent, markdown = fileContent,
repository = repository, repository = repository,
enableWikiLink = enableWikiLink, enableWikiLink = enableWikiLink,
enableRefsLink = enableRefsLink, enableRefsLink = enableRefsLink,
enableAnchor = enableAnchor, enableAnchor = enableAnchor,
enableLineBreaks = false enableLineBreaks = false
)(context)) )(context)
)
} }
} }

View File

@@ -73,7 +73,9 @@ trait SuggestionProvider {
* *
* Each element can be accessed as `option` in `template()` or `replace()` method. * Each element can be accessed as `option` in `template()` or `replace()` method.
*/ */
def options(repository: RepositoryInfo): Seq[(String, String)] = values(repository).map { value => (value, value) } def options(repository: RepositoryInfo): Seq[(String, String)] = values(repository).map { value =>
(value, value)
}
/** /**
* JavaScript fragment to generate a label of completion proposal. The default is: `option.label`. * JavaScript fragment to generate a label of completion proposal. The default is: `option.label`.

View File

@@ -2,17 +2,16 @@ package gitbucket.core.service
import gitbucket.core.model.Profile._ import gitbucket.core.model.Profile._
import gitbucket.core.model.Profile.profile.blockingApi._ import gitbucket.core.model.Profile.profile.blockingApi._
import gitbucket.core.model.{Account, AccessToken} import gitbucket.core.model.{AccessToken, Account}
import gitbucket.core.util.StringUtil import gitbucket.core.util.StringUtil
import scala.util.Random import java.security.SecureRandom
trait AccessTokenService { trait AccessTokenService {
def makeAccessTokenString: String = { def makeAccessTokenString: String = {
val bytes = new Array[Byte](20) val bytes = new Array[Byte](20)
Random.nextBytes(bytes) AccessTokenService.secureRandom.nextBytes(bytes)
bytes.map("%02x".format(_)).mkString bytes.map("%02x".format(_)).mkString
} }
@@ -30,10 +29,7 @@ trait AccessTokenService {
hash = tokenToHash(token) hash = tokenToHash(token)
} while (AccessTokens.filter(_.tokenHash === hash.bind).exists.run) } while (AccessTokens.filter(_.tokenHash === hash.bind).exists.run)
val newToken = AccessToken( val newToken = AccessToken(userName = userName, note = note, tokenHash = hash)
userName = userName,
note = note,
tokenHash = hash)
val tokenId = (AccessTokens returning AccessTokens.map(_.accessTokenId)) insert newToken val tokenId = (AccessTokens returning AccessTokens.map(_.accessTokenId)) insert newToken
(tokenId, token) (tokenId, token)
} }
@@ -41,16 +37,24 @@ trait AccessTokenService {
def getAccountByAccessToken(token: String)(implicit s: Session): Option[Account] = def getAccountByAccessToken(token: String)(implicit s: Session): Option[Account] =
Accounts Accounts
.join(AccessTokens) .join(AccessTokens)
.filter { case (ac, t) => (ac.userName === t.userName) && (t.tokenHash === tokenToHash(token).bind) && (ac.removed === false.bind) } .filter {
case (ac, t) =>
(ac.userName === t.userName) && (t.tokenHash === tokenToHash(token).bind) && (ac.removed === false.bind)
}
.map { case (ac, t) => ac } .map { case (ac, t) => ac }
.firstOption .firstOption
def getAccessTokens(userName: String)(implicit s: Session): List[AccessToken] = def getAccessTokens(userName: String)(implicit s: Session): List[AccessToken] =
AccessTokens.filter(_.userName === userName.bind).sortBy(_.accessTokenId.desc).list AccessTokens.filter(_.userName === userName.bind).sortBy(_.accessTokenId.desc).list
def hasAccessToken(userName: String)(implicit s: Session): Boolean =
AccessTokens.filter(_.userName === userName.bind).exists.run
def deleteAccessToken(userName: String, accessTokenId: Int)(implicit s: Session): Unit = def deleteAccessToken(userName: String, accessTokenId: Int)(implicit s: Session): Unit =
AccessTokens filter (t => t.userName === userName.bind && t.accessTokenId === accessTokenId) delete AccessTokens filter (t => t.userName === userName.bind && t.accessTokenId === accessTokenId) delete
} }
object AccessTokenService extends AccessTokenService object AccessTokenService extends AccessTokenService {
private val secureRandom = new SecureRandom()
}

View File

@@ -21,11 +21,13 @@ trait AccountFederationService {
* @param fullName Fullname (defaults to username) * @param fullName Fullname (defaults to username)
* @return Account * @return Account
*/ */
def getOrCreateFederatedUser(issuer: String, def getOrCreateFederatedUser(
issuer: String,
subject: String, subject: String,
mailAddress: String, mailAddress: String,
preferredUserName: Option[String], preferredUserName: Option[String],
fullName: Option[String])(implicit s: Session): Option[Account] = fullName: Option[String]
)(implicit s: Session): Option[Account] =
getAccountByFederation(issuer, subject) match { getAccountByFederation(issuer, subject) match {
case Some(account) if !account.isRemoved => case Some(account) if !account.isRemoved =>
Some(account) Some(account)
@@ -34,7 +36,7 @@ trait AccountFederationService {
None None
case None => case None =>
findAvailableUserName(preferredUserName, mailAddress) flatMap { userName => findAvailableUserName(preferredUserName, mailAddress) flatMap { userName =>
createAccount(userName, "", fullName.getOrElse(userName), mailAddress, isAdmin = false, None, None) createAccount(userName, "[DUMMY]", fullName.getOrElse(userName), mailAddress, isAdmin = false, None, None)
createAccountFederation(issuer, subject, userName) createAccountFederation(issuer, subject, userName)
getAccountByUserName(userName) getAccountByUserName(userName)
} }
@@ -49,13 +51,19 @@ trait AccountFederationService {
* @param preferredUserName Username * @param preferredUserName Username
* @return Available username * @return Available username
*/ */
def findAvailableUserName(preferredUserName: Option[String], mailAddress: String)(implicit s: Session): Option[String] = { def findAvailableUserName(preferredUserName: Option[String], mailAddress: String)(
preferredUserName.flatMap(n => extractSafeStringForUserName(n)).orElse(extractSafeStringForUserName(mailAddress)) match { implicit s: Session
): Option[String] = {
preferredUserName
.flatMap(n => extractSafeStringForUserName(n))
.orElse(extractSafeStringForUserName(mailAddress)) match {
case Some(safeUserName) => case Some(safeUserName) =>
getAccountByUserName(safeUserName, includeRemoved = true) match { getAccountByUserName(safeUserName, includeRemoved = true) match {
case None => Some(safeUserName) case None => Some(safeUserName)
case Some(_) => case Some(_) =>
logger.info(s"User ($safeUserName) already exists. preferredUserName=$preferredUserName, mailAddress=$mailAddress") logger.info(
s"User ($safeUserName) already exists. preferredUserName=$preferredUserName, mailAddress=$mailAddress"
)
None None
} }
case None => case None =>
@@ -65,11 +73,16 @@ trait AccountFederationService {
} }
def getAccountByFederation(issuer: String, subject: String)(implicit s: Session): Option[Account] = def getAccountByFederation(issuer: String, subject: String)(implicit s: Session): Option[Account] =
AccountFederations.filter(_.byPrimaryKey(issuer, subject)) AccountFederations
.join(Accounts).on { case af ~ ac => af.userName === ac.userName } .filter(_.byPrimaryKey(issuer, subject))
.join(Accounts)
.on { case af ~ ac => af.userName === ac.userName }
.map { case _ ~ ac => ac } .map { case _ ~ ac => ac }
.firstOption .firstOption
def hasAccountFederation(userName: String)(implicit s: Session): Boolean =
AccountFederations.filter(_.userName === userName.bind).exists.run
def createAccountFederation(issuer: String, subject: String, userName: String)(implicit s: Session): Unit = def createAccountFederation(issuer: String, subject: String, userName: String)(implicit s: Session): Unit =
AccountFederations insert AccountFederation(issuer, subject, userName) AccountFederations insert AccountFederation(issuer, subject, userName)
} }

View File

@@ -1,11 +1,11 @@
package gitbucket.core.service package gitbucket.core.service
import org.slf4j.LoggerFactory import org.slf4j.LoggerFactory
import gitbucket.core.model.{GroupMember, Account} import gitbucket.core.model.{Account, AccountExtraMailAddress, GroupMember}
import gitbucket.core.model.Profile._ import gitbucket.core.model.Profile._
import gitbucket.core.model.Profile.profile.blockingApi._ import gitbucket.core.model.Profile.profile.blockingApi._
import gitbucket.core.model.Profile.dateColumnType import gitbucket.core.model.Profile.dateColumnType
import gitbucket.core.util.{StringUtil, LDAPUtil} import gitbucket.core.util.{LDAPUtil, StringUtil}
import StringUtil._ import StringUtil._
import gitbucket.core.service.SystemSettingsService.SystemSettings import gitbucket.core.service.SystemSettingsService.SystemSettings
@@ -13,14 +13,16 @@ trait AccountService {
private val logger = LoggerFactory.getLogger(classOf[AccountService]) private val logger = LoggerFactory.getLogger(classOf[AccountService])
def authenticate(settings: SystemSettings, userName: String, password: String)(implicit s: Session): Option[Account] = { def authenticate(settings: SystemSettings, userName: String, password: String)(
implicit s: Session
): Option[Account] = {
val account = if (settings.ldapAuthentication) { val account = if (settings.ldapAuthentication) {
ldapAuthentication(settings, userName, password) ldapAuthentication(settings, userName, password)
} else { } else {
defaultAuthentication(userName, password) defaultAuthentication(userName, password)
} }
if(account.isEmpty){ if (account.isEmpty) {
logger.info(s"Failed to authenticate: $userName") logger.info(s"Failed to authenticate: $userName")
} }
@@ -31,43 +33,62 @@ trait AccountService {
* Authenticate by internal database. * Authenticate by internal database.
*/ */
private def defaultAuthentication(userName: String, password: String)(implicit s: Session) = { private def defaultAuthentication(userName: String, password: String)(implicit s: Session) = {
val pbkdf2re = """^\$pbkdf2-sha256\$(\d+)\$([0-9a-zA-Z+/=]+)\$([0-9a-zA-Z+/=]+)$""".r
getAccountByUserName(userName).collect { getAccountByUserName(userName).collect {
case account if(!account.isGroupAccount && account.password == sha1(password)) => Some(account) case account if !account.isGroupAccount =>
account.password match {
case pbkdf2re(iter, salt, hash) if (pbkdf2_sha256(iter.toInt, salt, password) == hash) => Some(account)
case p if p == sha1(password) =>
updateAccount(account.copy(password = pbkdf2_sha256(password)))
Some(account)
case _ => None
}
case account if (!account.isGroupAccount && account.password == sha1(password)) => Some(account)
} getOrElse None } getOrElse None
} }
/** /**
* Authenticate by LDAP. * Authenticate by LDAP.
*/ */
private def ldapAuthentication(settings: SystemSettings, userName: String, password: String) private def ldapAuthentication(settings: SystemSettings, userName: String, password: String)(
(implicit s: Session): Option[Account] = { implicit s: Session
): Option[Account] = {
LDAPUtil.authenticate(settings.ldap.get, userName, password) match { LDAPUtil.authenticate(settings.ldap.get, userName, password) match {
case Right(ldapUserInfo) => { case Right(ldapUserInfo) => {
// Create or update account by LDAP information // Create or update account by LDAP information
getAccountByUserName(ldapUserInfo.userName, true) match { getAccountByUserName(ldapUserInfo.userName, true) match {
case Some(x) if(!x.isRemoved) => { case Some(x) if (!x.isRemoved) => {
if(settings.ldap.get.mailAttribute.getOrElse("").isEmpty) { if (settings.ldap.get.mailAttribute.getOrElse("").isEmpty) {
updateAccount(x.copy(fullName = ldapUserInfo.fullName)) updateAccount(x.copy(fullName = ldapUserInfo.fullName))
} else { } else {
updateAccount(x.copy(mailAddress = ldapUserInfo.mailAddress, fullName = ldapUserInfo.fullName)) updateAccount(x.copy(mailAddress = ldapUserInfo.mailAddress, fullName = ldapUserInfo.fullName))
} }
getAccountByUserName(ldapUserInfo.userName) getAccountByUserName(ldapUserInfo.userName)
} }
case Some(x) if(x.isRemoved) => { case Some(x) if (x.isRemoved) => {
logger.info("LDAP Authentication Failed: Account is already registered but disabled.") logger.info("LDAP Authentication Failed: Account is already registered but disabled.")
defaultAuthentication(userName, password) defaultAuthentication(userName, password)
} }
case None => getAccountByMailAddress(ldapUserInfo.mailAddress, true) match { case None =>
case Some(x) if(!x.isRemoved) => { getAccountByMailAddress(ldapUserInfo.mailAddress, true) match {
case Some(x) if (!x.isRemoved) => {
updateAccount(x.copy(fullName = ldapUserInfo.fullName)) updateAccount(x.copy(fullName = ldapUserInfo.fullName))
getAccountByUserName(ldapUserInfo.userName) getAccountByUserName(ldapUserInfo.userName)
} }
case Some(x) if(x.isRemoved) => { case Some(x) if (x.isRemoved) => {
logger.info("LDAP Authentication Failed: Account is already registered but disabled.") logger.info("LDAP Authentication Failed: Account is already registered but disabled.")
defaultAuthentication(userName, password) defaultAuthentication(userName, password)
} }
case None => { case None => {
createAccount(ldapUserInfo.userName, "", ldapUserInfo.fullName, ldapUserInfo.mailAddress, false, None, None) createAccount(
ldapUserInfo.userName,
"",
ldapUserInfo.fullName,
ldapUserInfo.mailAddress,
false,
None,
None
)
getAccountByUserName(ldapUserInfo.userName) getAccountByUserName(ldapUserInfo.userName)
} }
} }
@@ -81,38 +102,68 @@ trait AccountService {
} }
def getAccountByUserName(userName: String, includeRemoved: Boolean = false)(implicit s: Session): Option[Account] = def getAccountByUserName(userName: String, includeRemoved: Boolean = false)(implicit s: Session): Option[Account] =
Accounts filter(t => (t.userName === userName.bind) && (t.removed === false.bind, !includeRemoved)) firstOption Accounts filter (t => (t.userName === userName.bind) && (t.removed === false.bind, !includeRemoved)) firstOption
def getAccountsByUserNames(userNames: Set[String], knowns:Set[Account], includeRemoved: Boolean = false)(implicit s: Session): Map[String, Account] = { def getAccountByUserNameIgnoreCase(userName: String, includeRemoved: Boolean = false)(
implicit s: Session
): Option[Account] =
Accounts filter (
t => (t.userName.toLowerCase === userName.toLowerCase.bind) && (t.removed === false.bind, !includeRemoved)
) firstOption
def getAccountsByUserNames(userNames: Set[String], knowns: Set[Account], includeRemoved: Boolean = false)(
implicit s: Session
): Map[String, Account] = {
val map = knowns.map(a => a.userName -> a).toMap val map = knowns.map(a => a.userName -> a).toMap
val needs = userNames -- map.keySet val needs = userNames -- map.keySet
if(needs.isEmpty){ if (needs.isEmpty) {
map map
}else{ } else {
map ++ Accounts.filter(t => (t.userName inSetBind needs) && (t.removed === false.bind, !includeRemoved)).list.map(a => a.userName -> a).toMap map ++ Accounts
.filter(t => (t.userName inSetBind needs) && (t.removed === false.bind, !includeRemoved))
.list
.map(a => a.userName -> a)
.toMap
} }
} }
def getAccountByMailAddress(mailAddress: String, includeRemoved: Boolean = false)(implicit s: Session): Option[Account] = def getAccountByMailAddress(mailAddress: String, includeRemoved: Boolean = false)(
Accounts filter(t => (t.mailAddress.toLowerCase === mailAddress.toLowerCase.bind) && (t.removed === false.bind, !includeRemoved)) firstOption implicit s: Session
): Option[Account] =
(Accounts joinLeft AccountExtraMailAddresses on { case (a, e) => a.userName === e.userName })
.filter {
case (a, x) =>
((a.mailAddress.toLowerCase === mailAddress.toLowerCase.bind) ||
(x.map { e =>
e.extraMailAddress.toLowerCase === mailAddress.toLowerCase.bind
}
.getOrElse(false.bind))) && (a.removed === false.bind, !includeRemoved)
}
.map { case (a, e) => a } firstOption
def getAllUsers(includeRemoved: Boolean = true, includeGroups: Boolean = true)(implicit s: Session): List[Account] = def getAllUsers(includeRemoved: Boolean = true, includeGroups: Boolean = true)(implicit s: Session): List[Account] = {
{
Accounts filter { t => Accounts filter { t =>
(1.bind === 1.bind) && (1.bind === 1.bind) &&
(t.groupAccount === false.bind, !includeGroups) && (t.groupAccount === false.bind, !includeGroups) &&
(t.removed === false.bind, !includeRemoved) (t.removed === false.bind, !includeRemoved)
} sortBy(_.userName) list } sortBy (_.userName) list
} }
def isLastAdministrator(account: Account)(implicit s: Session): Boolean = { def isLastAdministrator(account: Account)(implicit s: Session): Boolean = {
if(account.isAdmin){ if (account.isAdmin) {
(Accounts filter (_.removed === false.bind) filter (_.isAdmin === true.bind) map (_.userName.length)).first == 1 (Accounts filter (_.removed === false.bind) filter (_.isAdmin === true.bind) map (_.userName.length)).first == 1
} else false } else false
} }
def createAccount(userName: String, password: String, fullName: String, mailAddress: String, isAdmin: Boolean, description: Option[String], url: Option[String]) def createAccount(
(implicit s: Session): Unit = userName: String,
password: String,
fullName: String,
mailAddress: String,
isAdmin: Boolean,
description: Option[String],
url: Option[String]
)(implicit s: Session): Unit =
Accounts insert Account( Accounts insert Account(
userName = userName, userName = userName,
password = password, password = password,
@@ -126,13 +177,29 @@ trait AccountService {
image = None, image = None,
isGroupAccount = false, isGroupAccount = false,
isRemoved = false, isRemoved = false,
description = description) description = description
)
def updateAccount(account: Account)(implicit s: Session): Unit = def updateAccount(account: Account)(implicit s: Session): Unit =
Accounts Accounts
.filter { a => a.userName === account.userName.bind } .filter { a =>
.map { a => (a.password, a.fullName, a.mailAddress, a.isAdmin, a.url.?, a.registeredDate, a.updatedDate, a.lastLoginDate.?, a.removed, a.description.?) } a.userName === account.userName.bind
.update ( }
.map { a =>
(
a.password,
a.fullName,
a.mailAddress,
a.isAdmin,
a.url.?,
a.registeredDate,
a.updatedDate,
a.lastLoginDate.?,
a.removed,
a.description.?
)
}
.update(
account.password, account.password,
account.fullName, account.fullName,
account.mailAddress, account.mailAddress,
@@ -142,11 +209,21 @@ trait AccountService {
currentDate, currentDate,
account.lastLoginDate, account.lastLoginDate,
account.isRemoved, account.isRemoved,
account.description) account.description
)
def updateAvatarImage(userName: String, image: Option[String])(implicit s: Session): Unit = def updateAvatarImage(userName: String, image: Option[String])(implicit s: Session): Unit =
Accounts.filter(_.userName === userName.bind).map(_.image.?).update(image) Accounts.filter(_.userName === userName.bind).map(_.image.?).update(image)
def getAccountExtraMailAddresses(userName: String)(implicit s: Session): List[String] = {
AccountExtraMailAddresses.filter(_.userName === userName.bind).map(_.extraMailAddress) list
}
def updateAccountExtraMailAddresses(userName: String, mails: List[String])(implicit s: Session): Unit = {
AccountExtraMailAddresses.filter(_.userName === userName.bind).delete
mails.map(AccountExtraMailAddresses insert AccountExtraMailAddress(userName, _))
}
def updateLastLoginDate(userName: String)(implicit s: Session): Unit = def updateLastLoginDate(userName: String)(implicit s: Session): Unit =
Accounts.filter(_.userName === userName.bind).map(_.lastLoginDate).update(currentDate) Accounts.filter(_.userName === userName.bind).map(_.lastLoginDate).update(currentDate)
@@ -164,17 +241,22 @@ trait AccountService {
image = None, image = None,
isGroupAccount = true, isGroupAccount = true,
isRemoved = false, isRemoved = false,
description = description) description = description
)
def updateGroup(groupName: String, description: Option[String], url: Option[String], removed: Boolean)(implicit s: Session): Unit = def updateGroup(groupName: String, description: Option[String], url: Option[String], removed: Boolean)(
Accounts.filter(_.userName === groupName.bind) implicit s: Session
): Unit =
Accounts
.filter(_.userName === groupName.bind)
.map(t => (t.url.?, t.description.?, t.updatedDate, t.removed)) .map(t => (t.url.?, t.description.?, t.updatedDate, t.removed))
.update(url, description, currentDate, removed) .update(url, description, currentDate, removed)
def updateGroupMembers(groupName: String, members: List[(String, Boolean)])(implicit s: Session): Unit = { def updateGroupMembers(groupName: String, members: List[(String, Boolean)])(implicit s: Session): Unit = {
GroupMembers.filter(_.groupName === groupName.bind).delete GroupMembers.filter(_.groupName === groupName.bind).delete
members.foreach { case (userName, isManager) => members.foreach {
GroupMembers insert GroupMember (groupName, userName, isManager) case (userName, isManager) =>
GroupMembers insert GroupMember(groupName, userName, isManager)
} }
} }

View File

@@ -15,9 +15,11 @@ trait ActivityService {
def getActivitiesByUser(activityUserName: String, isPublic: Boolean)(implicit s: Session): List[Activity] = def getActivitiesByUser(activityUserName: String, isPublic: Boolean)(implicit s: Session): List[Activity] =
Activities Activities
.join(Repositories).on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName)) .join(Repositories)
.filter { case (t1, t2) => .on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName))
if(isPublic){ .filter {
case (t1, t2) =>
if (isPublic) {
(t1.activityUserName === activityUserName.bind) && (t2.isPrivate === false.bind) (t1.activityUserName === activityUserName.bind) && (t2.isPrivate === false.bind)
} else { } else {
(t1.activityUserName === activityUserName.bind) (t1.activityUserName === activityUserName.bind)
@@ -30,173 +32,339 @@ trait ActivityService {
def getRecentActivities()(implicit s: Session): List[Activity] = def getRecentActivities()(implicit s: Session): List[Activity] =
Activities Activities
.join(Repositories).on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName)) .join(Repositories)
.on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName))
.filter { case (t1, t2) => t2.isPrivate === false.bind } .filter { case (t1, t2) => t2.isPrivate === false.bind }
.sortBy { case (t1, t2) => t1.activityId desc } .sortBy { case (t1, t2) => t1.activityId desc }
.map { case (t1, t2) => t1 } .map { case (t1, t2) => t1 }
.take(30) .take(30)
.list .list
def getRecentActivitiesByOwners(owners : Set[String])(implicit s: Session): List[Activity] = def getRecentActivitiesByOwners(owners: Set[String])(implicit s: Session): List[Activity] =
Activities Activities
.join(Repositories).on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName)) .join(Repositories)
.on((t1, t2) => t1.byRepository(t2.userName, t2.repositoryName))
.filter { case (t1, t2) => (t2.isPrivate === false.bind) || (t2.userName inSetBind owners) } .filter { case (t1, t2) => (t2.isPrivate === false.bind) || (t2.userName inSetBind owners) }
.sortBy { case (t1, t2) => t1.activityId desc } .sortBy { case (t1, t2) => t1.activityId desc }
.map { case (t1, t2) => t1 } .map { case (t1, t2) => t1 }
.take(30) .take(30)
.list .list
def recordCreateRepositoryActivity(userName: String, repositoryName: String, activityUserName: String) def recordCreateRepositoryActivity(userName: String, repositoryName: String, activityUserName: String)(
(implicit s: Session): Unit = implicit s: Session
Activities insert Activity(userName, repositoryName, activityUserName, ): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"create_repository", "create_repository",
s"[user:${activityUserName}] created [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] created [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordCreateIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String) def recordCreateIssueActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
title: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"open_issue", "open_issue",
s"[user:${activityUserName}] opened issue [issue:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] opened issue [issue:${userName}/${repositoryName}#${issueId}]",
Some(title), Some(title),
currentDate) currentDate
)
def recordCloseIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String) def recordCloseIssueActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
title: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"close_issue", "close_issue",
s"[user:${activityUserName}] closed issue [issue:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] closed issue [issue:${userName}/${repositoryName}#${issueId}]",
Some(title), Some(title),
currentDate) currentDate
)
def recordClosePullRequestActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String) def recordClosePullRequestActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
title: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"close_issue", "close_issue",
s"[user:${activityUserName}] closed pull request [pullreq:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] closed pull request [pullreq:${userName}/${repositoryName}#${issueId}]",
Some(title), Some(title),
currentDate) currentDate
)
def recordReopenIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String) def recordReopenIssueActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
title: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"reopen_issue", "reopen_issue",
s"[user:${activityUserName}] reopened issue [issue:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] reopened issue [issue:${userName}/${repositoryName}#${issueId}]",
Some(title), Some(title),
currentDate) currentDate
)
def recordCommentIssueActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, comment: String) def recordCommentIssueActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
comment: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"comment_issue", "comment_issue",
s"[user:${activityUserName}] commented on issue [issue:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] commented on issue [issue:${userName}/${repositoryName}#${issueId}]",
Some(cut(comment, 200)), Some(cut(comment, 200)),
currentDate) currentDate
)
def recordCommentPullRequestActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, comment: String) def recordCommentPullRequestActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
comment: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"comment_issue", "comment_issue",
s"[user:${activityUserName}] commented on pull request [pullreq:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] commented on pull request [pullreq:${userName}/${repositoryName}#${issueId}]",
Some(cut(comment, 200)), Some(cut(comment, 200)),
currentDate) currentDate
)
def recordCommentCommitActivity(userName: String, repositoryName: String, activityUserName: String, commitId: String, comment: String) def recordCommentCommitActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
commitId: String,
comment: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"comment_commit", "comment_commit",
s"[user:${activityUserName}] commented on commit [commit:${userName}/${repositoryName}@${commitId}]", s"[user:${activityUserName}] commented on commit [commit:${userName}/${repositoryName}@${commitId}]",
Some(cut(comment, 200)), Some(cut(comment, 200)),
currentDate currentDate
) )
def recordCreateWikiPageActivity(userName: String, repositoryName: String, activityUserName: String, pageName: String) def recordCreateWikiPageActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
pageName: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"create_wiki", "create_wiki",
s"[user:${activityUserName}] created the [repo:${userName}/${repositoryName}] wiki", s"[user:${activityUserName}] created the [repo:${userName}/${repositoryName}] wiki",
Some(pageName), Some(pageName),
currentDate) currentDate
)
def recordEditWikiPageActivity(userName: String, repositoryName: String, activityUserName: String, pageName: String, commitId: String) def recordEditWikiPageActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
pageName: String,
commitId: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"edit_wiki", "edit_wiki",
s"[user:${activityUserName}] edited the [repo:${userName}/${repositoryName}] wiki", s"[user:${activityUserName}] edited the [repo:${userName}/${repositoryName}] wiki",
Some(pageName + ":" + commitId), Some(pageName + ":" + commitId),
currentDate) currentDate
)
def recordPushActivity(userName: String, repositoryName: String, activityUserName: String, def recordPushActivity(
branchName: String, commits: List[JGitUtil.CommitInfo])(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
branchName: String,
commits: List[JGitUtil.CommitInfo]
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"push", "push",
s"[user:${activityUserName}] pushed to [branch:${userName}/${repositoryName}#${branchName}] at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] pushed to [branch:${userName}/${repositoryName}#${branchName}] at [repo:${userName}/${repositoryName}]",
Some(commits.take(5).map { commit => commit.id + ":" + commit.shortMessage }.mkString("\n")), Some(
currentDate) commits
.take(5)
.map { commit =>
commit.id + ":" + commit.shortMessage
}
.mkString("\n")
),
currentDate
)
def recordCreateTagActivity(userName: String, repositoryName: String, activityUserName: String, def recordCreateTagActivity(
tagName: String, commits: List[JGitUtil.CommitInfo])(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
tagName: String,
commits: List[JGitUtil.CommitInfo]
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"create_tag", "create_tag",
s"[user:${activityUserName}] created tag [tag:${userName}/${repositoryName}#${tagName}] at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] created tag [tag:${userName}/${repositoryName}#${tagName}] at [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordDeleteTagActivity(userName: String, repositoryName: String, activityUserName: String, def recordDeleteTagActivity(
tagName: String, commits: List[JGitUtil.CommitInfo])(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
tagName: String,
commits: List[JGitUtil.CommitInfo]
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"delete_tag", "delete_tag",
s"[user:${activityUserName}] deleted tag ${tagName} at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] deleted tag ${tagName} at [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordCreateBranchActivity(userName: String, repositoryName: String, activityUserName: String, branchName: String) def recordCreateBranchActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
branchName: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"create_branch", "create_branch",
s"[user:${activityUserName}] created branch [branch:${userName}/${repositoryName}#${branchName}] at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] created branch [branch:${userName}/${repositoryName}#${branchName}] at [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordDeleteBranchActivity(userName: String, repositoryName: String, activityUserName: String, branchName: String) def recordDeleteBranchActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
branchName: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"delete_branch", "delete_branch",
s"[user:${activityUserName}] deleted branch ${branchName} at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] deleted branch ${branchName} at [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordForkActivity(userName: String, repositoryName: String, activityUserName: String, forkedUserName: String)(implicit s: Session): Unit = def recordForkActivity(userName: String, repositoryName: String, activityUserName: String, forkedUserName: String)(
Activities insert Activity(userName, repositoryName, activityUserName, implicit s: Session
): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"fork", "fork",
s"[user:${activityUserName}] forked [repo:${userName}/${repositoryName}] to [repo:${forkedUserName}/${repositoryName}]", s"[user:${activityUserName}] forked [repo:${userName}/${repositoryName}] to [repo:${forkedUserName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
def recordPullRequestActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, title: String) def recordPullRequestActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
title: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"open_pullreq", "open_pullreq",
s"[user:${activityUserName}] opened pull request [pullreq:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] opened pull request [pullreq:${userName}/${repositoryName}#${issueId}]",
Some(title), Some(title),
currentDate) currentDate
)
def recordMergeActivity(userName: String, repositoryName: String, activityUserName: String, issueId: Int, message: String) def recordMergeActivity(
(implicit s: Session): Unit = userName: String,
Activities insert Activity(userName, repositoryName, activityUserName, repositoryName: String,
activityUserName: String,
issueId: Int,
message: String
)(implicit s: Session): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"merge_pullreq", "merge_pullreq",
s"[user:${activityUserName}] merged pull request [pullreq:${userName}/${repositoryName}#${issueId}]", s"[user:${activityUserName}] merged pull request [pullreq:${userName}/${repositoryName}#${issueId}]",
Some(message), Some(message),
currentDate) currentDate
)
def recordReleaseActivity(userName: String, repositoryName: String, activityUserName: String, name: String)(implicit s: Session): Unit = def recordReleaseActivity(userName: String, repositoryName: String, activityUserName: String, name: String)(
Activities insert Activity(userName, repositoryName, activityUserName, implicit s: Session
): Unit =
Activities insert Activity(
userName,
repositoryName,
activityUserName,
"release", "release",
s"[user:${activityUserName}] released ${name} at [repo:${userName}/${repositoryName}]", s"[user:${activityUserName}] released ${name} at [repo:${userName}/${repositoryName}]",
None, None,
currentDate) currentDate
)
private def cut(value: String, length: Int): String = private def cut(value: String, length: Int): String =
if(value.length > length) value.substring(0, length) + "..." else value if (value.length > length) value.substring(0, length) + "..." else value
} }

View File

@@ -6,19 +6,34 @@ import gitbucket.core.model.Profile.dateColumnType
import gitbucket.core.model.{CommitState, CommitStatus, Account} import gitbucket.core.model.{CommitState, CommitStatus, Account}
trait CommitStatusService { trait CommitStatusService {
/** insert or update */ /** insert or update */
def createCommitStatus(userName: String, repositoryName: String, sha: String, context: String, state: CommitState, def createCommitStatus(
targetUrl: Option[String], description: Option[String], now: java.util.Date, creator: Account)(implicit s: Session): Int = userName: String,
repositoryName: String,
sha: String,
context: String,
state: CommitState,
targetUrl: Option[String],
description: Option[String],
now: java.util.Date,
creator: Account
)(implicit s: Session): Int =
CommitStatuses CommitStatuses
.filter(t => t.byCommit(userName, repositoryName, sha) && t.context === context.bind ) .filter(t => t.byCommit(userName, repositoryName, sha) && t.context === context.bind)
.map(_.commitStatusId).firstOption match { .map(_.commitStatusId)
.firstOption match {
case Some(id: Int) => { case Some(id: Int) => {
CommitStatuses.filter(_.byPrimaryKey(id)).map { t => CommitStatuses
(t.state , t.targetUrl , t.updatedDate , t.creator, t.description) .filter(_.byPrimaryKey(id))
}.update((state, targetUrl, now, creator.userName, description)) .map { t =>
(t.state, t.targetUrl, t.updatedDate, t.creator, t.description)
}
.update((state, targetUrl, now, creator.userName, description))
id id
} }
case None => (CommitStatuses returning CommitStatuses.map(_.commitStatusId)) insert CommitStatus( case None =>
(CommitStatuses returning CommitStatuses.map(_.commitStatusId)) insert CommitStatus(
userName = userName, userName = userName,
repositoryName = repositoryName, repositoryName = repositoryName,
commitId = sha, commitId = sha,
@@ -28,23 +43,38 @@ trait CommitStatusService {
description = description, description = description,
creator = creator.userName, creator = creator.userName,
registeredDate = now, registeredDate = now,
updatedDate = now) updatedDate = now
)
} }
def getCommitStatus(userName: String, repositoryName: String, id: Int)(implicit s: Session) :Option[CommitStatus] = def getCommitStatus(userName: String, repositoryName: String, id: Int)(implicit s: Session): Option[CommitStatus] =
CommitStatuses.filter(t => t.byPrimaryKey(id) && t.byRepository(userName, repositoryName)).firstOption CommitStatuses.filter(t => t.byPrimaryKey(id) && t.byRepository(userName, repositoryName)).firstOption
def getCommitStatus(userName: String, repositoryName: String, sha: String, context: String)(implicit s: Session) :Option[CommitStatus] = def getCommitStatus(userName: String, repositoryName: String, sha: String, context: String)(
implicit s: Session
): Option[CommitStatus] =
CommitStatuses.filter(t => t.byCommit(userName, repositoryName, sha) && t.context === context.bind).firstOption CommitStatuses.filter(t => t.byCommit(userName, repositoryName, sha) && t.context === context.bind).firstOption
def getCommitStatues(userName: String, repositoryName: String, sha: String)(implicit s: Session) :List[CommitStatus] = def getCommitStatues(userName: String, repositoryName: String, sha: String)(implicit s: Session): List[CommitStatus] =
byCommitStatues(userName, repositoryName, sha).list byCommitStatues(userName, repositoryName, sha).list
def getRecentStatuesContexts(userName: String, repositoryName: String, time: java.util.Date)(implicit s: Session) :List[String] = def getRecentStatuesContexts(userName: String, repositoryName: String, time: java.util.Date)(
CommitStatuses.filter(t => t.byRepository(userName, repositoryName)).filter(t => t.updatedDate > time.bind).groupBy(_.context).map(_._1).list implicit s: Session
): List[String] =
CommitStatuses
.filter(t => t.byRepository(userName, repositoryName))
.filter(t => t.updatedDate > time.bind)
.groupBy(_.context)
.map(_._1)
.list
def getCommitStatuesWithCreator(userName: String, repositoryName: String, sha: String)(implicit s: Session) :List[(CommitStatus, Account)] = def getCommitStatuesWithCreator(userName: String, repositoryName: String, sha: String)(
byCommitStatues(userName, repositoryName, sha).join(Accounts).filter { case (t, a) => t.creator === a.userName }.list implicit s: Session
): List[(CommitStatus, Account)] =
byCommitStatues(userName, repositoryName, sha)
.join(Accounts)
.filter { case (t, a) => t.creator === a.userName }
.list
protected def byCommitStatues(userName: String, repositoryName: String, sha: String)(implicit s: Session) = protected def byCommitStatues(userName: String, repositoryName: String, sha: String)(implicit s: Session) =
CommitStatuses.filter(t => t.byCommit(userName, repositoryName, sha)).sortBy(_.updatedDate desc) CommitStatuses.filter(t => t.byCommit(userName, repositoryName, sha)).sortBy(_.updatedDate desc)

View File

@@ -1,15 +1,22 @@
package gitbucket.core.service package gitbucket.core.service
import java.io.File
import gitbucket.core.model.CommitComment import gitbucket.core.model.CommitComment
import gitbucket.core.model.Profile._ import gitbucket.core.model.Profile._
import gitbucket.core.model.Profile.profile.blockingApi._ import gitbucket.core.model.Profile.profile.blockingApi._
import gitbucket.core.model.Profile.dateColumnType import gitbucket.core.model.Profile.dateColumnType
import gitbucket.core.util.Directory._
import gitbucket.core.util.{FileUtil, StringUtil}
import org.apache.commons.io.FileUtils
trait CommitsService { trait CommitsService {
def getCommitComments(owner: String, repository: String, commitId: String, includePullRequest: Boolean)(implicit s: Session) = def getCommitComments(owner: String, repository: String, commitId: String, includePullRequest: Boolean)(
CommitComments filter { implicit s: Session
t => t.byCommit(owner, repository, commitId) && (t.issueId.isEmpty || includePullRequest) ) =
CommitComments filter { t =>
t.byCommit(owner, repository, commitId) && (t.issueId.isEmpty || includePullRequest)
} list } list
def getCommitComment(owner: String, repository: String, commentId: String)(implicit s: Session) = def getCommitComment(owner: String, repository: String, commentId: String)(implicit s: Session) =
@@ -20,9 +27,17 @@ trait CommitsService {
else else
None None
def createCommitComment(owner: String, repository: String, commitId: String, loginUser: String, def createCommitComment(
content: String, fileName: Option[String], oldLine: Option[Int], newLine: Option[Int], owner: String,
issueId: Option[Int])(implicit s: Session): Int = repository: String,
commitId: String,
loginUser: String,
content: String,
fileName: Option[String],
oldLine: Option[Int],
newLine: Option[Int],
issueId: Option[Int]
)(implicit s: Session): Int =
CommitComments returning CommitComments.map(_.commentId) insert CommitComment( CommitComments returning CommitComments.map(_.commentId) insert CommitComment(
userName = owner, userName = owner,
repositoryName = repository, repositoryName = repository,
@@ -34,21 +49,75 @@ trait CommitsService {
newLine = newLine, newLine = newLine,
registeredDate = currentDate, registeredDate = currentDate,
updatedDate = currentDate, updatedDate = currentDate,
issueId = issueId) issueId = issueId,
originalCommitId = commitId,
originalOldLine = oldLine,
originalNewLine = newLine
)
def updateCommitCommentPosition(commentId: Int, commitId: String, oldLine: Option[Int], newLine: Option[Int])(implicit s: Session): Unit = def updateCommitCommentPosition(commentId: Int, commitId: String, oldLine: Option[Int], newLine: Option[Int])(
CommitComments.filter(_.byPrimaryKey(commentId)) implicit s: Session
): Unit =
CommitComments
.filter(_.byPrimaryKey(commentId))
.map { t => .map { t =>
(t.commitId, t.oldLine, t.newLine) (t.commitId, t.oldLine, t.newLine)
}.update(commitId, oldLine, newLine) }
.update(commitId, oldLine, newLine)
def updateCommitComment(commentId: Int, content: String)(implicit s: Session) = { def updateCommitComment(commentId: Int, content: String)(implicit s: Session) = {
CommitComments CommitComments
.filter (_.byPrimaryKey(commentId)) .filter(_.byPrimaryKey(commentId))
.map { t => (t.content, t.updatedDate) } .map { t =>
.update (content, currentDate) (t.content, t.updatedDate)
}
.update(content, currentDate)
} }
def deleteCommitComment(commentId: Int)(implicit s: Session) = def deleteCommitComment(commentId: Int)(implicit s: Session) =
CommitComments filter (_.byPrimaryKey(commentId)) delete CommitComments filter (_.byPrimaryKey(commentId)) delete
def saveCommitCommentDiff(
owner: String,
repository: String,
commitId: String,
fileName: String,
oldLine: Option[Int],
newLine: Option[Int],
diffJson: String
): Unit = {
val dir = new File(getDiffDir(owner, repository), FileUtil.checkFilename(commitId))
if (!dir.exists) {
dir.mkdirs()
}
val file = diffFile(dir, fileName, oldLine, newLine)
FileUtils.write(file, diffJson, "UTF-8")
}
def loadCommitCommentDiff(
owner: String,
repository: String,
commitId: String,
fileName: String,
oldLine: Option[Int],
newLine: Option[Int]
): Option[String] = {
val dir = new File(getDiffDir(owner, repository), FileUtil.checkFilename(commitId))
val file = diffFile(dir, fileName, oldLine, newLine)
if (file.exists) {
Option(FileUtils.readFileToString(file, "UTF-8"))
} else None
}
private def diffFile(dir: java.io.File, fileName: String, oldLine: Option[Int], newLine: Option[Int]): File = {
new File(
dir,
StringUtil.sha1(
fileName +
"_oldLine:" + oldLine.map(_.toString).getOrElse("") +
"_newLine:" + newLine.map(_.toString).getOrElse("")
)
)
}
} }

Some files were not shown because too many files have changed in this diff Show More