Discontinue draft workflow

Migration is implemented to either replace draft changes with private
(default) or work-in-progress changes.

We bump the change index version to trigger online reindexing.
That's needed because we don't support reindexing during init step, and
we need to reindex all changes, because DRAFT is removed from the
Change.Status enum and index still contains the old references to the
changes that were migrated to private or work-in-progress changes. To
make online reindexing work, draft status is replaced with the new
change status.

PolyGerrit UI is not addressed in this change and will be done in a
follow-up change.

User branches are not updated in this change to clean-up My-Menu entry
in user preferences. This will be done in a follow-up change.

refs/meta/config is not updated in this change to clean-up draft
related permissions. This will be done in a follow-up change.

ChangeControl#isPatchVisible() call sites are migrated to the
ChangeControl#isVisible() and isVisible() is made public again. This and
other temporary changes to ChangeControl class will be cleaned up in
follow up changes.

Bug: Issue 6880
Change-Id: Icfcb34efe1ff0ea1d39e94ed500db776f5770d8f
This commit is contained in:
David Ostrovsky 2017-02-13 21:16:58 +01:00 committed by Changcheng Xiao
parent 80522c3e22
commit 6ffb7d9358
129 changed files with 237 additions and 2908 deletions
Documentation
gerrit-acceptance-framework/src/test/java/com/google/gerrit/acceptance
gerrit-acceptance-tests/src/test/java/com/google/gerrit/acceptance
gerrit-common/src/main/java/com/google/gerrit/common
gerrit-extension-api/src/main/java/com/google/gerrit/extensions
gerrit-gwtui-common/src/main/java/com/google/gerrit/client/info
gerrit-gwtui/src/main/java/com/google/gerrit/client
gerrit-reviewdb/src/main/java/com/google/gerrit/reviewdb/client
gerrit-server/src/main/java/com/google/gerrit/server

@ -405,23 +405,6 @@ Further documentation on how to push can be found on the
link:user-upload.html#push_create[Upload changes] page. link:user-upload.html#push_create[Upload changes] page.
==== refs/publish/*
`+refs/publish/*+` is an alternative name to `+refs/for/*+` when pushing new changes
and patch sets.
==== refs/drafts/*
Draft workflow is slated for removal. Consider using
link:intro-user.html#private-changes[private changes] or
link:user-upload.html#wip[work-in-progress changes] instead.
Before the remove process is complete, pushing with `+refs/drafts/*+` to create a new
draft change will get a link:intro-user.html#private-changes[private change] instead.
Pushing with `+refs/drafts/*+` to an existing change to create a draft patch set will
get a link:user-upload.html#[change_edit][change edit] instead.
[[access_categories]] [[access_categories]]
== Access Categories == Access Categories
@ -429,7 +412,6 @@ Gerrit has several permission categories that can be granted to groups
within projects, enabling functionality for that group's members. within projects, enabling functionality for that group's members.
[[category_abandon]] [[category_abandon]]
=== Abandon === Abandon
@ -852,36 +834,6 @@ The change owner and any explicitly added reviewers can always see
private changes (even without having the `View Private Changes` access private changes (even without having the `View Private Changes` access
right assigned). right assigned).
[[category_view_drafts]]
=== View Drafts
This category permits users to view draft changes uploaded by other
users.
The change owner and any explicitly added reviewers can always see
draft changes (even without having the `View Drafts` access right
assigned).
[[category_publish_drafts]]
=== Publish Drafts
This category permits users to publish draft changes uploaded by other
users.
The change owner can always publish draft changes (even without having
the `Publish Drafts` access right assigned).
[[category_delete_drafts]]
=== Delete Drafts
This category permits users to delete draft changes uploaded by other
users.
The change owner can always delete draft changes (even without having
the `Delete Drafts` access right assigned).
[[category_delete_own_changes]] [[category_delete_own_changes]]
=== Delete Own Changes === Delete Own Changes
@ -949,13 +901,7 @@ Suggested access rights to grant:
If it's desired to have the possibility to upload temporarily hidden If it's desired to have the possibility to upload temporarily hidden
changes there's a specific permission for that. This enables someone changes there's a specific permission for that. This enables someone
to add specific reviewers for early feedback before making the change to add specific reviewers for early feedback before making the change
publicly visible. If you want to allow others than the owners to publicly visible.
publish a draft you also need to grant them `Publish Drafts`.
Optional access rights to grant:
* xref:category_push[`Push`] to 'refs/drafts/*'
* xref:category_publish_drafts[`Publish Drafts`] to 'refs/heads/*'
[[examples_developer]] [[examples_developer]]

@ -109,16 +109,6 @@ branch.
(option is mutually exclusive with --abandon, --publish --delete, --rebase (option is mutually exclusive with --abandon, --publish --delete, --rebase
and --json) and --json)
--publish::
Publish the specified draft patch set(s).
(option is mutually exclusive with --submit, --restore, --abandon, --delete
and --json)
--delete::
Delete the specified draft patch set(s).
(option is mutually exclusive with --submit, --restore, --abandon, --publish,
--rebase and --json)
--code-review:: --code-review::
--verified:: --verified::
Set the label to the value 'N'. The exact option names Set the label to the value 'N'. The exact option names

@ -46,7 +46,7 @@ Only subscribe to specific event types:
---- ----
$ ssh -p 29418 review.example.com gerrit stream-events \ $ ssh -p 29418 review.example.com gerrit stream-events \
-s draft-published -s patchset-created -s ref-replicated -s patchset-created -s ref-replicated
---- ----
== SCHEMA == SCHEMA
@ -143,21 +143,6 @@ comment:: Review comment cover message.
eventCreatedOn:: Time in seconds since the UNIX epoch when this event was eventCreatedOn:: Time in seconds since the UNIX epoch when this event was
created. created.
=== Draft Published
Sent when a draft change has been published.
type:: "draft-published"
change:: link:json.html#change[change attribute]
patchSet:: link:json.html#patchSet[patchSet attribute]
uploader:: link:json.html#account[account attribute]
eventCreatedOn:: Time in seconds since the UNIX epoch when this event was
created.
=== Dropped Output === Dropped Output
Sent to notify a client that events have been dropped. Sent to notify a client that events have been dropped.
@ -201,11 +186,6 @@ created.
Sent when a new change has been uploaded, or a new patch set has been uploaded Sent when a new change has been uploaded, or a new patch set has been uploaded
to an existing change. to an existing change.
Note that this event is also sent for changes or patch sets uploaded as draft,
but is only visible to the change owner, any existing reviewers, and users who
belong to a group that is granted the
link:access-control.html#category_view_drafts[View Drafts] capability.
type:: "patchset-created" type:: "patchset-created"
change:: link:json.html#change[change attribute] change:: link:json.html#change[change attribute]

@ -3491,10 +3491,9 @@ operations when multiple changes are impacted at once.
[[receive.checkMagicRefs]]receive.checkMagicRefs:: [[receive.checkMagicRefs]]receive.checkMagicRefs::
+ +
If true, Gerrit will verify the destination repository has If true, Gerrit will verify the destination repository has
no references under the magic 'refs/drafts', 'refs/for', or no references under the magic 'refs/for' branch namespace. Names under
'refs/publish' branch namespaces. Names under these locations these locations confuse clients when trying to upload code reviews so
confuse clients when trying to upload code reviews so Gerrit Gerrit requires them to be empty.
requires them to be empty.
+ +
If false Gerrit skips the sanity check and assumes administrators If false Gerrit skips the sanity check and assumes administrators
have ensured the repository does not contain any magic references. have ensured the repository does not contain any magic references.

@ -247,16 +247,6 @@ Documentation] |
link:https://gerrit.googlesource.com/plugins/emoticons/+doc/master/src/main/resources/Documentation/config.md[ link:https://gerrit.googlesource.com/plugins/emoticons/+doc/master/src/main/resources/Documentation/config.md[
Configuration] Configuration]
[[force-draft]]
=== force-draft
Provides an ssh command to force a change or patch set to draft status.
This is useful for administrators to be able to easily completely
delete a change or patch set from the server.
link:https://gerrit-review.googlesource.com/#/admin/projects/plugins/force-draft[
Project]
[[gitblit]] [[gitblit]]
=== gitblit === gitblit
@ -668,24 +658,6 @@ Documentation] |
link:https://gerrit.googlesource.com/plugins/websession-flatfile/+doc/master/src/main/resources/Documentation/config.md[ link:https://gerrit.googlesource.com/plugins/websession-flatfile/+doc/master/src/main/resources/Documentation/config.md[
Configuration] Configuration]
[[wip]]
=== wip
This plugin adds a new button that allows a change owner to set a
change to Work In Progress, and a button to change from WIP back to a
"Ready For Review" state.
Any change in the WIP state will not show up in anyone's Review
Requests. Pushing a new patchset will reset the change to Review In
Progress.
link:https://gerrit-review.googlesource.com/#/admin/projects/plugins/wip[
Project] |
link:https://gerrit.googlesource.com/plugins/wip/+doc/master/src/main/resources/Documentation/about.md[
Documentation] |
link:https://gerrit.googlesource.com/plugins/wip/+doc/master/src/main/resources/Documentation/config.md[
Configuration]
[[x-docs]] [[x-docs]]
=== x-docs === x-docs

@ -536,28 +536,6 @@ Alternatively, rather than completely ignoring the change, it can be muted.
Muting a change means it will always be marked as "reviewed" in dashboards, Muting a change means it will always be marked as "reviewed" in dashboards,
until a new patch set is uploaded. until a new patch set is uploaded.
[[drafts]]
== Working with Drafts
Drafts will be deprecated and removed soon. Consider using
link:#private-changes[private changes] or
link:user-upload.html#wip[work-in-progress changes] instead.
Changes cannot be uploaded as drafts any more, but existing draft changes
and patch sets can still be published and deleted.
By default draft changes are only visible to the change owner. This gives
you the possibility to have some staging before making your changes visible
to the reviewers. Draft changes can also be used to backup unfinished changes.
Draft changes have the state link:user-review-ui.html#draft[Draft] and
can be link:user-review-ui.html#publish[published] or
link:user-review-ui.html#delete[deleted] from the change screen.
By link:user-review-ui.html#reviewers[adding reviewers] to a draft
change the change is made visible to these users. This way you can
collaborate with other users in privacy.
[[inline-edit]] [[inline-edit]]
== Inline Edit == Inline Edit

@ -41,8 +41,6 @@ status:: Current state of this change.
NEW;; Change is still being reviewed. NEW;; Change is still being reviewed.
DRAFT;; Change is a draft change that only consists of draft patchsets.
MERGED;; Change has been merged to its branch. MERGED;; Change has been merged to its branch.
ABANDONED;; Change was abandoned by its owner or administrator. ABANDONED;; Change was abandoned by its owner or administrator.
@ -108,8 +106,6 @@ author:: Author of this patchset in <<account,account attribute>>.
createdOn:: Time in seconds since the UNIX epoch when this patchset createdOn:: Time in seconds since the UNIX epoch when this patchset
was created. was created.
isDraft:: Whether or not the patch set is a draft patch set.
kind:: Kind of change uploaded. kind:: Kind of change uploaded.
REWORK;; Nontrivial content changes. REWORK;; Nontrivial content changes.

@ -1255,10 +1255,6 @@ any account.
"url": "#/dashboard/self", "url": "#/dashboard/self",
"name": "Changes" "name": "Changes"
}, },
{
"url": "#/q/owner:self+is:draft",
"name": "Drafts"
},
{ {
"url": "#/q/has:draft", "url": "#/q/has:draft",
"name": "Draft Comments" "name": "Draft Comments"
@ -1313,10 +1309,6 @@ link:#preferences-input[PreferencesInput] entity.
"url": "#/dashboard/self", "url": "#/dashboard/self",
"name": "Changes" "name": "Changes"
}, },
{
"url": "#/q/owner:self+is:draft",
"name": "Drafts"
},
{ {
"url": "#/q/has:draft", "url": "#/q/has:draft",
"name": "Draft Comments" "name": "Draft Comments"
@ -1369,10 +1361,6 @@ link:#preferences-info[PreferencesInfo] entity.
"url": "#/dashboard/self", "url": "#/dashboard/self",
"name": "Changes" "name": "Changes"
}, },
{
"url": "#/q/owner:self+is:draft",
"name": "Drafts"
},
{ {
"url": "#/q/has:draft", "url": "#/q/has:draft",
"name": "Draft Comments" "name": "Draft Comments"

@ -26,7 +26,7 @@ request body.
"subject" : "Let's support 100% Gerrit workflow direct in browser", "subject" : "Let's support 100% Gerrit workflow direct in browser",
"branch" : "master", "branch" : "master",
"topic" : "create-change-in-browser", "topic" : "create-change-in-browser",
"status" : "DRAFT" "status" : "NEW"
} }
---- ----
@ -47,7 +47,7 @@ the resulting change.
"topic": "create-change-in-browser", "topic": "create-change-in-browser",
"change_id": "I8473b95934b5732ac55d26311a706c9c2bde9941", "change_id": "I8473b95934b5732ac55d26311a706c9c2bde9941",
"subject": "Let's support 100% Gerrit workflow direct in browser", "subject": "Let's support 100% Gerrit workflow direct in browser",
"status": "DRAFT", "status": "NEW",
"created": "2014-05-05 07:15:44.639000000", "created": "2014-05-05 07:15:44.639000000",
"updated": "2014-05-05 07:15:44.639000000", "updated": "2014-05-05 07:15:44.639000000",
"mergeable": true, "mergeable": true,
@ -1838,25 +1838,6 @@ message if the set of changes to be submitted with this change
includes changes the caller cannot read. includes changes the caller cannot read.
[[publish-draft-change]]
=== Publish Draft Change
--
'POST /changes/link:#change-id[\{change-id\}]/publish'
--
Publishes a draft change.
.Request
----
POST /changes/myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9940/publish HTTP/1.0
Content-Type: application/json; charset=UTF-8
----
.Response
----
HTTP/1.1 204 No Content
----
[[delete-change]] [[delete-change]]
=== Delete Change === Delete Change
-- --
@ -1869,10 +1850,6 @@ New or abandoned changes can be deleted by their owner if the user is granted
the link:access-control.html#category_delete_own_changes[Delete Own Changes] permission, the link:access-control.html#category_delete_own_changes[Delete Own Changes] permission,
otherwise only by administrators. otherwise only by administrators.
Draft changes can only be deleted by their owner or other users who have the
permissions to view and delete drafts. If the draft workflow is disabled, only
administrators with those permissions may delete draft changes.
.Request .Request
---- ----
DELETE /changes/myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9940 HTTP/1.0 DELETE /changes/myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9940 HTTP/1.0
@ -4017,44 +3994,6 @@ message is contained in the response body.
"revision 674ac754f91e64a0efb8087e59a176484bd534d1 is not current revision" "revision 674ac754f91e64a0efb8087e59a176484bd534d1 is not current revision"
---- ----
[[publish-draft-revision]]
=== Publish Draft Revision
--
'POST /changes/link:#change-id[\{change-id\}]/revisions/link:#revision-id[\{revision-id\}]/publish'
--
Publishes a draft revision.
.Request
----
POST /changes/myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9940/revisions/current/publish HTTP/1.0
Content-Type: application/json; charset=UTF-8
----
.Response
----
HTTP/1.1 204 No Content
----
[[delete-draft-revision]]
=== Delete Draft Revision
--
'DELETE /changes/link:#change-id[\{change-id\}]/revisions/link:#revision-id[\{revision-id\}]'
--
Deletes a draft revision.
.Request
----
DELETE /changes/myProject~master~I8473b95934b5732ac55d26311a706c9c2bde9940/revisions/674ac754f91e64a0efb8087e59a176484bd534d1 HTTP/1.0
Content-Type: application/json; charset=UTF-8
----
.Response
----
HTTP/1.1 204 No Content
----
[[get-patch]] [[get-patch]]
=== Get Patch === Get Patch
-- --
@ -5682,7 +5621,7 @@ is enabled).
|`subject` || |`subject` ||
The subject of the change (header line of the commit message). The subject of the change (header line of the commit message).
|`status` || |`status` ||
The status of the change (`NEW`, `MERGED`, `ABANDONED`, `DRAFT`). The status of the change (`NEW`, `MERGED`, `ABANDONED`).
|`created` || |`created` ||
The link:rest-api.html#timestamp[timestamp] of when the change was The link:rest-api.html#timestamp[timestamp] of when the change was
created. created.
@ -6657,7 +6596,7 @@ link:#commit-info[CommitInfo] entity.
|`_revision_number` |optional|The revision number. |`_revision_number` |optional|The revision number.
|`_current_revision_number`|optional|The current revision number. |`_current_revision_number`|optional|The current revision number.
|`status` |optional|The status of the change. The status of |`status` |optional|The status of the change. The status of
the change is one of (`NEW`, `MERGED`, `ABANDONED`, `DRAFT`). the change is one of (`NEW`, `MERGED`, `ABANDONED`).
|=========================== |===========================
[[related-changes-info]] [[related-changes-info]]
@ -6881,7 +6820,6 @@ link:#list-changes[Query Changes].
[options="header",cols="1,^1,5"] [options="header",cols="1,^1,5"]
|=========================== |===========================
|Field Name ||Description |Field Name ||Description
|`draft` |not set if `false`|Whether the patch set is a draft.
|`kind` ||The change kind. Valid values are `REWORK`, `TRIVIAL_REBASE`, |`kind` ||The change kind. Valid values are `REWORK`, `TRIVIAL_REBASE`,
`MERGE_FIRST_PARENT_UPDATE`, `NO_CODE_CHANGE`, and `NO_CHANGE`. `MERGE_FIRST_PARENT_UPDATE`, `NO_CODE_CHANGE`, and `NO_CHANGE`.
|`_number` ||The patch set number. |`_number` ||The patch set number.

@ -1021,10 +1021,6 @@ PreferencesInfo] is returned.
"url": "#/dashboard/self", "url": "#/dashboard/self",
"name": "Changes" "name": "Changes"
}, },
{
"url": "#/q/owner:self+is:draft",
"name": "Drafts"
},
{ {
"url": "#/q/has:draft", "url": "#/q/has:draft",
"name": "Draft Comments" "name": "Draft Comments"
@ -1104,10 +1100,6 @@ PreferencesInfo] is returned.
"url": "#/dashboard/self", "url": "#/dashboard/self",
"name": "Changes" "name": "Changes"
}, },
{
"url": "#/q/owner:self+is:draft",
"name": "Drafts"
},
{ {
"url": "#/q/has:draft", "url": "#/q/has:draft",
"name": "Draft Comments" "name": "Draft Comments"
@ -1396,9 +1388,6 @@ section.
|`allow_blame` |not set if `false`| |`allow_blame` |not set if `false`|
link:config-gerrit.html#change.allowBlame[Whether blame on side by side diff is link:config-gerrit.html#change.allowBlame[Whether blame on side by side diff is
allowed]. allowed].
|`allow_drafts` |not set if `false`|
link:config-gerrit.html#change.allowDrafts[Whether draft workflow is
allowed].
|`large_change` || |`large_change` ||
link:config-gerrit.html#change.largeChange[Number of changed lines from link:config-gerrit.html#change.largeChange[Number of changed lines from
which on a change is considered as a large change]. which on a change is considered as a large change].

@ -70,10 +70,10 @@ For more details, see link:cmd-hook-commit-msg.html[commit-msg].
Change Upload Change Upload
-------------- --------------
During upload by pushing to `+refs/for/*+`, `+refs/drafts/*+` or During upload by pushing to `+refs/for/*+` or `+refs/heads/*+`,
`+refs/heads/*+`, Gerrit will try to find an existing review the Gerrit will try to find an existing review the uploaded commit
uploaded commit relates to. For an existing review to match, the relates to. For an existing review to match, the following properties
following properties have to match: have to match:
* Change-Id * Change-Id
* Repository name * Repository name
@ -104,7 +104,7 @@ message if additional revisions to a change are required.
By default, Gerrit will prevent pushing for review if no Change-Id is provided, By default, Gerrit will prevent pushing for review if no Change-Id is provided,
with the following message: with the following message:
! [remote rejected] HEAD -> refs/publish/master (missing Change-Id in commit ! [remote rejected] HEAD -> refs/for/master (missing Change-Id in commit
message footer) message footer)
However, repositories can be configured to allow commits without Change-Ids However, repositories can be configured to allow commits without Change-Ids

@ -61,13 +61,6 @@ The change was successfully merged into the destination branch.
+ +
The change was abandoned. The change was abandoned.
- [[draft]]`Draft`:
+
The change is a draft that is only visible to the change owner, the
reviewers that were explicitly added to the change, and users who have
the link:access-control.html#category_view_drafts[View Drafts] global
capability assigned.
[[commit-info]] [[commit-info]]
=== Commit Info Block === Commit Info Block
@ -258,28 +251,14 @@ open changes.
Users can only cherry-pick changes to branches for which they are Users can only cherry-pick changes to branches for which they are
allowed to upload changes for review. allowed to upload changes for review.
** [[publish]]`Publish`:
+
Publishes the currently viewed draft patch set. If this is the first
patch set of a change that is published, the change will be published
as well.
+
The `Publish` button is only available if a draft patch set is viewed
and the user is the change owner or has the
link:access-control.html#category_publish_drafts[Publish Drafts] access
right assigned.
** [[delete]]`Delete Change` / `Delete Revision`: ** [[delete]]`Delete Change` / `Delete Revision`:
+ +
Deletes the change / the currently viewed draft patch set. Deletes the change.
+ +
For open or abandoned changes, the `Delete Change` button will be available For open or abandoned changes, the `Delete Change` button will be available
and if the user is the change owner and is granted the and if the user is the change owner and is granted the
link:access-control.html#category_delete_own_changes[Delete Own Changes] link:access-control.html#category_delete_own_changes[Delete Own Changes]
permission or if they are an administrator. For draft changes, permission or if they are an administrator.
the `Delete Change` / `Delete Revision` buttons will be available if the user is
the change owner or has the
link:access-control.html#category_delete_drafts[Delete Drafts] access right assigned.
** [[plugin-actions]]Further actions may be available if plugins are installed. ** [[plugin-actions]]Further actions may be available if plugins are installed.

@ -14,7 +14,6 @@ matches the search, the change will be presented instead of a list.
|All > Open | status:open '(or is:open)' |All > Open | status:open '(or is:open)'
|All > Merged | status:merged |All > Merged | status:merged
|All > Abandoned | status:abandoned |All > Abandoned | status:abandoned
|My > Drafts | owner:self is:draft
|My > Watched Changes | status:open is:watched |My > Watched Changes | status:open is:watched
|My > Starred Changes | is:starred |My > Starred Changes | is:starred
|My > Draft Comments | has:draft |My > Draft Comments | has:draft
@ -337,10 +336,6 @@ is:open, is:pending::
+ +
True if the change is open. True if the change is open.
is:draft::
+
True if the change is a draft.
is:closed:: is:closed::
+ +
True if the change is either merged or abandoned. True if the change is either merged or abandoned.

@ -18,7 +18,6 @@ import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.TruthJUnit.assume; import static com.google.common.truth.TruthJUnit.assume;
import static com.google.gerrit.acceptance.GitUtil.initSsh; import static com.google.gerrit.acceptance.GitUtil.initSsh;
import static com.google.gerrit.extensions.api.changes.SubmittedTogetherOption.NON_VISIBLE_CHANGES; import static com.google.gerrit.extensions.api.changes.SubmittedTogetherOption.NON_VISIBLE_CHANGES;
import static com.google.gerrit.extensions.client.ListChangesOption.ALL_REVISIONS;
import static com.google.gerrit.reviewdb.client.Patch.COMMIT_MSG; import static com.google.gerrit.reviewdb.client.Patch.COMMIT_MSG;
import static com.google.gerrit.reviewdb.client.Patch.MERGE_LIST; import static com.google.gerrit.reviewdb.client.Patch.MERGE_LIST;
import static com.google.gerrit.server.group.SystemGroupBackend.REGISTERED_USERS; import static com.google.gerrit.server.group.SystemGroupBackend.REGISTERED_USERS;
@ -35,7 +34,6 @@ import com.google.common.jimfs.Jimfs;
import com.google.common.primitives.Chars; import com.google.common.primitives.Chars;
import com.google.gerrit.acceptance.AcceptanceTestRequestScope.Context; import com.google.gerrit.acceptance.AcceptanceTestRequestScope.Context;
import com.google.gerrit.common.Nullable; import com.google.gerrit.common.Nullable;
import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.common.data.AccessSection; import com.google.gerrit.common.data.AccessSection;
import com.google.gerrit.common.data.ContributorAgreement; import com.google.gerrit.common.data.ContributorAgreement;
import com.google.gerrit.common.data.GroupReference; import com.google.gerrit.common.data.GroupReference;
@ -50,7 +48,6 @@ import com.google.gerrit.extensions.api.groups.GroupInput;
import com.google.gerrit.extensions.api.projects.BranchApi; import com.google.gerrit.extensions.api.projects.BranchApi;
import com.google.gerrit.extensions.api.projects.BranchInput; import com.google.gerrit.extensions.api.projects.BranchInput;
import com.google.gerrit.extensions.api.projects.ProjectInput; import com.google.gerrit.extensions.api.projects.ProjectInput;
import com.google.gerrit.extensions.client.ChangeStatus;
import com.google.gerrit.extensions.client.InheritableBoolean; import com.google.gerrit.extensions.client.InheritableBoolean;
import com.google.gerrit.extensions.client.ListChangesOption; import com.google.gerrit.extensions.client.ListChangesOption;
import com.google.gerrit.extensions.client.ProjectWatchInfo; import com.google.gerrit.extensions.client.ProjectWatchInfo;
@ -60,14 +57,11 @@ import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.ChangeType; import com.google.gerrit.extensions.common.ChangeType;
import com.google.gerrit.extensions.common.DiffInfo; import com.google.gerrit.extensions.common.DiffInfo;
import com.google.gerrit.extensions.common.EditInfo; import com.google.gerrit.extensions.common.EditInfo;
import com.google.gerrit.extensions.common.RevisionInfo;
import com.google.gerrit.extensions.restapi.BinaryResult; import com.google.gerrit.extensions.restapi.BinaryResult;
import com.google.gerrit.extensions.restapi.IdString; import com.google.gerrit.extensions.restapi.IdString;
import com.google.gerrit.extensions.restapi.RestApiException; import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.reviewdb.client.AccountGroup; import com.google.gerrit.reviewdb.client.AccountGroup;
import com.google.gerrit.reviewdb.client.Branch; import com.google.gerrit.reviewdb.client.Branch;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.Change.Status;
import com.google.gerrit.reviewdb.client.PatchSet; import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.Project; import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.reviewdb.client.RefNames; import com.google.gerrit.reviewdb.client.RefNames;
@ -104,14 +98,11 @@ import com.google.gerrit.server.mail.send.EmailHeader;
import com.google.gerrit.server.notedb.ChangeNoteUtil; import com.google.gerrit.server.notedb.ChangeNoteUtil;
import com.google.gerrit.server.notedb.ChangeNotes; import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.notedb.MutableNotesMigration; import com.google.gerrit.server.notedb.MutableNotesMigration;
import com.google.gerrit.server.notedb.PatchSetState;
import com.google.gerrit.server.project.ProjectCache; import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.project.Util; import com.google.gerrit.server.project.Util;
import com.google.gerrit.server.query.change.ChangeData; import com.google.gerrit.server.query.change.ChangeData;
import com.google.gerrit.server.query.change.InternalChangeQuery; import com.google.gerrit.server.query.change.InternalChangeQuery;
import com.google.gerrit.server.update.BatchUpdate; import com.google.gerrit.server.update.BatchUpdate;
import com.google.gerrit.server.update.BatchUpdateOp;
import com.google.gerrit.server.update.ChangeContext;
import com.google.gerrit.testutil.ConfigSuite; import com.google.gerrit.testutil.ConfigSuite;
import com.google.gerrit.testutil.FakeEmailSender; import com.google.gerrit.testutil.FakeEmailSender;
import com.google.gerrit.testutil.FakeEmailSender.Message; import com.google.gerrit.testutil.FakeEmailSender.Message;
@ -307,16 +298,6 @@ public abstract class AbstractDaemonTest {
return cfg; return cfg;
} }
protected static Config allowDraftsDisabledConfig() {
Config cfg = new Config();
cfg.setBoolean("change", null, "allowDrafts", false);
return cfg;
}
protected boolean isAllowDrafts() {
return cfg.getBoolean("change", "allowDrafts", true);
}
protected boolean isSubmitWholeTopicEnabled() { protected boolean isSubmitWholeTopicEnabled() {
return cfg.getBoolean("change", null, "submitWholeTopic", false); return cfg.getBoolean("change", null, "submitWholeTopic", false);
} }
@ -638,139 +619,6 @@ public abstract class AbstractDaemonTest {
repo, "refs/for/master/" + name(topic), commitMsg, fileName, content); repo, "refs/for/master/" + name(topic), commitMsg, fileName, content);
} }
protected PushOneCommit.Result createDraftChange() throws Exception {
return createDraftChange("");
}
protected PushOneCommit.Result createDraftChange(String topic) throws Exception {
PushOneCommit.Result r =
Strings.isNullOrEmpty(topic)
? createChange()
: createChangeWithTopic(testRepo, topic, "message", "a.txt", "content");
markChangeAsDraft(r.getChange().change().getId());
setCurrentPatchSetAsDraft(r.getChange().change().getId());
ChangeData cd = Iterables.getOnlyElement(queryProvider.get().byKeyPrefix(r.getChangeId()));
assertThat(cd.change().getStatus()).isEqualTo(Status.DRAFT);
return r;
}
protected String createDraftChangeWith2PS() throws Exception {
PushOneCommit.Result r = createDraftChange();
amendChangeAndMarkChangeAndPatchSetsAsDraft(r.getChangeId());
return r.getChangeId();
}
protected void markChangeAsDraft(Change.Id id) throws Exception {
markChangeAsDraft(project, id);
}
protected void markChangeAsDraft(Project.NameKey project, Change.Id id) throws Exception {
try (BatchUpdate batchUpdate =
batchUpdateFactory.create(db, project, atrScope.get().getUser(), TimeUtil.nowTs())) {
batchUpdate.addOp(id, new MarkChangeAsDraftUpdateOp());
batchUpdate.execute();
}
ChangeStatus changeStatus = gApi.changes().id(id.get()).get().status;
assertThat(changeStatus).isEqualTo(ChangeStatus.DRAFT);
}
protected void setCurrentPatchSetAsDraft(Change.Id id) throws Exception {
try (BatchUpdate batchUpdate =
batchUpdateFactory.create(db, project, atrScope.get().getUser(), TimeUtil.nowTs())) {
batchUpdate.addOp(id, new SetDraftStatusOfCurrentPatchSetOp(true));
batchUpdate.execute();
}
ChangeInfo changeInfo =
gApi.changes().id(id.get()).get(EnumSet.of(ListChangesOption.ALL_REVISIONS));
RevisionInfo revisionInfo = changeInfo.revisions.get(changeInfo.currentRevision);
assertThat(revisionInfo.draft).isEqualTo(Boolean.TRUE);
}
protected void setDraftStatusOfPatchSets(Change.Id id, boolean draftStatus) throws Exception {
try (BatchUpdate batchUpdate =
batchUpdateFactory.create(db, project, atrScope.get().getUser(), TimeUtil.nowTs())) {
batchUpdate.addOp(id, new DraftStatusOfPatchSetsUpdateOp(draftStatus));
batchUpdate.execute();
}
Boolean expectedDraftStatus = draftStatus ? Boolean.TRUE : null;
List<Boolean> patchSetDraftStatuses = getPatchSetDraftStatuses(id);
patchSetDraftStatuses.forEach(status -> assertThat(status).isEqualTo(expectedDraftStatus));
}
private List<Boolean> getPatchSetDraftStatuses(Change.Id id) throws Exception {
Collection<RevisionInfo> revisionInfos =
gApi.changes().id(id.get()).get(ALL_REVISIONS).revisions.values();
return revisionInfos.stream().map(revisionInfo -> revisionInfo.draft).collect(toList());
}
private static class MarkChangeAsDraftUpdateOp implements BatchUpdateOp {
@Override
public boolean updateChange(ChangeContext ctx) throws Exception {
Change change = ctx.getChange();
// Change status in database.
change.setStatus(Change.Status.DRAFT);
// Change status in NoteDb.
PatchSet.Id currentPatchSetId = change.currentPatchSetId();
ctx.getUpdate(currentPatchSetId).setStatus(Change.Status.DRAFT);
return true;
}
}
private class SetDraftStatusOfCurrentPatchSetOp implements BatchUpdateOp {
private final boolean draftStatus;
SetDraftStatusOfCurrentPatchSetOp(boolean draftStatus) {
this.draftStatus = draftStatus;
}
@Override
public boolean updateChange(ChangeContext ctx) throws Exception {
PatchSet currentPatchSet = psUtil.current(db, ctx.getNotes());
// Change status in ReviewDb.
currentPatchSet.setDraft(draftStatus);
db.patchSets().update(Collections.singleton(currentPatchSet));
// Change status in NoteDb.
PatchSetState patchSetState = draftStatus ? PatchSetState.DRAFT : PatchSetState.PUBLISHED;
ctx.getUpdate(currentPatchSet.getId()).setPatchSetState(patchSetState);
return true;
}
}
private class DraftStatusOfPatchSetsUpdateOp implements BatchUpdateOp {
private final boolean draftStatus;
DraftStatusOfPatchSetsUpdateOp(boolean draftStatus) {
this.draftStatus = draftStatus;
}
@Override
public boolean updateChange(ChangeContext ctx) throws Exception {
Collection<PatchSet> patchSets = psUtil.byChange(db, ctx.getNotes());
// Change status in database.
patchSets.forEach(patchSet -> patchSet.setDraft(draftStatus));
db.patchSets().update(patchSets);
// Change status in NoteDb.
PatchSetState patchSetState = draftStatus ? PatchSetState.DRAFT : PatchSetState.PUBLISHED;
patchSets
.stream()
.map(PatchSet::getId)
.map(ctx::getUpdate)
.forEach(changeUpdate -> changeUpdate.setPatchSetState(patchSetState));
return true;
}
}
protected PushOneCommit.Result createWorkInProgressChange() throws Exception { protected PushOneCommit.Result createWorkInProgressChange() throws Exception {
return pushTo("refs/for/master%wip"); return pushTo("refs/for/master%wip");
} }
@ -864,23 +712,6 @@ public abstract class AbstractDaemonTest {
revision(r).submit(); revision(r).submit();
} }
protected PushOneCommit.Result amendChangeAndMarkPatchSetAsDraft(String changeId)
throws Exception {
PushOneCommit.Result r = amendChange(changeId, "refs/for/master");
r.assertOkStatus();
setCurrentPatchSetAsDraft(r.getChange().getId());
return r;
}
protected PushOneCommit.Result amendChangeAndMarkChangeAndPatchSetsAsDraft(String changeId)
throws Exception {
PushOneCommit.Result r = amendChange(changeId, "refs/for/master");
r.assertOkStatus();
markChangeAsDraft(r.getChange().getId());
setCurrentPatchSetAsDraft(r.getChange().change().getId());
return r;
}
protected ChangeInfo info(String id) throws RestApiException { protected ChangeInfo info(String id) throws RestApiException {
return gApi.changes().id(id).info(); return gApi.changes().id(id).info();
} }

@ -73,7 +73,7 @@ public class GeneralPreferencesIT extends AbstractDaemonTest {
public void getAndSetPreferences() throws Exception { public void getAndSetPreferences() throws Exception {
GeneralPreferencesInfo o = gApi.accounts().id(user42.id.toString()).getPreferences(); GeneralPreferencesInfo o = gApi.accounts().id(user42.id.toString()).getPreferences();
assertPrefs(o, GeneralPreferencesInfo.defaults(), "my", "changeTable"); assertPrefs(o, GeneralPreferencesInfo.defaults(), "my", "changeTable");
assertThat(o.my).hasSize(7); assertThat(o.my).hasSize(6);
assertThat(o.changeTable).isEmpty(); assertThat(o.changeTable).isEmpty();
GeneralPreferencesInfo i = GeneralPreferencesInfo.defaults(); GeneralPreferencesInfo i = GeneralPreferencesInfo.defaults();

@ -98,17 +98,6 @@ public class AbandonIT extends AbstractDaemonTest {
changeAbandoner.batchAbandon(batchUpdateFactory, new Project.NameKey(project1Name), user, list); changeAbandoner.batchAbandon(batchUpdateFactory, new Project.NameKey(project1Name), user, list);
} }
@Test
public void abandonDraft() throws Exception {
PushOneCommit.Result r = createDraftChange();
String changeId = r.getChangeId();
assertThat(info(changeId).status).isEqualTo(ChangeStatus.DRAFT);
exception.expect(ResourceConflictException.class);
exception.expectMessage("draft changes cannot be abandoned");
gApi.changes().id(changeId).abandon();
}
@Test @Test
@GerritConfig(name = "changeCleanup.abandonAfter", value = "1w") @GerritConfig(name = "changeCleanup.abandonAfter", value = "1w")
public void abandonInactiveOpenChanges() throws Exception { public void abandonInactiveOpenChanges() throws Exception {

@ -835,23 +835,6 @@ public class ChangeIT extends AbstractDaemonTest {
gApi.changes().id(changeId).rebase(); gApi.changes().id(changeId).rebase();
} }
@Test
public void publish() throws Exception {
PushOneCommit.Result r = createDraftChange();
assertThat(info(r.getChangeId()).status).isEqualTo(ChangeStatus.DRAFT);
gApi.changes().id(r.getChangeId()).publish();
assertThat(info(r.getChangeId()).status).isEqualTo(ChangeStatus.NEW);
}
@Test
public void deleteDraftChange() throws Exception {
PushOneCommit.Result r = createDraftChange();
assertThat(query(r.getChangeId())).hasSize(1);
assertThat(info(r.getChangeId()).status).isEqualTo(ChangeStatus.DRAFT);
gApi.changes().id(r.getChangeId()).delete();
assertThat(query(r.getChangeId())).isEmpty();
}
@Test @Test
public void deleteNewChangeAsAdmin() throws Exception { public void deleteNewChangeAsAdmin() throws Exception {
PushOneCommit.Result changeResult = createChange(); PushOneCommit.Result changeResult = createChange();
@ -2311,7 +2294,6 @@ public class ChangeIT extends AbstractDaemonTest {
gApi.changes().id(r1.getChangeId()).revision(r1.getCommit().name()).submit(); gApi.changes().id(r1.getChangeId()).revision(r1.getCommit().name()).submit();
createChange(); createChange();
createDraftChange();
setApiUser(user); setApiUser(user);
AcceptanceTestRequestScope.Context ctx = disableDb(); AcceptanceTestRequestScope.Context ctx = disableDb();
@ -2462,64 +2444,6 @@ public class ChangeIT extends AbstractDaemonTest {
gApi.changes().create(in).get(); gApi.changes().create(in).get();
} }
@Test
public void createNewPatchSetOnVisibleDraftPatchSet() throws Exception {
// Clone separate repositories of the same project as admin and as user
TestRepository<InMemoryRepository> adminTestRepo = cloneProject(project, admin);
TestRepository<InMemoryRepository> userTestRepo = cloneProject(project, user);
// Create change as admin
PushOneCommit push = pushFactory.create(db, admin.getIdent(), adminTestRepo);
PushOneCommit.Result r1 = push.to("refs/for/master");
r1.assertOkStatus();
// Amend draft as admin
PushOneCommit.Result r2 =
amendChange(r1.getChangeId(), "refs/for/master", admin, adminTestRepo);
r2.assertOkStatus();
setCurrentPatchSetAsDraft(r2.getChange().getId());
// Add user as reviewer to make this patch set visible
AddReviewerInput in = new AddReviewerInput();
in.reviewer = user.email;
gApi.changes().id(r1.getChangeId()).addReviewer(in);
// Fetch change
GitUtil.fetch(userTestRepo, r2.getPatchSet().getRefName() + ":ps");
userTestRepo.reset("ps");
// Amend change as user
PushOneCommit.Result r3 = amendChange(r2.getChangeId(), "refs/for/master", user, userTestRepo);
r3.assertOkStatus();
setCurrentPatchSetAsDraft(r3.getChange().getId());
}
@Test
public void createNewPatchSetOnInvisibleDraftPatchSet() throws Exception {
// Clone separate repositories of the same project as admin and as user
TestRepository<InMemoryRepository> adminTestRepo = cloneProject(project, admin);
TestRepository<InMemoryRepository> userTestRepo = cloneProject(project, user);
// Create change as admin
PushOneCommit push = pushFactory.create(db, admin.getIdent(), adminTestRepo);
PushOneCommit.Result r1 = push.to("refs/for/master");
r1.assertOkStatus();
// Amend draft as admin
PushOneCommit.Result r2 =
amendChange(r1.getChangeId(), "refs/for/master", admin, adminTestRepo);
r2.assertOkStatus();
setCurrentPatchSetAsDraft(r2.getChange().getId());
// Fetch change
GitUtil.fetch(userTestRepo, r1.getPatchSet().getRefName() + ":ps");
userTestRepo.reset("ps");
// Amend change as user
PushOneCommit.Result r3 = amendChange(r1.getChangeId(), "refs/for/master", user, userTestRepo);
r3.assertErrorStatus("cannot add patch set to " + r3.getChange().change().getChangeId() + ".");
}
@Test @Test
public void createNewPatchSetWithoutPermission() throws Exception { public void createNewPatchSetWithoutPermission() throws Exception {
// Create new project with clean permissions // Create new project with clean permissions
@ -2591,63 +2515,6 @@ public class ChangeIT extends AbstractDaemonTest {
r2.assertOkStatus(); r2.assertOkStatus();
} }
@Test
public void createNewPatchSetAsReviewerOnDraftChange() throws Exception {
// Clone separate repositories of the same project as admin and as user
TestRepository<?> adminTestRepo = cloneProject(project, admin);
TestRepository<?> userTestRepo = cloneProject(project, user);
// Create change as admin
PushOneCommit push = pushFactory.create(db, admin.getIdent(), adminTestRepo);
PushOneCommit.Result r1 = push.to("refs/for/master");
r1.assertOkStatus();
markChangeAsDraft(r1.getChange().getId());
// Add user as reviewer
AddReviewerInput in = new AddReviewerInput();
in.reviewer = user.email;
gApi.changes().id(r1.getChangeId()).addReviewer(in);
// Fetch change
GitUtil.fetch(userTestRepo, r1.getPatchSet().getRefName() + ":ps");
userTestRepo.reset("ps");
// Amend change as user
PushOneCommit.Result r2 = amendChange(r1.getChangeId(), "refs/for/master", user, userTestRepo);
r2.assertOkStatus();
}
@Test
public void createNewDraftPatchSetOnDraftChange() throws Exception {
// Create new project with clean permissions
Project.NameKey p = createProject("addPatchSet4");
// Clone separate repositories of the same project as admin and as user
TestRepository<?> adminTestRepo = cloneProject(p, admin);
TestRepository<?> userTestRepo = cloneProject(p, user);
// Block default permission
block(p, "refs/for/*", Permission.ADD_PATCH_SET, REGISTERED_USERS);
// Create change as admin
PushOneCommit push = pushFactory.create(db, admin.getIdent(), adminTestRepo);
PushOneCommit.Result r1 = push.to("refs/for/master");
r1.assertOkStatus();
markChangeAsDraft(p, r1.getChange().getId());
// Add user as reviewer
AddReviewerInput in = new AddReviewerInput();
in.reviewer = user.email;
gApi.changes().id(r1.getChangeId()).addReviewer(in);
// Fetch change
GitUtil.fetch(userTestRepo, r1.getPatchSet().getRefName() + ":ps");
userTestRepo.reset("ps");
// Amend change as user
PushOneCommit.Result r2 = amendChange(r1.getChangeId(), "refs/for/master", user, userTestRepo);
r2.assertErrorStatus("cannot add patch set to " + r1.getChange().getId().id + ".");
}
@Test @Test
public void createMergePatchSet() throws Exception { public void createMergePatchSet() throws Exception {
PushOneCommit.Result start = pushTo("refs/heads/master"); PushOneCommit.Result start = pushTo("refs/heads/master");

@ -301,12 +301,6 @@ public class RevisionIT extends AbstractDaemonTest {
gApi.changes().id(r.getChange().getId().get()).current().review(ReviewInput.approve()); gApi.changes().id(r.getChange().getId().get()).current().review(ReviewInput.approve());
} }
@Test
public void deleteDraft() throws Exception {
PushOneCommit.Result r = createDraftChange();
gApi.changes().id(r.getChangeId()).revision(r.getCommit().name()).delete();
}
@Test @Test
public void cherryPick() throws Exception { public void cherryPick() throws Exception {
PushOneCommit.Result r = pushTo("refs/for/master%topic=someTopic"); PushOneCommit.Result r = pushTo("refs/for/master%topic=someTopic");

@ -42,6 +42,7 @@ import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.GerritConfig; import com.google.gerrit.acceptance.GerritConfig;
import com.google.gerrit.acceptance.GitUtil; import com.google.gerrit.acceptance.GitUtil;
import com.google.gerrit.acceptance.PushOneCommit; import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.Sandboxed;
import com.google.gerrit.acceptance.TestAccount; import com.google.gerrit.acceptance.TestAccount;
import com.google.gerrit.acceptance.TestProjectInput; import com.google.gerrit.acceptance.TestProjectInput;
import com.google.gerrit.common.data.LabelType; import com.google.gerrit.common.data.LabelType;
@ -61,6 +62,7 @@ import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.ChangeMessageInfo; import com.google.gerrit.extensions.common.ChangeMessageInfo;
import com.google.gerrit.extensions.common.CommentInfo; import com.google.gerrit.extensions.common.CommentInfo;
import com.google.gerrit.extensions.common.EditInfo; import com.google.gerrit.extensions.common.EditInfo;
import com.google.gerrit.extensions.common.EditInfoSubject;
import com.google.gerrit.extensions.common.LabelInfo; import com.google.gerrit.extensions.common.LabelInfo;
import com.google.gerrit.extensions.common.RevisionInfo; import com.google.gerrit.extensions.common.RevisionInfo;
import com.google.gerrit.reviewdb.client.AccountGroup; import com.google.gerrit.reviewdb.client.AccountGroup;
@ -636,34 +638,6 @@ public abstract class AbstractPushForReview extends AbstractDaemonTest {
r.assertOkStatus(); r.assertOkStatus();
} }
@Test
public void pushForMasterAsDraft() throws Exception {
// create draft by pushing to 'refs/drafts/' will get a private change.
PushOneCommit.Result r = pushTo("refs/drafts/master");
r.assertOkStatus();
r.assertChange(Change.Status.NEW, null);
assertThat(gApi.changes().id(r.getChangeId()).get().isPrivate).isTrue();
// create draft by using 'draft' option will get a private change, too.
r = pushTo("refs/for/master%draft");
r.assertOkStatus();
r.assertChange(Change.Status.NEW, null);
assertThat(gApi.changes().id(r.getChangeId()).get().isPrivate).isTrue();
}
@Test
public void publishDraftChangeByPushingNonDraftPatchSet() throws Exception {
PushOneCommit.Result r = createDraftChange();
r.assertOkStatus();
r.assertChange(Change.Status.DRAFT, null);
// publish draft change by pushing non-draft patch set
r = amendChange(r.getChangeId(), "refs/for/master");
r.assertOkStatus();
r.assertChange(Change.Status.NEW, null);
}
@Test @Test
public void pushForMasterAsEdit() throws Exception { public void pushForMasterAsEdit() throws Exception {
PushOneCommit.Result r = pushTo("refs/for/master"); PushOneCommit.Result r = pushTo("refs/for/master");
@ -1819,6 +1793,42 @@ public abstract class AbstractPushForReview extends AbstractDaemonTest {
assertThat(getPublishedComments(r.getChangeId())).isEmpty(); assertThat(getPublishedComments(r.getChangeId())).isEmpty();
} }
@Test
public void pushDraftGetsPrivateChange() throws Exception {
String changeId1 = createChange("refs/drafts/master").getChangeId();
String changeId2 = createChange("refs/for/master%draft").getChangeId();
ChangeInfo info1 = gApi.changes().id(changeId1).get();
ChangeInfo info2 = gApi.changes().id(changeId2).get();
assertThat(info1.status).isEqualTo(ChangeStatus.NEW);
assertThat(info2.status).isEqualTo(ChangeStatus.NEW);
assertThat(info1.isPrivate).isEqualTo(true);
assertThat(info2.isPrivate).isEqualTo(true);
assertThat(info1.revisions).hasSize(1);
assertThat(info2.revisions).hasSize(1);
}
@Sandboxed
@Test
public void pushWithDraftOptionToExistingNewChangeGetsChangeEdit() throws Exception {
String changeId = createChange().getChangeId();
EditInfoSubject.assertThat(getEdit(changeId)).isAbsent();
ChangeInfo changeInfo = gApi.changes().id(changeId).get();
ChangeStatus originalChangeStatus = changeInfo.status;
PushOneCommit.Result result = amendChange(changeId, "refs/drafts/master");
result.assertOkStatus();
changeInfo = gApi.changes().id(changeId).get();
assertThat(changeInfo.status).isEqualTo(originalChangeStatus);
assertThat(changeInfo.isPrivate).isNull();
assertThat(changeInfo.revisions).hasSize(1);
EditInfoSubject.assertThat(getEdit(changeId)).isPresent();
}
@GerritConfig(name = "receive.maxBatchCommits", value = "2") @GerritConfig(name = "receive.maxBatchCommits", value = "2")
@Test @Test
public void maxBatchCommits() throws Exception { public void maxBatchCommits() throws Exception {

@ -1,47 +0,0 @@
// Copyright (C) 2014 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.acceptance.git;
import static com.google.gerrit.server.group.SystemGroupBackend.ANONYMOUS_USERS;
import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.NoHttpd;
import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.common.data.Permission;
import org.junit.Before;
import org.junit.Test;
@NoHttpd
public class DraftChangeBlockedIT extends AbstractDaemonTest {
@Before
public void setUp() throws Exception {
block("refs/drafts/*", Permission.PUSH, ANONYMOUS_USERS);
}
@Test
public void pushDraftChange_Blocked() throws Exception {
// create draft by pushing to 'refs/drafts/'
PushOneCommit.Result r = pushTo("refs/drafts/master");
r.assertErrorStatus("cannot upload drafts");
}
@Test
public void pushDraftChangeMagic_Blocked() throws Exception {
// create draft by using 'draft' option
PushOneCommit.Result r = pushTo("refs/for/master%draft");
r.assertErrorStatus("cannot upload drafts");
}
}

@ -319,53 +319,6 @@ public class RefAdvertisementIT extends AbstractDaemonTest {
} }
} }
@Test
public void uploadPackDraftRefs() throws Exception {
allow("refs/heads/*", Permission.READ, REGISTERED_USERS);
PushOneCommit.Result br = createDraftChange();
br.assertOkStatus();
Change.Id c5 = br.getChange().getId();
String r5 = changeRefPrefix(c5);
// Only admin can see admin's draft change (5).
setApiUser(admin);
assertUploadPackRefs(
"HEAD",
r1 + "1",
r1 + "meta",
r2 + "1",
r2 + "meta",
r3 + "1",
r3 + "meta",
r4 + "1",
r4 + "meta",
r5 + "1",
r5 + "meta",
"refs/heads/branch",
"refs/heads/master",
RefNames.REFS_CONFIG,
"refs/tags/branch-tag",
"refs/tags/master-tag");
// user can't.
setApiUser(user);
assertUploadPackRefs(
"HEAD",
r1 + "1",
r1 + "meta",
r2 + "1",
r2 + "meta",
r3 + "1",
r3 + "meta",
r4 + "1",
r4 + "meta",
"refs/heads/branch",
"refs/heads/master",
"refs/tags/branch-tag",
"refs/tags/master-tag");
}
@Test @Test
public void uploadPackNoSearchingChangeCacheImpl() throws Exception { public void uploadPackNoSearchingChangeCacheImpl() throws Exception {
allow("refs/heads/*", Permission.READ, REGISTERED_USERS); allow("refs/heads/*", Permission.READ, REGISTERED_USERS);

@ -169,12 +169,6 @@ public class SubmitOnPushIT extends AbstractDaemonTest {
r.assertErrorStatus("update by submit not permitted"); r.assertErrorStatus("update by submit not permitted");
} }
@Test
public void submitOnPushingDraft_Error() throws Exception {
PushOneCommit.Result r = pushTo("refs/for/master%draft,submit");
r.assertErrorStatus();
}
@Test @Test
public void submitOnPushToNonExistingBranch_Error() throws Exception { public void submitOnPushToNonExistingBranch_Error() throws Exception {
String branchName = "non-existing"; String branchName = "non-existing";

@ -520,20 +520,6 @@ public abstract class AbstractSubmit extends AbstractDaemonTest {
.containsExactlyElementsIn(expected); .containsExactlyElementsIn(expected);
} }
@Test
public void submitDraftChange() throws Exception {
PushOneCommit.Result draft = createDraftChange();
Change.Id num = draft.getChange().getId();
submitWithConflict(
draft.getChangeId(),
"Failed to submit 1 change due to the following problems:\n"
+ "Change "
+ num
+ ": Change "
+ num
+ " is draft");
}
@Test @Test
public void submitWorkInProgressChange() throws Exception { public void submitWorkInProgressChange() throws Exception {
PushOneCommit.Result change = createWorkInProgressChange(); PushOneCommit.Result change = createWorkInProgressChange();
@ -548,21 +534,6 @@ public abstract class AbstractSubmit extends AbstractDaemonTest {
+ " is work in progress"); + " is work in progress");
} }
@Test
public void submitDraftPatchSet() throws Exception {
PushOneCommit.Result change = createChange();
PushOneCommit.Result draft = amendChangeAndMarkPatchSetAsDraft(change.getChangeId());
Change.Id num = draft.getChange().getId();
submitWithConflict(
draft.getChangeId(),
"Failed to submit 1 change due to the following problems:\n"
+ "Change "
+ num
+ ": submit rule error: "
+ "Cannot submit draft patch sets");
}
@Test @Test
public void submitWithHiddenBranchInSameTopic() throws Exception { public void submitWithHiddenBranchInSameTopic() throws Exception {
assume().that(isSubmitWholeTopicEnabled()).isTrue(); assume().that(isSubmitWholeTopicEnabled()).isTrue();

@ -139,28 +139,6 @@ public class ActionsIT extends AbstractDaemonTest {
} }
} }
@Test
public void revisionActionsETagWithHiddenDraftInTopic() throws Exception {
String change = createChangeWithTopic().getChangeId();
approve(change);
setApiUser(user);
String etag1 = getETag(change);
setApiUser(admin);
String draft = createDraftChange("topic").getChangeId();
approve(draft);
setApiUser(user);
String etag2 = getETag(change);
if (isSubmitWholeTopicEnabled()) {
assertThat(etag2).isNotEqualTo(etag1);
} else {
assertThat(etag2).isEqualTo(etag1);
}
}
@Test @Test
public void revisionActionsAnonymousETag() throws Exception { public void revisionActionsAnonymousETag() throws Exception {
String parent = createChange().getChangeId(); String parent = createChange().getChangeId();

@ -276,19 +276,6 @@ public class ChangeReviewersByEmailIT extends AbstractDaemonTest {
assertThat(result.reviewers).isNull(); assertThat(result.reviewers).isNull();
} }
@Test
public void rejectOnNonPublicChange() throws Exception {
assume().that(notesMigration.readChanges()).isTrue();
PushOneCommit.Result r = createDraftChange();
AddReviewerResult result =
gApi.changes().id(r.getChangeId()).addReviewer("Foo Bar <foo.bar@gerritcodereview.com>");
assertThat(result.error)
.isEqualTo(
"Foo Bar <foo.bar@gerritcodereview.com> does not have permission to see this change");
assertThat(result.reviewers).isNull();
}
@Test @Test
public void rejectWhenFeatureIsDisabled() throws Exception { public void rejectWhenFeatureIsDisabled() throws Exception {
assume().that(notesMigration.readChanges()).isTrue(); assume().that(notesMigration.readChanges()).isTrue();

@ -24,9 +24,7 @@ import static org.eclipse.jgit.lib.Constants.SIGNED_OFF_BY_TAG;
import com.google.common.base.Strings; import com.google.common.base.Strings;
import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Iterables;
import com.google.gerrit.acceptance.AbstractDaemonTest; import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.GerritConfig;
import com.google.gerrit.acceptance.PushOneCommit; import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.PushOneCommit.Result; import com.google.gerrit.acceptance.PushOneCommit.Result;
import com.google.gerrit.acceptance.RestResponse; import com.google.gerrit.acceptance.RestResponse;
@ -138,13 +136,6 @@ public class CreateChangeIT extends AbstractDaemonTest {
"%sAdministrator <%s>", SIGNED_OFF_BY_TAG, admin.getIdent().getEmailAddress())); "%sAdministrator <%s>", SIGNED_OFF_BY_TAG, admin.getIdent().getEmailAddress()));
} }
@Test
@GerritConfig(name = "change.allowDrafts", value = "true")
public void createNewDraftChangeNotAllowed() throws Exception {
ChangeInput ci = newChangeInput(ChangeStatus.DRAFT);
assertCreateFails(ci, BadRequestException.class, "unsupported change status");
}
@Test @Test
public void createNewPrivateChange() throws Exception { public void createNewPrivateChange() throws Exception {
ChangeInput input = newChangeInput(ChangeStatus.NEW); ChangeInput input = newChangeInput(ChangeStatus.NEW);
@ -388,9 +379,7 @@ public class CreateChangeIT extends AbstractDaemonTest {
assertThat(out.workInProgress).isEqualTo(in.workInProgress); assertThat(out.workInProgress).isEqualTo(in.workInProgress);
assertThat(out.revisions).hasSize(1); assertThat(out.revisions).hasSize(1);
assertThat(out.submitted).isNull(); assertThat(out.submitted).isNull();
assertThat(out.submitter).isNull(); assertThat(in.status).isEqualTo(ChangeStatus.NEW);
Boolean draft = Iterables.getOnlyElement(out.revisions.values()).draft;
assertThat(booleanToDraftStatus(draft)).isEqualTo(in.status);
return out; return out;
} }
@ -402,13 +391,6 @@ public class CreateChangeIT extends AbstractDaemonTest {
gApi.changes().create(in); gApi.changes().create(in);
} }
private ChangeStatus booleanToDraftStatus(Boolean draft) {
if (draft == null) {
return ChangeStatus.NEW;
}
return draft ? ChangeStatus.DRAFT : ChangeStatus.NEW;
}
// TODO(davido): Expose setting of account preferences in the API // TODO(davido): Expose setting of account preferences in the API
private void setSignedOffByFooter() throws Exception { private void setSignedOffByFooter() throws Exception {
RestResponse r = adminRestSession.get("/accounts/" + admin.email + "/preferences"); RestResponse r = adminRestSession.get("/accounts/" + admin.email + "/preferences");

@ -1,261 +0,0 @@
// Copyright (C) 2013 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.acceptance.rest.change;
import static com.google.common.truth.Truth.assertThat;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.NoHttpd;
import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.TestAccount;
import com.google.gerrit.extensions.api.changes.DraftInput;
import com.google.gerrit.extensions.api.changes.ReviewInput;
import com.google.gerrit.extensions.api.changes.ReviewInput.CommentInput;
import com.google.gerrit.extensions.client.ChangeStatus;
import com.google.gerrit.extensions.client.Side;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.ChangeMessage;
import com.google.gerrit.reviewdb.client.Comment;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.RefNames;
import com.google.gerrit.server.config.AllUsersName;
import com.google.gerrit.server.query.change.ChangeData;
import com.google.inject.Inject;
import java.util.HashMap;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;
import org.junit.Test;
@NoHttpd
public class DeleteDraftPatchSetIT extends AbstractDaemonTest {
@Inject private AllUsersName allUsers;
@Test
public void deletePatchSetNotDraft() throws Exception {
String changeId = createChange().getChangeId();
PatchSet ps = getCurrentPatchSet(changeId);
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.NEW);
exception.expect(ResourceConflictException.class);
exception.expectMessage("Patch set is not a draft");
setApiUser(admin);
deletePatchSet(changeId, ps);
}
@Test
public void deleteDraftPatchSetNoACL() throws Exception {
String changeId = createDraftChangeWith2PS();
PatchSet ps = getCurrentPatchSet(changeId);
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.DRAFT);
exception.expect(ResourceNotFoundException.class);
exception.expectMessage("Not found: " + changeId);
setApiUser(user);
deletePatchSet(changeId, ps);
}
@Test
public void deleteDraftPatchSetAndChange() throws Exception {
String changeId = createDraftChangeWith2PS();
PatchSet ps = getCurrentPatchSet(changeId);
Change.Id id = ps.getId().getParentKey();
DraftInput din = new DraftInput();
din.path = "a.txt";
din.message = "comment on a.txt";
gApi.changes().id(changeId).current().createDraft(din);
if (notesMigration.commitChangeWrites()) {
assertThat(getDraftRef(admin, id)).isNotNull();
}
ChangeData cd = getChange(changeId);
assertThat(cd.patchSets()).hasSize(2);
assertThat(cd.change().currentPatchSetId().get()).isEqualTo(2);
assertThat(cd.change().getStatus()).isEqualTo(Change.Status.DRAFT);
deletePatchSet(changeId, ps);
cd = getChange(changeId);
assertThat(cd.patchSets()).hasSize(1);
assertThat(cd.change().currentPatchSetId().get()).isEqualTo(1);
ps = getCurrentPatchSet(changeId);
deletePatchSet(changeId, ps);
assertThat(queryProvider.get().byKeyPrefix(changeId)).isEmpty();
if (notesMigration.commitChangeWrites()) {
assertThat(getDraftRef(admin, id)).isNull();
assertThat(getMetaRef(id)).isNull();
}
exception.expect(ResourceNotFoundException.class);
gApi.changes().id(id.get());
}
@Test
public void deleteDraftPS1() throws Exception {
String changeId = createDraftChangeWith2PS();
ReviewInput rin = new ReviewInput();
rin.message = "Change message";
CommentInput cin = new CommentInput();
cin.line = 1;
cin.patchSet = 1;
cin.path = PushOneCommit.FILE_NAME;
cin.side = Side.REVISION;
cin.message = "Inline comment";
rin.comments = new HashMap<>();
rin.comments.put(cin.path, ImmutableList.of(cin));
gApi.changes().id(changeId).revision(1).review(rin);
ChangeData cd = getChange(changeId);
PatchSet.Id delPsId = new PatchSet.Id(cd.getId(), 1);
PatchSet ps = cd.patchSet(delPsId);
deletePatchSet(changeId, ps);
cd = getChange(changeId);
assertThat(cd.patchSets()).hasSize(1);
assertThat(Iterables.getOnlyElement(cd.patchSets()).getId().get()).isEqualTo(2);
// Other entities based on deleted patch sets are also deleted.
for (ChangeMessage m : cd.messages()) {
assertThat(m.getPatchSetId()).named(m.toString()).isNotEqualTo(delPsId);
}
for (Comment c : cd.publishedComments()) {
assertThat(c.key.patchSetId).named(c.toString()).isNotEqualTo(delPsId.get());
}
}
@Test
public void deleteDraftPS2() throws Exception {
String changeId = createDraftChangeWith2PS();
ReviewInput rin = new ReviewInput();
rin.message = "Change message";
CommentInput cin = new CommentInput();
cin.line = 1;
cin.patchSet = 1;
cin.path = PushOneCommit.FILE_NAME;
cin.side = Side.REVISION;
cin.message = "Inline comment";
rin.comments = new HashMap<>();
rin.comments.put(cin.path, ImmutableList.of(cin));
gApi.changes().id(changeId).revision(1).review(rin);
ChangeData cd = getChange(changeId);
PatchSet.Id delPsId = new PatchSet.Id(cd.getId(), 2);
PatchSet ps = cd.patchSet(delPsId);
deletePatchSet(changeId, ps);
cd = getChange(changeId);
assertThat(cd.patchSets()).hasSize(1);
assertThat(Iterables.getOnlyElement(cd.patchSets()).getId().get()).isEqualTo(1);
// Other entities based on deleted patch sets are also deleted.
for (ChangeMessage m : cd.messages()) {
assertThat(m.getPatchSetId()).named(m.toString()).isNotEqualTo(delPsId);
}
for (Comment c : cd.publishedComments()) {
assertThat(c.key.patchSetId).named(c.toString()).isNotEqualTo(delPsId.get());
}
}
@Test
public void deleteCurrentDraftPatchSetWhenPreviousPatchSetDoesNotExist() throws Exception {
String changeId = createChange().getChangeId();
amendChangeAndMarkPatchSetAsDraft(changeId);
amendChangeAndMarkPatchSetAsDraft(changeId);
deletePatchSet(changeId, 2);
deletePatchSet(changeId, 3);
ChangeData cd = getChange(changeId);
assertThat(cd.patchSets()).hasSize(1);
assertThat(Iterables.getOnlyElement(cd.patchSets()).getId().get()).isEqualTo(1);
assertThat(cd.currentPatchSet().getId().get()).isEqualTo(1);
}
@Test
public void deleteDraftPatchSetAndPushNewDraftPatchSet() throws Exception {
// Create change
PushOneCommit.Result r1 = createDraftChange();
String changeId = r1.getChangeId();
String revPs1 = r1.getChange().currentPatchSet().getRevision().get();
// Push draft patch set
PushOneCommit.Result r2 = amendChangeAndMarkPatchSetAsDraft(changeId);
String revPs2 = r2.getChange().currentPatchSet().getRevision().get();
assertThat(gApi.changes().id(r1.getChange().getId().get()).get().currentRevision)
.isEqualTo(revPs2);
// Remove draft patch set
gApi.changes().id(r1.getChange().getId().get()).revision(revPs2).delete();
assertThat(gApi.changes().id(r1.getChange().getId().get()).get().currentRevision)
.isEqualTo(revPs1);
// Push new draft patch set
amendChangeAndMarkPatchSetAsDraft(changeId);
String revPs3 = r2.getChange().currentPatchSet().getRevision().get();
assertThat(gApi.changes().id(r1.getChange().getId().get()).get().currentRevision)
.isEqualTo(revPs3);
// Check that all patch sets have different SHA1s
assertThat(revPs1).doesNotMatch(revPs2);
assertThat(revPs2).doesNotMatch(revPs3);
}
private Ref getDraftRef(TestAccount account, Change.Id changeId) throws Exception {
try (Repository repo = repoManager.openRepository(allUsers)) {
return repo.exactRef(RefNames.refsDraftComments(changeId, account.id));
}
}
private Ref getMetaRef(Change.Id changeId) throws Exception {
try (Repository repo = repoManager.openRepository(project)) {
return repo.exactRef(RefNames.changeMetaRef(changeId));
}
}
private PatchSet getCurrentPatchSet(String changeId) throws Exception {
return getChange(changeId).currentPatchSet();
}
private ChangeData getChange(String changeId) throws Exception {
return Iterables.getOnlyElement(queryProvider.get().byKeyPrefix(changeId));
}
private void deletePatchSet(String changeId, PatchSet ps) throws Exception {
deletePatchSet(changeId, ps.getId().get());
}
private void deletePatchSet(String changeId, int ps) throws Exception {
gApi.changes().id(changeId).revision(ps).delete();
}
}

@ -1,295 +0,0 @@
// Copyright (C) 2013 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.acceptance.rest.change;
import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.TruthJUnit.assume;
import com.google.common.collect.Iterables;
import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.RestResponse;
import com.google.gerrit.acceptance.RestSession;
import com.google.gerrit.acceptance.TestProjectInput;
import com.google.gerrit.common.data.Permission;
import com.google.gerrit.extensions.api.changes.ReviewInput;
import com.google.gerrit.extensions.client.ChangeStatus;
import com.google.gerrit.extensions.client.ReviewerState;
import com.google.gerrit.extensions.common.AccountInfo;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.EditInfo;
import com.google.gerrit.extensions.common.EditInfoSubject;
import com.google.gerrit.extensions.common.LabelInfo;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.MethodNotAllowedException;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.testutil.ConfigSuite;
import java.util.Collection;
import java.util.Optional;
import org.eclipse.jgit.lib.Config;
import org.junit.Test;
public class DraftChangeIT extends AbstractDaemonTest {
@ConfigSuite.Config
public static Config allowDraftsDisabled() {
return allowDraftsDisabledConfig();
}
@Test
public void forceCreateAndPublishDraftChangeWhenAllowDraftsDisabled() throws Exception {
PushOneCommit.Result result = createDraftChange();
result.assertOkStatus();
String changeId = result.getChangeId();
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.DRAFT);
assertThat(c.revisions.get(c.currentRevision).draft).isTrue();
publishPatchSet(changeId).assertNoContent();
assertThat(get(triplet).status).isEqualTo(ChangeStatus.NEW);
}
@Test
public void deleteDraftChange() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result result = createDraftChange();
result.assertOkStatus();
String changeId = result.getChangeId();
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.DRAFT);
deleteChange(changeId, adminRestSession).assertNoContent();
exception.expect(ResourceNotFoundException.class);
get(triplet);
}
@Test
public void deleteDraftChangeOfAnotherUser() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result changeResult = createDraftChange();
changeResult.assertOkStatus();
String changeId = changeResult.getChangeId();
// The user needs to be able to see the draft change (which reviewers can).
gApi.changes().id(changeId).addReviewer(user.fullName);
setApiUser(user);
exception.expect(AuthException.class);
exception.expectMessage("delete not permitted");
gApi.changes().id(changeId).delete();
}
@Test
@TestProjectInput(cloneAs = "user")
public void deleteDraftChangeWhenDraftsNotAllowedAsNormalUser() throws Exception {
assume().that(isAllowDrafts()).isFalse();
setApiUser(user);
// We can't create a draft change while the draft workflow is disabled.
// For this reason, we create a normal change and modify the database.
PushOneCommit.Result changeResult =
pushFactory.create(db, user.getIdent(), testRepo).to("refs/for/master");
Change.Id id = changeResult.getChange().getId();
markChangeAsDraft(id);
setDraftStatusOfPatchSets(id, true);
String changeId = changeResult.getChangeId();
exception.expect(MethodNotAllowedException.class);
exception.expectMessage("Draft workflow is disabled");
gApi.changes().id(changeId).delete();
}
@Test
@TestProjectInput(cloneAs = "user")
public void deleteDraftChangeWhenDraftsNotAllowedAsAdmin() throws Exception {
assume().that(isAllowDrafts()).isFalse();
setApiUser(user);
// We can't create a draft change while the draft workflow is disabled.
// For this reason, we create a normal change and modify the database.
PushOneCommit.Result changeResult =
pushFactory.create(db, user.getIdent(), testRepo).to("refs/for/master");
Change.Id id = changeResult.getChange().getId();
markChangeAsDraft(id);
setDraftStatusOfPatchSets(id, true);
String changeId = changeResult.getChangeId();
// Grant those permissions to admins.
grant(project, "refs/*", Permission.VIEW_DRAFTS);
grant(project, "refs/*", Permission.DELETE_DRAFTS);
try {
setApiUser(admin);
gApi.changes().id(changeId).delete();
} finally {
removePermission(project, "refs/*", Permission.DELETE_DRAFTS);
removePermission(project, "refs/*", Permission.VIEW_DRAFTS);
}
setApiUser(user);
assertThat(query(changeId)).isEmpty();
}
@Test
public void deleteDraftChangeWithNonDraftPatchSet() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result changeResult = createDraftChange();
Change.Id id = changeResult.getChange().getId();
setDraftStatusOfPatchSets(id, false);
String changeId = changeResult.getChangeId();
exception.expect(ResourceConflictException.class);
exception.expectMessage(
String.format("Cannot delete draft change %s: patch set 1 is not a draft", id));
gApi.changes().id(changeId).delete();
}
@Test
public void publishDraftChange() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result result = createDraftChange();
result.assertOkStatus();
String changeId = result.getChangeId();
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.DRAFT);
assertThat(c.revisions.get(c.currentRevision).draft).isTrue();
publishChange(changeId).assertNoContent();
c = get(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.NEW);
assertThat(c.revisions.get(c.currentRevision).draft).isNull();
}
@Test
public void publishDraftPatchSet() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result result = createDraftChange();
result.assertOkStatus();
String changeId = result.getChangeId();
String triplet = project.get() + "~master~" + changeId;
ChangeInfo c = get(triplet);
assertThat(c.id).isEqualTo(triplet);
assertThat(c.status).isEqualTo(ChangeStatus.DRAFT);
publishPatchSet(changeId).assertNoContent();
assertThat(get(triplet).status).isEqualTo(ChangeStatus.NEW);
}
@Test
public void createDraftChangeWhenDraftsNotAllowed() throws Exception {
assume().that(isAllowDrafts()).isFalse();
PushOneCommit.Result r = pushTo("refs/drafts/master");
r.assertErrorStatus();
}
@Test
public void listApprovalsOnDraftChange() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result result = createDraftChange();
result.assertOkStatus();
String changeId = result.getChangeId();
String triplet = project.get() + "~master~" + changeId;
gApi.changes().id(triplet).addReviewer(user.fullName);
ChangeInfo info = get(triplet);
LabelInfo label = info.labels.get("Code-Review");
assertThat(label.all).hasSize(1);
assertThat(label.all.get(0)._accountId).isEqualTo(user.id.get());
assertThat(label.all.get(0).value).isEqualTo(0);
Collection<AccountInfo> ccs = info.reviewers.get(ReviewerState.REVIEWER);
assertThat(ccs).hasSize(1);
assertThat(ccs.iterator().next()._accountId).isEqualTo(user.id.get());
setApiUser(user);
gApi.changes().id(triplet).current().review(ReviewInput.recommend());
setApiUser(admin);
label = get(triplet).labels.get("Code-Review");
assertThat(label.all).hasSize(1);
assertThat(label.all.get(0)._accountId).isEqualTo(user.id.get());
assertThat(label.all.get(0).value).isEqualTo(1);
}
@Test
public void pushWithDraftOptionGetsPrivateChange() throws Exception {
assume().that(isAllowDrafts()).isTrue();
PushOneCommit.Result result = createChange("refs/drafts/master");
String changeId = result.getChangeId();
ChangeInfo changeInfo = gApi.changes().id(changeId).get();
assertThat(changeInfo.status).isEqualTo(ChangeStatus.NEW);
assertThat(changeInfo.isPrivate).isEqualTo(true);
assertThat(changeInfo.revisions).hasSize(1);
assertThat(Iterables.getOnlyElement(changeInfo.revisions.values()).draft).isNull();
}
@Test
public void pushWithDraftOptionToExistingNewChangeGetsChangeEdit() throws Exception {
assume().that(isAllowDrafts()).isTrue();
pushWithDraftOptionToExistingChangeGetsChangeEdit(createChange().getChangeId());
}
@Test
public void pushWithDraftOptionToExistingDraftChangeGetsChangeEdit() throws Exception {
assume().that(isAllowDrafts()).isTrue();
pushWithDraftOptionToExistingChangeGetsChangeEdit(createDraftChange().getChangeId());
}
private void pushWithDraftOptionToExistingChangeGetsChangeEdit(String changeId) throws Exception {
Optional<EditInfo> edit = getEdit(changeId);
EditInfoSubject.assertThat(edit).isAbsent();
ChangeInfo changeInfo = gApi.changes().id(changeId).get();
ChangeStatus originalChangeStatus = changeInfo.status;
Boolean originalPatchSetStatus = Iterables.getOnlyElement(changeInfo.revisions.values()).draft;
PushOneCommit.Result result = amendChange(changeId, "refs/drafts/master");
result.assertOkStatus();
changeInfo = gApi.changes().id(changeId).get();
assertThat(changeInfo.status).isEqualTo(originalChangeStatus);
assertThat(changeInfo.isPrivate).isNull();
assertThat(changeInfo.revisions).hasSize(1);
assertThat(Iterables.getOnlyElement(changeInfo.revisions.values()).draft)
.isEqualTo(originalPatchSetStatus);
edit = getEdit(changeId);
EditInfoSubject.assertThat(edit).isPresent();
}
private static RestResponse deleteChange(String changeId, RestSession s) throws Exception {
return s.delete("/changes/" + changeId);
}
private RestResponse publishChange(String changeId) throws Exception {
return adminRestSession.post("/changes/" + changeId + "/publish");
}
private RestResponse publishPatchSet(String changeId) throws Exception {
PatchSet patchSet =
Iterables.getOnlyElement(queryProvider.get().byKeyPrefix(changeId)).currentPatchSet();
return adminRestSession.post(
"/changes/" + changeId + "/revisions/" + patchSet.getRevision().get() + "/publish");
}
}

@ -19,7 +19,6 @@ import static org.junit.Assert.fail;
import com.google.gerrit.acceptance.GitUtil; import com.google.gerrit.acceptance.GitUtil;
import com.google.gerrit.acceptance.PushOneCommit; import com.google.gerrit.acceptance.PushOneCommit;
import com.google.gerrit.acceptance.TestProjectInput;
import com.google.gerrit.extensions.api.changes.ChangeApi; import com.google.gerrit.extensions.api.changes.ChangeApi;
import com.google.gerrit.extensions.api.changes.CherryPickInput; import com.google.gerrit.extensions.api.changes.CherryPickInput;
import com.google.gerrit.extensions.api.changes.ReviewInput; import com.google.gerrit.extensions.api.changes.ReviewInput;
@ -509,30 +508,6 @@ public class SubmitByMergeIfNecessaryIT extends AbstractSubmitByMerge {
assertChangeMergedEvents(); assertChangeMergedEvents();
} }
@Test
@TestProjectInput(createEmptyCommit = false)
public void mergeWithMissingChange() throws Exception {
// create a draft change
PushOneCommit.Result draftResult = createDraftChange();
// create a new change based on the draft change
PushOneCommit.Result changeResult = createChange();
// delete the draft change
gApi.changes().id(draftResult.getChangeId()).delete();
// approve and submit the change
submitWithConflict(
changeResult.getChangeId(),
"Failed to submit 1 change due to the following problems:\n"
+ "Change "
+ changeResult.getChange().getId()
+ ": depends on change that was not submitted");
assertRefUpdatedEvents();
assertChangeMergedEvents();
}
@Test @Test
public void testPreviewSubmitTgz() throws Exception { public void testPreviewSubmitTgz() throws Exception {
Project.NameKey p1 = createProject("project-name"); Project.NameKey p1 = createProject("project-name");

@ -25,15 +25,11 @@ import com.google.gerrit.extensions.api.changes.SubmittedTogetherInfo;
import com.google.gerrit.extensions.client.ChangeStatus; import com.google.gerrit.extensions.client.ChangeStatus;
import com.google.gerrit.extensions.client.ListChangesOption; import com.google.gerrit.extensions.client.ListChangesOption;
import com.google.gerrit.extensions.client.SubmitType; import com.google.gerrit.extensions.client.SubmitType;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.FileInfo; import com.google.gerrit.extensions.common.FileInfo;
import com.google.gerrit.extensions.common.RevisionInfo; import com.google.gerrit.extensions.common.RevisionInfo;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.Project; import com.google.gerrit.reviewdb.client.Project;
import com.google.gerrit.testutil.ConfigSuite; import com.google.gerrit.testutil.ConfigSuite;
import java.util.EnumSet; import java.util.EnumSet;
import java.util.List;
import org.eclipse.jgit.junit.TestRepository; import org.eclipse.jgit.junit.TestRepository;
import org.eclipse.jgit.lib.Config; import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.revwalk.RevCommit; import org.eclipse.jgit.revwalk.RevCommit;
@ -155,101 +151,6 @@ public class SubmittedTogetherIT extends AbstractDaemonTest {
} }
} }
@Test
public void hiddenDraftInTopic() throws Exception {
String id1 = createChange("subject", "a", "1", "topic").getChangeId();
createDraftChange("topic");
setApiUser(user);
SubmittedTogetherInfo result =
gApi.changes().id(id1).submittedTogether(EnumSet.of(NON_VISIBLE_CHANGES));
if (isSubmitWholeTopicEnabled()) {
assertThat(result.changes).hasSize(1);
assertThat(result.changes.get(0).changeId).isEqualTo(id1);
assertThat(result.nonVisibleChanges).isEqualTo(1);
} else {
assertThat(result.changes).isEmpty();
assertThat(result.nonVisibleChanges).isEqualTo(0);
}
}
@Test
public void hiddenDraftInTopicOldApi() throws Exception {
String id1 = createChange("subject", "a", "1", "topic").getChangeId();
createDraftChange("topic");
setApiUser(user);
if (isSubmitWholeTopicEnabled()) {
exception.expect(AuthException.class);
exception.expectMessage("change would be submitted with a change that you cannot see");
gApi.changes().id(id1).submittedTogether();
} else {
List<ChangeInfo> result = gApi.changes().id(id1).submittedTogether();
assertThat(result).isEmpty();
}
}
@Test
public void draftPatchSetInTopic() throws Exception {
RevCommit initialHead = getRemoteHead();
RevCommit a1 = commitBuilder().add("a", "1").message("change 1").create();
pushHead(testRepo, "refs/for/master/" + name("topic"), false);
String id1 = getChangeId(a1);
testRepo.reset(initialHead);
String parentId = createChange().getChangeId();
// TODO(jrn): use insertChangeId(id1) once jgit TestRepository accepts the leading "I".
commitBuilder()
.insertChangeId(id1.substring(1))
.add("a", "2")
.message("draft patch set on change 1")
.create();
pushHead(testRepo, "refs/for/master/" + name("topic"), false);
setCurrentPatchSetAsDraft(new Change.Id(gApi.changes().id(id1).get()._number));
testRepo.reset(initialHead);
RevCommit b = commitBuilder().message("change with same topic").create();
pushHead(testRepo, "refs/for/master/" + name("topic"), false);
String id2 = getChangeId(b);
if (isSubmitWholeTopicEnabled()) {
setApiUser(user);
assertSubmittedTogether(id2, id2, id1);
setApiUser(admin);
assertSubmittedTogether(id2, id2, id1, parentId);
} else {
setApiUser(user);
assertSubmittedTogether(id2);
setApiUser(admin);
assertSubmittedTogether(id2);
}
}
@Test
public void doNotRevealVisibleAncestorOfHiddenDraft() throws Exception {
RevCommit initialHead = getRemoteHead();
createChange().getChangeId();
createDraftChange("topic");
testRepo.reset(initialHead);
String id = createChange("subject", "b", "1", "topic").getChangeId();
setApiUser(user);
SubmittedTogetherInfo result =
gApi.changes().id(id).submittedTogether(EnumSet.of(NON_VISIBLE_CHANGES));
if (isSubmitWholeTopicEnabled()) {
assertThat(result.changes).hasSize(1);
assertThat(result.changes.get(0).changeId).isEqualTo(id);
assertThat(result.nonVisibleChanges).isEqualTo(2);
} else {
assertThat(result.changes).isEmpty();
assertThat(result.nonVisibleChanges).isEqualTo(0);
}
}
@Test @Test
public void topicChaining() throws Exception { public void topicChaining() throws Exception {
RevCommit initialHead = getRemoteHead(); RevCommit initialHead = getRemoteHead();

@ -42,7 +42,6 @@ import com.google.gerrit.extensions.api.changes.ReviewInput;
import com.google.gerrit.extensions.api.changes.ReviewInput.CommentInput; import com.google.gerrit.extensions.api.changes.ReviewInput.CommentInput;
import com.google.gerrit.extensions.api.changes.ReviewInput.DraftHandling; import com.google.gerrit.extensions.api.changes.ReviewInput.DraftHandling;
import com.google.gerrit.extensions.client.Side; import com.google.gerrit.extensions.client.Side;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.CommentInfo; import com.google.gerrit.extensions.common.CommentInfo;
import com.google.gerrit.extensions.restapi.ResourceNotFoundException; import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
import com.google.gerrit.extensions.restapi.RestApiException; import com.google.gerrit.extensions.restapi.RestApiException;
@ -816,20 +815,6 @@ public class ChangeRebuilderIT extends AbstractDaemonTest {
assertThat(nc.getOriginalSubject()).isEqualTo(orig); assertThat(nc.getOriginalSubject()).isEqualTo(orig);
} }
@Test
public void deleteDraftPS1WithNoOtherEntities() throws Exception {
String r = createDraftChangeWith2PS();
gApi.changes().id(r).revision(1).delete();
ChangeInfo changeInfo = get(r);
Change.Id id = new Change.Id(changeInfo._number);
checker.rebuildAndCheckChanges(id);
setNotesMigration(true, true);
ChangeNotes notes = notesFactory.create(db, project, id);
assertThat(notes.getPatchSets().keySet()).hasSize(1);
}
@Test @Test
public void ignorePatchLineCommentsOnPatchSet0() throws Exception { public void ignorePatchLineCommentsOnPatchSet0() throws Exception {
PushOneCommit.Result r = createChange(); PushOneCommit.Result r = createChange();

@ -16,7 +16,6 @@ package com.google.gerrit.acceptance.ssh;
import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.Truth.assertWithMessage; import static com.google.common.truth.Truth.assertWithMessage;
import static com.google.gerrit.acceptance.GitUtil.initSsh;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import com.google.gerrit.acceptance.AbstractDaemonTest; import com.google.gerrit.acceptance.AbstractDaemonTest;
@ -290,27 +289,6 @@ public class QueryIT extends AbstractDaemonTest {
assertThat(changes.get(0).submitRecords.size()).isEqualTo(1); assertThat(changes.get(0).submitRecords.size()).isEqualTo(1);
} }
@Test
public void queryWithNonVisibleCurrentPatchSet() throws Exception {
String changeId = createChange().getChangeId();
amendChangeAndMarkPatchSetAsDraft(changeId);
String query = "--current-patch-set --patch-sets " + changeId;
List<ChangeAttribute> changes = executeSuccessfulQuery(query);
assertThat(changes.size()).isEqualTo(1);
assertThat(changes.get(0).patchSets).isNotNull();
assertThat(changes.get(0).patchSets).hasSize(2);
assertThat(changes.get(0).currentPatchSet).isNotNull();
SshSession userSession = new SshSession(server, user);
initSsh(user);
userSession.open();
changes = executeSuccessfulQuery(query, userSession);
assertThat(changes.size()).isEqualTo(1);
assertThat(changes.get(0).patchSets).hasSize(1);
assertThat(changes.get(0).currentPatchSet).isNull();
userSession.close();
}
@Test @Test
public void allChangeOptionsAreServedWithoutExceptions() throws Exception { public void allChangeOptionsAreServedWithoutExceptions() throws Exception {
PushOneCommit.Result r = createChange(); PushOneCommit.Result r = createChange();

@ -142,7 +142,6 @@ public class PageLinks {
switch (status) { switch (status) {
case ABANDONED: case ABANDONED:
return toChangeQuery(status(status) + " " + op("topic", topic)); return toChangeQuery(status(status) + " " + op("topic", topic));
case DRAFT:
case MERGED: case MERGED:
case NEW: case NEW:
return toChangeQuery( return toChangeQuery(
@ -169,7 +168,6 @@ public class PageLinks {
return "status:abandoned"; return "status:abandoned";
case MERGED: case MERGED:
return "status:merged"; return "status:merged";
case DRAFT:
case NEW: case NEW:
default: default:
return "status:open"; return "status:open";

@ -27,7 +27,6 @@ public class Permission implements Comparable<Permission> {
public static final String DELETE = "delete"; public static final String DELETE = "delete";
public static final String CREATE_TAG = "createTag"; public static final String CREATE_TAG = "createTag";
public static final String CREATE_SIGNED_TAG = "createSignedTag"; public static final String CREATE_SIGNED_TAG = "createSignedTag";
public static final String DELETE_DRAFTS = "deleteDrafts";
public static final String DELETE_OWN_CHANGES = "deleteOwnChanges"; public static final String DELETE_OWN_CHANGES = "deleteOwnChanges";
public static final String EDIT_HASHTAGS = "editHashtags"; public static final String EDIT_HASHTAGS = "editHashtags";
public static final String EDIT_ASSIGNEE = "editAssignee"; public static final String EDIT_ASSIGNEE = "editAssignee";
@ -38,7 +37,6 @@ public class Permission implements Comparable<Permission> {
public static final String LABEL = "label-"; public static final String LABEL = "label-";
public static final String LABEL_AS = "labelAs-"; public static final String LABEL_AS = "labelAs-";
public static final String OWNER = "owner"; public static final String OWNER = "owner";
public static final String PUBLISH_DRAFTS = "publishDrafts";
public static final String PUSH = "push"; public static final String PUSH = "push";
public static final String PUSH_MERGE = "pushMerge"; public static final String PUSH_MERGE = "pushMerge";
public static final String READ = "read"; public static final String READ = "read";
@ -46,7 +44,6 @@ public class Permission implements Comparable<Permission> {
public static final String REMOVE_REVIEWER = "removeReviewer"; public static final String REMOVE_REVIEWER = "removeReviewer";
public static final String SUBMIT = "submit"; public static final String SUBMIT = "submit";
public static final String SUBMIT_AS = "submitAs"; public static final String SUBMIT_AS = "submitAs";
public static final String VIEW_DRAFTS = "viewDrafts";
public static final String VIEW_PRIVATE_CHANGES = "viewPrivateChanges"; public static final String VIEW_PRIVATE_CHANGES = "viewPrivateChanges";
private static final List<String> NAMES_LC; private static final List<String> NAMES_LC;
@ -74,14 +71,11 @@ public class Permission implements Comparable<Permission> {
NAMES_LC.add(REMOVE_REVIEWER.toLowerCase()); NAMES_LC.add(REMOVE_REVIEWER.toLowerCase());
NAMES_LC.add(SUBMIT.toLowerCase()); NAMES_LC.add(SUBMIT.toLowerCase());
NAMES_LC.add(SUBMIT_AS.toLowerCase()); NAMES_LC.add(SUBMIT_AS.toLowerCase());
NAMES_LC.add(VIEW_DRAFTS.toLowerCase());
NAMES_LC.add(VIEW_PRIVATE_CHANGES.toLowerCase()); NAMES_LC.add(VIEW_PRIVATE_CHANGES.toLowerCase());
NAMES_LC.add(EDIT_TOPIC_NAME.toLowerCase()); NAMES_LC.add(EDIT_TOPIC_NAME.toLowerCase());
NAMES_LC.add(EDIT_HASHTAGS.toLowerCase()); NAMES_LC.add(EDIT_HASHTAGS.toLowerCase());
NAMES_LC.add(EDIT_ASSIGNEE.toLowerCase()); NAMES_LC.add(EDIT_ASSIGNEE.toLowerCase());
NAMES_LC.add(DELETE_DRAFTS.toLowerCase());
NAMES_LC.add(DELETE_OWN_CHANGES.toLowerCase()); NAMES_LC.add(DELETE_OWN_CHANGES.toLowerCase());
NAMES_LC.add(PUBLISH_DRAFTS.toLowerCase());
LABEL_INDEX = NAMES_LC.indexOf(Permission.LABEL); LABEL_INDEX = NAMES_LC.indexOf(Permission.LABEL);
LABEL_AS_INDEX = NAMES_LC.indexOf(Permission.LABEL_AS.toLowerCase()); LABEL_AS_INDEX = NAMES_LC.indexOf(Permission.LABEL_AS.toLowerCase());

@ -159,6 +159,7 @@ public interface ChangeApi {
throws RestApiException; throws RestApiException;
/** Publishes a draft change. */ /** Publishes a draft change. */
@Deprecated
void publish() throws RestApiException; void publish() throws RestApiException;
/** Rebase the current revision of a change using default options. */ /** Rebase the current revision of a change using default options. */
@ -403,6 +404,7 @@ public interface ChangeApi {
throw new NotImplementedException(); throw new NotImplementedException();
} }
@Deprecated
@Override @Override
public void rebase() throws RestApiException { public void rebase() throws RestApiException {
throw new NotImplementedException(); throw new NotImplementedException();

@ -31,6 +31,7 @@ import java.util.Map;
import java.util.Set; import java.util.Set;
public interface RevisionApi { public interface RevisionApi {
@Deprecated
void delete() throws RestApiException; void delete() throws RestApiException;
String description() throws RestApiException; String description() throws RestApiException;
@ -47,6 +48,7 @@ public interface RevisionApi {
BinaryResult submitPreview(String format) throws RestApiException; BinaryResult submitPreview(String format) throws RestApiException;
@Deprecated
void publish() throws RestApiException; void publish() throws RestApiException;
ChangeApi cherryPick(CherryPickInput in) throws RestApiException; ChangeApi cherryPick(CherryPickInput in) throws RestApiException;
@ -155,6 +157,7 @@ public interface RevisionApi {
* interface. * interface.
*/ */
class NotImplemented implements RevisionApi { class NotImplemented implements RevisionApi {
@Deprecated
@Override @Override
public void delete() throws RestApiException { public void delete() throws RestApiException {
throw new NotImplementedException(); throw new NotImplementedException();
@ -175,6 +178,7 @@ public interface RevisionApi {
throw new NotImplementedException(); throw new NotImplementedException();
} }
@Deprecated
@Override @Override
public void publish() throws RestApiException { public void publish() throws RestApiException {
throw new NotImplementedException(); throw new NotImplementedException();

@ -33,22 +33,6 @@ public enum ChangeStatus {
*/ */
NEW, NEW,
/**
* Change is a draft change that only consists of draft patchsets.
*
* <p>This is a change that is not meant to be submitted or reviewed yet. If the uploader
* publishes the change, it becomes a NEW change. Publishing is a one-way action, a change cannot
* return to DRAFT status. Draft changes are only visible to the uploader and those explicitly
* added as reviewers. Note that currently draft changes cannot be abandoned.
*
* <p>Changes in the DRAFT state can be moved to:
*
* <ul>
* <li>{@link #NEW} - when the change is published, it becomes a new change.
* </ul>
*/
DRAFT,
/** /**
* Change is closed, and submitted to its destination branch. * Change is closed, and submitted to its destination branch.
* *

@ -20,7 +20,6 @@ import java.util.Map;
public class RevisionInfo { public class RevisionInfo {
public transient boolean isCurrent; public transient boolean isCurrent;
public Boolean draft;
public ChangeKind kind; public ChangeKind kind;
public int _number; public int _number;
public Timestamp created; public Timestamp created;

@ -1,25 +0,0 @@
// Copyright (C) 2015 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.extensions.events;
import com.google.gerrit.extensions.annotations.ExtensionPoint;
/** Notified whenever a Draft is published. */
@ExtensionPoint
public interface DraftPublishedListener {
interface Event extends RevisionEvent {}
void onDraftPublished(Event event);
}

@ -419,8 +419,6 @@ public class ChangeInfo extends JavaScriptObject {
public final native String name() /*-{ return this.name; }-*/; public final native String name() /*-{ return this.name; }-*/;
public final native boolean draft() /*-{ return this.draft || false; }-*/;
public final native AccountInfo uploader() /*-{ return this.uploader; }-*/; public final native AccountInfo uploader() /*-{ return this.uploader; }-*/;
public final native boolean isEdit() /*-{ return this._number == 0; }-*/; public final native boolean isEdit() /*-{ return this._number == 0; }-*/;

@ -91,8 +91,6 @@ public interface GerritConstants extends Constants {
String menuMyChanges(); String menuMyChanges();
String menuMyDrafts();
String menuMyWatchedChanges(); String menuMyWatchedChanges();
String menuMyStarredChanges(); String menuMyStarredChanges();
@ -175,8 +173,6 @@ public interface GerritConstants extends Constants {
String jumpMine(); String jumpMine();
String jumpMineDrafts();
String jumpMineWatched(); String jumpMineWatched();
String jumpMineStarred(); String jumpMineStarred();

@ -53,7 +53,6 @@ menuAllAbandoned = Abandoned
menuMine = My menuMine = My
menuMyChanges = Changes menuMyChanges = Changes
menuMyDrafts = Drafts
menuMyStarredChanges = Starred Changes menuMyStarredChanges = Starred Changes
menuMyWatchedChanges = Watched Changes menuMyWatchedChanges = Watched Changes
menuMyDraftComments = Draft Comments menuMyDraftComments = Draft Comments
@ -105,7 +104,6 @@ jumpAllMerged = Go to all merged changes
jumpAllAbandoned = Go to all abandoned changes jumpAllAbandoned = Go to all abandoned changes
jumpMine = Go to my dashboard jumpMine = Go to my dashboard
jumpMineWatched = Go to watched changes jumpMineWatched = Go to watched changes
jumpMineDrafts = Go to drafts
jumpMineStarred = Go to starred changes jumpMineStarred = Go to starred changes
jumpMineDraftComments = Go to draft comments jumpMineDraftComments = Go to draft comments

@ -70,13 +70,6 @@ public class JumpKeys {
Gerrit.display(PageLinks.MINE); Gerrit.display(PageLinks.MINE);
} }
}); });
jumps.add(
new KeyCommand(0, 'd', Gerrit.C.jumpMineDrafts()) {
@Override
public void onKeyPress(KeyPressEvent event) {
Gerrit.display(PageLinks.toChangeQuery("owner:self is:draft"));
}
});
jumps.add( jumps.add(
new KeyCommand(0, 'c', Gerrit.C.jumpMineDraftComments()) { new KeyCommand(0, 'c', Gerrit.C.jumpMineDraftComments()) {
@Override @Override

@ -129,7 +129,6 @@ public class SearchSuggestOracle extends HighlightSuggestOracle {
suggestions.add("is:reviewer"); suggestions.add("is:reviewer");
suggestions.add("is:open"); suggestions.add("is:open");
suggestions.add("is:pending"); suggestions.add("is:pending");
suggestions.add("is:draft");
suggestions.add("is:private"); suggestions.add("is:private");
suggestions.add("is:closed"); suggestions.add("is:closed");
suggestions.add("is:merged"); suggestions.add("is:merged");
@ -145,7 +144,6 @@ public class SearchSuggestOracle extends HighlightSuggestOracle {
suggestions.add("status:closed"); suggestions.add("status:closed");
suggestions.add("status:merged"); suggestions.add("status:merged");
suggestions.add("status:abandoned"); suggestions.add("status:abandoned");
suggestions.add("status:draft");
suggestions.add("added:"); suggestions.add("added:");
suggestions.add("deleted:"); suggestions.add("deleted:");

@ -137,7 +137,6 @@ permissionNames = \
createTag, \ createTag, \
createSignedTag, \ createSignedTag, \
delete, \ delete, \
deleteDrafts, \
deleteOwnChanges, \ deleteOwnChanges, \
editAssignee, \ editAssignee, \
editHashtags, \ editHashtags, \
@ -146,7 +145,6 @@ permissionNames = \
forgeCommitter, \ forgeCommitter, \
forgeServerAsCommitter, \ forgeServerAsCommitter, \
owner, \ owner, \
publishDrafts, \
push, \ push, \
pushMerge, \ pushMerge, \
read, \ read, \
@ -154,7 +152,6 @@ permissionNames = \
removeReviewer, \ removeReviewer, \
submit, \ submit, \
submitAs, \ submitAs, \
viewDrafts, \
viewPrivateChanges viewPrivateChanges
abandon = Abandon abandon = Abandon
@ -163,7 +160,6 @@ create = Create Reference
createTag = Create Annotated Tag createTag = Create Annotated Tag
createSignedTag = Create Signed Tag createSignedTag = Create Signed Tag
delete = Delete Reference delete = Delete Reference
deleteDrafts = Delete Drafts
deleteOwnChanges = Delete Own Changes deleteOwnChanges = Delete Own Changes
editAssignee = Edit Assignee editAssignee = Edit Assignee
editHashtags = Edit Hashtags editHashtags = Edit Hashtags
@ -172,7 +168,6 @@ forgeAuthor = Forge Author Identity
forgeCommitter = Forge Committer Identity forgeCommitter = Forge Committer Identity
forgeServerAsCommitter = Forge Server Identity forgeServerAsCommitter = Forge Server Identity
owner = Owner owner = Owner
publishDrafts = Publish Drafts
push = Push push = Push
pushMerge = Push Merge Commit pushMerge = Push Merge Commit
read = Read read = Read
@ -180,7 +175,6 @@ rebase = Rebase
removeReviewer = Remove Reviewer removeReviewer = Remove Reviewer
submit = Submit submit = Submit
submitAs = Submit (On Behalf Of) submitAs = Submit (On Behalf Of)
viewDrafts = View Drafts
viewPrivateChanges = View Private Changes viewPrivateChanges = View Private Changes
refErrorEmpty = Reference must be supplied refErrorEmpty = Reference must be supplied

@ -26,16 +26,6 @@ import com.google.gwt.user.client.ui.Button;
public class ChangeActions { public class ChangeActions {
static void publish(
Project.NameKey project, Change.Id id, String revision, Button... draftButtons) {
ChangeApi.publish(project.get(), id.get(), revision, cs(project, id, draftButtons));
}
static void delete(
Project.NameKey project, Change.Id id, String revision, Button... draftButtons) {
ChangeApi.deleteRevision(project.get(), id.get(), revision, cs(project, id, draftButtons));
}
static void delete(Project.NameKey project, Change.Id id, Button... draftButtons) { static void delete(Project.NameKey project, Change.Id id, Button... draftButtons) {
ChangeApi.deleteChange(project.get(), id.get(), mine(draftButtons)); ChangeApi.deleteChange(project.get(), id.get(), mine(draftButtons));
} }

@ -43,8 +43,6 @@ public interface ChangeConstants extends Constants {
String author(); String author();
String draft();
String notAvailable(); String notAvailable();
String relatedChanges(); String relatedChanges();
@ -78,6 +76,4 @@ public interface ChangeConstants extends Constants {
String deleteChangeEdit(); String deleteChangeEdit();
String deleteChange(); String deleteChange();
String deleteDraftRevision();
} }

@ -13,7 +13,6 @@ patchSet = Patch Set
commit = Commit commit = Commit
date = Date date = Date
author = Author / Committer author = Author / Committer
draft = (DRAFT)
notAvailable = N/A notAvailable = N/A
relatedChanges = Related Changes relatedChanges = Related Changes
@ -35,4 +34,3 @@ deleteChangeEdit = Delete Change Edit?\n\
\n\ \n\
All changes made in the edit revision will be lost. All changes made in the edit revision will be lost.
deleteChange = Delete Change? deleteChange = Delete Change?
deleteDraftRevision = Delete Draft Revision?

@ -234,8 +234,6 @@ public class ChangeScreen extends Screen {
@UiField Button publishEdit; @UiField Button publishEdit;
@UiField Button rebaseEdit; @UiField Button rebaseEdit;
@UiField Button deleteEdit; @UiField Button deleteEdit;
@UiField Button publish;
@UiField Button deleteRevision;
@UiField Button openAll; @UiField Button openAll;
@UiField Button editMode; @UiField Button editMode;
@UiField Button reviewMode; @UiField Button reviewMode;
@ -562,8 +560,7 @@ public class ChangeScreen extends Screen {
} }
} }
private void initRevisionsAction( private void initRevisionsAction(ChangeInfo info, String revision) {
ChangeInfo info, String revision, NativeMap<ActionInfo> actions) {
int currentPatchSet; int currentPatchSet;
if (info.currentRevision() != null && info.revisions().containsKey(info.currentRevision())) { if (info.currentRevision() != null && info.revisions().containsKey(info.currentRevision())) {
currentPatchSet = info.revision(info.currentRevision())._number(); currentPatchSet = info.revision(info.currentRevision())._number();
@ -591,18 +588,6 @@ public class ChangeScreen extends Screen {
patchSetsAction = patchSetsAction =
new PatchSetsAction( new PatchSetsAction(
info.projectNameKey(), info.legacyId(), revision, edit, style, headerLine, patchSets); info.projectNameKey(), info.legacyId(), revision, edit, style, headerLine, patchSets);
RevisionInfo revInfo = info.revision(revision);
if (revInfo.draft()) {
if (actions.containsKey("publish")) {
publish.setVisible(true);
publish.setTitle(actions.get("publish").title());
}
if (actions.containsKey("/")) {
deleteRevision.setVisible(true);
deleteRevision.setTitle(actions.get("/").title());
}
}
} }
private void initDownloadAction(ChangeInfo info, String revision) { private void initDownloadAction(ChangeInfo info, String revision) {
@ -702,18 +687,6 @@ public class ChangeScreen extends Screen {
} }
} }
@UiHandler("publish")
void onPublish(@SuppressWarnings("unused") ClickEvent e) {
ChangeActions.publish(getProject(), changeId, revision, publish, deleteRevision);
}
@UiHandler("deleteRevision")
void onDeleteRevision(@SuppressWarnings("unused") ClickEvent e) {
if (Window.confirm(Resources.C.deleteDraftRevision())) {
ChangeActions.delete(getProject(), changeId, revision, publish, deleteRevision);
}
}
@Override @Override
public void registerKeys() { public void registerKeys() {
super.registerKeys(); super.registerKeys();
@ -1323,10 +1296,7 @@ public class ChangeScreen extends Screen {
} }
private boolean isSubmittable(ChangeInfo info) { private boolean isSubmittable(ChangeInfo info) {
boolean canSubmit = boolean canSubmit = info.status().isOpen() && revision.equals(info.currentRevision());
info.status().isOpen()
&& revision.equals(info.currentRevision())
&& !info.revision(revision).draft();
if (canSubmit && info.status() == Change.Status.NEW) { if (canSubmit && info.status() == Change.Status.NEW) {
for (String name : info.labels()) { for (String name : info.labels()) {
LabelInfo label = info.label(name); LabelInfo label = info.label(name);
@ -1404,7 +1374,7 @@ public class ChangeScreen extends Screen {
// Properly render revision actions initially while waiting for // Properly render revision actions initially while waiting for
// the callback to populate them correctly. // the callback to populate them correctly.
NativeMap<ActionInfo> emptyMap = NativeMap.<ActionInfo>create(); NativeMap<ActionInfo> emptyMap = NativeMap.<ActionInfo>create();
initRevisionsAction(info, revision, emptyMap); initRevisionsAction(info, revision);
quickApprove.setVisible(false); quickApprove.setVisible(false);
actions.reloadRevisionActions(emptyMap); actions.reloadRevisionActions(emptyMap);
@ -1416,8 +1386,7 @@ public class ChangeScreen extends Screen {
statusText.setInnerText(Util.C.notCurrent()); statusText.setInnerText(Util.C.notCurrent());
labels.setVisible(false); labels.setVisible(false);
} else { } else {
Status s = info.revision(revision).draft() ? Status.DRAFT : info.status(); statusText.setInnerText(Util.toLongString(info.status()));
statusText.setInnerText(Util.toLongString(s));
} }
if (info.isPrivate()) { if (info.isPrivate()) {
@ -1444,7 +1413,7 @@ public class ChangeScreen extends Screen {
} }
private void renderRevisionInfo(ChangeInfo info, NativeMap<ActionInfo> actionMap) { private void renderRevisionInfo(ChangeInfo info, NativeMap<ActionInfo> actionMap) {
initRevisionsAction(info, revision, actionMap); initRevisionsAction(info, revision);
commit.setParentNotCurrent( commit.setParentNotCurrent(
actionMap.containsKey("rebase") && actionMap.get("rebase").enabled()); actionMap.containsKey("rebase") && actionMap.get("rebase").enabled());
actions.reloadRevisionActions(actionMap); actions.reloadRevisionActions(actionMap);

@ -415,13 +415,6 @@ limitations under the License.
<g:Button ui:field='deleteEdit' styleName='' visible='false'> <g:Button ui:field='deleteEdit' styleName='' visible='false'>
<div><ui:msg>Delete Edit</ui:msg></div> <div><ui:msg>Delete Edit</ui:msg></div>
</g:Button> </g:Button>
<g:Button ui:field='publish'
styleName='{style.highlight}' visible='false'>
<div><ui:msg>Publish</ui:msg></div>
</g:Button>
<g:Button ui:field='deleteRevision' styleName='' visible='false'>
<div><ui:msg>Delete Revision</ui:msg></div>
</g:Button>
<g:SimplePanel ui:field='headerExtensionMiddle' styleName='{style.headerExtension}'/> <g:SimplePanel ui:field='headerExtensionMiddle' styleName='{style.headerExtension}'/>
</div> </div>
</div> </div>

@ -192,9 +192,6 @@ class PatchSetsBox extends Composite {
} }
sb.openTd().setStyleName(style.legacy_id()); sb.openTd().setStyleName(style.legacy_id());
if (r.draft()) {
sb.append(Resources.C.draft()).append(' ');
}
sb.append(r.id()); sb.append(r.id());
sb.closeTd(); sb.closeTd();

@ -49,7 +49,7 @@ class QuickApprove extends Button implements ClickHandler {
setVisible(false); setVisible(false);
return; return;
} }
if (info.revision(commit).isEdit() || info.revision(commit).draft()) { if (info.revision(commit).isEdit()) {
setVisible(false); setVisible(false);
return; return;
} }

@ -245,25 +245,12 @@ public class ChangeApi {
call(project, id, commit, "submit").post(in, cb); call(project, id, commit, "submit").post(in, cb);
} }
/** Publish a specific revision of a draft change. */
public static void publish(
@Nullable String project, int id, String commit, AsyncCallback<JavaScriptObject> cb) {
JavaScriptObject in = JavaScriptObject.createObject();
call(project, id, commit, "publish").post(in, cb);
}
/** Delete a specific draft change. */ /** Delete a specific draft change. */
public static void deleteChange( public static void deleteChange(
@Nullable String project, int id, AsyncCallback<JavaScriptObject> cb) { @Nullable String project, int id, AsyncCallback<JavaScriptObject> cb) {
change(project, id).delete(cb); change(project, id).delete(cb);
} }
/** Delete a specific draft patch set. */
public static void deleteRevision(
@Nullable String project, int id, String commit, AsyncCallback<JavaScriptObject> cb) {
revision(project, id, commit).delete(cb);
}
/** Delete change edit. */ /** Delete change edit. */
public static void deleteEdit( public static void deleteEdit(
@Nullable String project, int id, AsyncCallback<JavaScriptObject> cb) { @Nullable String project, int id, AsyncCallback<JavaScriptObject> cb) {

@ -30,8 +30,6 @@ public class Util {
return ""; return "";
} }
switch (status) { switch (status) {
case DRAFT:
return C.statusLongDraft();
case NEW: case NEW:
return C.statusLongNew(); return C.statusLongNew();
case MERGED: case MERGED:

@ -398,7 +398,6 @@ abstract class DiffScreen extends Screen {
if (Gerrit.isSignedIn()) { if (Gerrit.isSignedIn()) {
keyMap keyMap
.on("G I", () -> Gerrit.display(PageLinks.MINE)) .on("G I", () -> Gerrit.display(PageLinks.MINE))
.on("G D", () -> Gerrit.display(PageLinks.toChangeQuery("owner:self is:draft")))
.on("G C", () -> Gerrit.display(PageLinks.toChangeQuery("has:draft"))) .on("G C", () -> Gerrit.display(PageLinks.toChangeQuery("has:draft")))
.on("G W", () -> Gerrit.display(PageLinks.toChangeQuery("is:watched status:open"))) .on("G W", () -> Gerrit.display(PageLinks.toChangeQuery("is:watched status:open")))
.on("G S", () -> Gerrit.display(PageLinks.toChangeQuery("is:starred"))); .on("G S", () -> Gerrit.display(PageLinks.toChangeQuery("is:starred")));

@ -302,8 +302,6 @@ public final class Change {
private static final char MIN_OPEN = 'a'; private static final char MIN_OPEN = 'a';
/** Database constant for {@link Status#NEW}. */ /** Database constant for {@link Status#NEW}. */
public static final char STATUS_NEW = 'n'; public static final char STATUS_NEW = 'n';
/** Database constant for {@link Status#DRAFT}. */
public static final char STATUS_DRAFT = 'd';
/** Maximum database status constant for an open change. */ /** Maximum database status constant for an open change. */
private static final char MAX_OPEN = 'z'; private static final char MAX_OPEN = 'z';
@ -340,27 +338,10 @@ public final class Change {
*/ */
NEW(STATUS_NEW, ChangeStatus.NEW), NEW(STATUS_NEW, ChangeStatus.NEW),
/**
* Change is a draft change that only consists of draft patchsets.
*
* <p>This is a change that is not meant to be submitted or reviewed yet. If the uploader
* publishes the change, it becomes a NEW change. Publishing is a one-way action, a change
* cannot return to DRAFT status. Draft changes are only visible to the uploader and those
* explicitly added as reviewers.
*
* <p>Changes in the DRAFT state can be moved to:
*
* <ul>
* <li>{@link #NEW} - when the change is published, it becomes a new change;
* </ul>
*/
DRAFT(STATUS_DRAFT, ChangeStatus.DRAFT),
/** /**
* Change is closed, and submitted to its destination branch. * Change is closed, and submitted to its destination branch.
* *
* <p>Once a change has been merged, it cannot be further modified by adding a replacement patch * <p>Once a change has been merged, it cannot be further modified by adding a replacement patch
* set. Draft comments however may be published, supporting a post-submit review.
*/ */
MERGED(STATUS_MERGED, ChangeStatus.MERGED), MERGED(STATUS_MERGED, ChangeStatus.MERGED),
@ -422,6 +403,14 @@ public final class Change {
return s; return s;
} }
} }
// TODO(davido): Remove in 3.0, after all sites upgraded to version,
// where DRAFT status was removed. This code path is still needed,
// when changes are deserialized from the secondary index, during
// the online migration to the new schema version wasn't completed.
if (c == 'd') {
return Status.NEW;
}
return null; return null;
} }

@ -168,8 +168,7 @@ public final class PatchSet {
@Column(id = 4) @Column(id = 4)
protected Timestamp createdOn; protected Timestamp createdOn;
@Column(id = 5) // @Column(id = 5)
protected boolean draft;
/** /**
* Opaque group identifier, usually assigned during creation. * Opaque group identifier, usually assigned during creation.
@ -209,7 +208,6 @@ public final class PatchSet {
this.revision = src.revision; this.revision = src.revision;
this.uploader = src.uploader; this.uploader = src.uploader;
this.createdOn = src.createdOn; this.createdOn = src.createdOn;
this.draft = src.draft;
this.groups = src.groups; this.groups = src.groups;
this.pushCertificate = src.pushCertificate; this.pushCertificate = src.pushCertificate;
this.description = src.description; this.description = src.description;
@ -247,14 +245,6 @@ public final class PatchSet {
createdOn = ts; createdOn = ts;
} }
public boolean isDraft() {
return draft;
}
public void setDraft(boolean draftStatus) {
draft = draftStatus;
}
public List<String> getGroups() { public List<String> getGroups() {
if (groups == null) { if (groups == null) {
return Collections.emptyList(); return Collections.emptyList();

@ -18,7 +18,6 @@ import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull; import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.collect.ImmutableMap.toImmutableMap; import static com.google.common.collect.ImmutableMap.toImmutableMap;
import static com.google.gerrit.server.ChangeUtil.PS_ID_ORDER; import static com.google.gerrit.server.ChangeUtil.PS_ID_ORDER;
import static com.google.gerrit.server.notedb.PatchSetState.DRAFT;
import static com.google.gerrit.server.notedb.PatchSetState.PUBLISHED; import static com.google.gerrit.server.notedb.PatchSetState.PUBLISHED;
import static java.util.function.Function.identity; import static java.util.function.Function.identity;
@ -34,7 +33,6 @@ import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.notedb.ChangeNotes; import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.notedb.ChangeUpdate; import com.google.gerrit.server.notedb.ChangeUpdate;
import com.google.gerrit.server.notedb.NotesMigration; import com.google.gerrit.server.notedb.NotesMigration;
import com.google.gerrit.server.notedb.PatchSetState;
import com.google.gwtorm.server.OrmException; import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject; import com.google.inject.Inject;
import com.google.inject.Singleton; import com.google.inject.Singleton;
@ -104,7 +102,6 @@ public class PatchSetUtil {
ChangeUpdate update, ChangeUpdate update,
PatchSet.Id psId, PatchSet.Id psId,
ObjectId commit, ObjectId commit,
boolean draft,
List<String> groups, List<String> groups,
String pushCertificate, String pushCertificate,
String description) String description)
@ -116,7 +113,6 @@ public class PatchSetUtil {
ps.setRevision(new RevId(commit.name())); ps.setRevision(new RevId(commit.name()));
ps.setUploader(update.getAccountId()); ps.setUploader(update.getAccountId());
ps.setCreatedOn(new Timestamp(update.getWhen().getTime())); ps.setCreatedOn(new Timestamp(update.getWhen().getTime()));
ps.setDraft(draft);
ps.setGroups(groups); ps.setGroups(groups);
ps.setPushCertificate(pushCertificate); ps.setPushCertificate(pushCertificate);
ps.setDescription(description); ps.setDescription(description);
@ -125,27 +121,16 @@ public class PatchSetUtil {
update.setCommit(rw, commit, pushCertificate); update.setCommit(rw, commit, pushCertificate);
update.setPsDescription(description); update.setPsDescription(description);
update.setGroups(groups); update.setGroups(groups);
if (draft) {
update.setPatchSetState(DRAFT);
}
return ps; return ps;
} }
public void publish(ReviewDb db, ChangeUpdate update, PatchSet ps) throws OrmException { public void publish(ReviewDb db, ChangeUpdate update, PatchSet ps) throws OrmException {
ensurePatchSetMatches(ps.getId(), update); ensurePatchSetMatches(ps.getId(), update);
ps.setDraft(false);
update.setPatchSetState(PUBLISHED); update.setPatchSetState(PUBLISHED);
db.patchSets().update(Collections.singleton(ps)); db.patchSets().update(Collections.singleton(ps));
} }
public void delete(ReviewDb db, ChangeUpdate update, PatchSet ps) throws OrmException {
ensurePatchSetMatches(ps.getId(), update);
checkArgument(ps.isDraft(), "cannot delete non-draft patch set %s", ps.getId());
update.setPatchSetState(PatchSetState.DELETED);
db.patchSets().delete(Collections.singleton(ps));
}
private void ensurePatchSetMatches(PatchSet.Id psId, ChangeUpdate update) { private void ensurePatchSetMatches(PatchSet.Id psId, ChangeUpdate update) {
Change.Id changeId = update.getChange().getId(); Change.Id changeId = update.getChange().getId();
checkArgument( checkArgument(

@ -141,7 +141,6 @@ public class GeneralPreferencesLoader {
} }
if (r.my.isEmpty()) { if (r.my.isEmpty()) {
r.my.add(new MenuItem("Changes", "#/dashboard/self", null)); r.my.add(new MenuItem("Changes", "#/dashboard/self", null));
r.my.add(new MenuItem("Drafts", "#/q/owner:self+is:draft", null));
r.my.add(new MenuItem("Draft Comments", "#/q/has:draft", null)); r.my.add(new MenuItem("Draft Comments", "#/q/has:draft", null));
r.my.add(new MenuItem("Edits", "#/q/has:edit", null)); r.my.add(new MenuItem("Edits", "#/q/has:edit", null));
r.my.add(new MenuItem("Watched Changes", "#/q/is:watched+is:open", null)); r.my.add(new MenuItem("Watched Changes", "#/q/is:watched+is:open", null));

@ -74,7 +74,6 @@ import com.google.gerrit.server.change.Mute;
import com.google.gerrit.server.change.PostHashtags; import com.google.gerrit.server.change.PostHashtags;
import com.google.gerrit.server.change.PostPrivate; import com.google.gerrit.server.change.PostPrivate;
import com.google.gerrit.server.change.PostReviewers; import com.google.gerrit.server.change.PostReviewers;
import com.google.gerrit.server.change.PublishDraftPatchSet;
import com.google.gerrit.server.change.PutAssignee; import com.google.gerrit.server.change.PutAssignee;
import com.google.gerrit.server.change.PutMessage; import com.google.gerrit.server.change.PutMessage;
import com.google.gerrit.server.change.PutTopic; import com.google.gerrit.server.change.PutTopic;
@ -117,7 +116,6 @@ class ChangeApiImpl implements ChangeApi {
private final Restore restore; private final Restore restore;
private final CreateMergePatchSet updateByMerge; private final CreateMergePatchSet updateByMerge;
private final Provider<SubmittedTogether> submittedTogether; private final Provider<SubmittedTogether> submittedTogether;
private final PublishDraftPatchSet.CurrentRevision publishDraftChange;
private final Rebase.CurrentRevision rebase; private final Rebase.CurrentRevision rebase;
private final DeleteChange deleteChange; private final DeleteChange deleteChange;
private final GetTopic getTopic; private final GetTopic getTopic;
@ -163,7 +161,6 @@ class ChangeApiImpl implements ChangeApi {
Restore restore, Restore restore,
CreateMergePatchSet updateByMerge, CreateMergePatchSet updateByMerge,
Provider<SubmittedTogether> submittedTogether, Provider<SubmittedTogether> submittedTogether,
PublishDraftPatchSet.CurrentRevision publishDraftChange,
Rebase.CurrentRevision rebase, Rebase.CurrentRevision rebase,
DeleteChange deleteChange, DeleteChange deleteChange,
GetTopic getTopic, GetTopic getTopic,
@ -207,7 +204,6 @@ class ChangeApiImpl implements ChangeApi {
this.restore = restore; this.restore = restore;
this.updateByMerge = updateByMerge; this.updateByMerge = updateByMerge;
this.submittedTogether = submittedTogether; this.submittedTogether = submittedTogether;
this.publishDraftChange = publishDraftChange;
this.rebase = rebase; this.rebase = rebase;
this.deleteChange = deleteChange; this.deleteChange = deleteChange;
this.getTopic = getTopic; this.getTopic = getTopic;
@ -403,13 +399,10 @@ class ChangeApiImpl implements ChangeApi {
} }
} }
@Deprecated
@Override @Override
public void publish() throws RestApiException { public void publish() throws RestApiException {
try { throw new UnsupportedOperationException("draft workflow is discontinued");
publishDraftChange.apply(change, null);
} catch (Exception e) {
throw asRestApiException("Cannot publish change", e);
}
} }
@Override @Override

@ -49,7 +49,6 @@ import com.google.gerrit.server.change.ApplyFix;
import com.google.gerrit.server.change.CherryPick; import com.google.gerrit.server.change.CherryPick;
import com.google.gerrit.server.change.Comments; import com.google.gerrit.server.change.Comments;
import com.google.gerrit.server.change.CreateDraftComment; import com.google.gerrit.server.change.CreateDraftComment;
import com.google.gerrit.server.change.DeleteDraftPatchSet;
import com.google.gerrit.server.change.DraftComments; import com.google.gerrit.server.change.DraftComments;
import com.google.gerrit.server.change.FileResource; import com.google.gerrit.server.change.FileResource;
import com.google.gerrit.server.change.Files; import com.google.gerrit.server.change.Files;
@ -65,7 +64,6 @@ import com.google.gerrit.server.change.ListRobotComments;
import com.google.gerrit.server.change.Mergeable; import com.google.gerrit.server.change.Mergeable;
import com.google.gerrit.server.change.PostReview; import com.google.gerrit.server.change.PostReview;
import com.google.gerrit.server.change.PreviewSubmit; import com.google.gerrit.server.change.PreviewSubmit;
import com.google.gerrit.server.change.PublishDraftPatchSet;
import com.google.gerrit.server.change.PutDescription; import com.google.gerrit.server.change.PutDescription;
import com.google.gerrit.server.change.Rebase; import com.google.gerrit.server.change.Rebase;
import com.google.gerrit.server.change.RebaseUtil; import com.google.gerrit.server.change.RebaseUtil;
@ -95,12 +93,10 @@ class RevisionApiImpl implements RevisionApi {
private final RevisionReviewers revisionReviewers; private final RevisionReviewers revisionReviewers;
private final RevisionReviewerApiImpl.Factory revisionReviewerApi; private final RevisionReviewerApiImpl.Factory revisionReviewerApi;
private final CherryPick cherryPick; private final CherryPick cherryPick;
private final DeleteDraftPatchSet deleteDraft;
private final Rebase rebase; private final Rebase rebase;
private final RebaseUtil rebaseUtil; private final RebaseUtil rebaseUtil;
private final Submit submit; private final Submit submit;
private final PreviewSubmit submitPreview; private final PreviewSubmit submitPreview;
private final PublishDraftPatchSet publish;
private final Reviewed.PutReviewed putReviewed; private final Reviewed.PutReviewed putReviewed;
private final Reviewed.DeleteReviewed deleteReviewed; private final Reviewed.DeleteReviewed deleteReviewed;
private final RevisionResource revision; private final RevisionResource revision;
@ -137,12 +133,10 @@ class RevisionApiImpl implements RevisionApi {
RevisionReviewers revisionReviewers, RevisionReviewers revisionReviewers,
RevisionReviewerApiImpl.Factory revisionReviewerApi, RevisionReviewerApiImpl.Factory revisionReviewerApi,
CherryPick cherryPick, CherryPick cherryPick,
DeleteDraftPatchSet deleteDraft,
Rebase rebase, Rebase rebase,
RebaseUtil rebaseUtil, RebaseUtil rebaseUtil,
Submit submit, Submit submit,
PreviewSubmit submitPreview, PreviewSubmit submitPreview,
PublishDraftPatchSet publish,
Reviewed.PutReviewed putReviewed, Reviewed.PutReviewed putReviewed,
Reviewed.DeleteReviewed deleteReviewed, Reviewed.DeleteReviewed deleteReviewed,
Files files, Files files,
@ -176,13 +170,11 @@ class RevisionApiImpl implements RevisionApi {
this.revisionReviewers = revisionReviewers; this.revisionReviewers = revisionReviewers;
this.revisionReviewerApi = revisionReviewerApi; this.revisionReviewerApi = revisionReviewerApi;
this.cherryPick = cherryPick; this.cherryPick = cherryPick;
this.deleteDraft = deleteDraft;
this.rebase = rebase; this.rebase = rebase;
this.rebaseUtil = rebaseUtil; this.rebaseUtil = rebaseUtil;
this.review = review; this.review = review;
this.submit = submit; this.submit = submit;
this.submitPreview = submitPreview; this.submitPreview = submitPreview;
this.publish = publish;
this.files = files; this.files = files;
this.putReviewed = putReviewed; this.putReviewed = putReviewed;
this.deleteReviewed = deleteReviewed; this.deleteReviewed = deleteReviewed;
@ -253,20 +245,12 @@ class RevisionApiImpl implements RevisionApi {
@Override @Override
public void publish() throws RestApiException { public void publish() throws RestApiException {
try { throw new UnsupportedOperationException("draft workflow is discontinued");
publish.apply(revision, new PublishDraftPatchSet.Input());
} catch (Exception e) {
throw asRestApiException("Cannot publish draft patch set", e);
}
} }
@Override @Override
public void delete() throws RestApiException { public void delete() throws RestApiException {
try { throw new UnsupportedOperationException("draft workflow is discontinued");
deleteDraft.apply(revision, null);
} catch (Exception e) {
throw asRestApiException("Cannot delete draft ps", e);
}
} }
@Override @Override

@ -204,7 +204,7 @@ public class Abandon extends RetryingRestModifyView<ChangeResource, AbandonInput
.setTitle("Abandon the change") .setTitle("Abandon the change")
.setVisible( .setVisible(
and( and(
change.getStatus().isOpen() && change.getStatus() != Change.Status.DRAFT, change.getStatus().isOpen(),
rsrc.permissions().database(dbProvider).testCond(ChangePermission.ABANDON))); rsrc.permissions().database(dbProvider).testCond(ChangePermission.ABANDON)));
} }
} }

@ -155,7 +155,6 @@ public class ActionJson {
copy.ref = revisionInfo.ref; copy.ref = revisionInfo.ref;
copy.created = revisionInfo.created; copy.created = revisionInfo.created;
copy.uploader = revisionInfo.uploader; copy.uploader = revisionInfo.uploader;
copy.draft = revisionInfo.draft;
copy.fetch = revisionInfo.fetch; copy.fetch = revisionInfo.fetch;
copy.kind = revisionInfo.kind; copy.kind = revisionInfo.kind;
copy.description = revisionInfo.description; copy.description = revisionInfo.description;

@ -405,7 +405,6 @@ public class ChangeInserter implements InsertChangeOp {
update, update,
psId, psId,
commitId, commitId,
false,
newGroups, newGroups,
pushCert, pushCert,
patchSetDescription); patchSetDescription);

@ -610,7 +610,7 @@ public class ChangeJson {
ctl = changeControlFactory.controlFor(db.get(), cd.change(), userProvider.get()); ctl = changeControlFactory.controlFor(db.get(), cd.change(), userProvider.get());
} }
if (needMessages) { if (needMessages) {
out.messages = messages(ctl, cd, src); out.messages = messages(ctl, cd);
} }
finish(out); finish(out);
@ -1112,8 +1112,8 @@ public class ChangeJson {
return result; return result;
} }
private Collection<ChangeMessageInfo> messages( private Collection<ChangeMessageInfo> messages(ChangeControl ctl, ChangeData cd)
ChangeControl ctl, ChangeData cd, Map<PatchSet.Id, PatchSet> map) throws OrmException { throws OrmException {
List<ChangeMessage> messages = cmUtil.byChange(db.get(), cd.notes()); List<ChangeMessage> messages = cmUtil.byChange(db.get(), cd.notes());
if (messages.isEmpty()) { if (messages.isEmpty()) {
return Collections.emptyList(); return Collections.emptyList();
@ -1122,8 +1122,7 @@ public class ChangeJson {
List<ChangeMessageInfo> result = Lists.newArrayListWithCapacity(messages.size()); List<ChangeMessageInfo> result = Lists.newArrayListWithCapacity(messages.size());
for (ChangeMessage message : messages) { for (ChangeMessage message : messages) {
PatchSet.Id patchNum = message.getPatchSetId(); PatchSet.Id patchNum = message.getPatchSetId();
PatchSet ps = patchNum != null ? map.get(patchNum) : null; if (patchNum == null || ctl.isVisible(db.get())) {
if (patchNum == null || ctl.isPatchVisible(ps, db.get())) {
ChangeMessageInfo cmi = new ChangeMessageInfo(); ChangeMessageInfo cmi = new ChangeMessageInfo();
cmi.id = message.getKey().get(); cmi.id = message.getKey().get();
cmi.author = accountLoader.get(message.getAuthor()); cmi.author = accountLoader.get(message.getAuthor());
@ -1257,7 +1256,7 @@ public class ChangeJson {
} else { } else {
want = id.equals(cd.change().currentPatchSetId()); want = id.equals(cd.change().currentPatchSetId());
} }
if (want && ctl.isPatchVisible(in, db.get())) { if (want && ctl.isVisible(db.get())) {
res.put(in.getRevision().get(), toRevisionInfo(cd, in, repo, rw, false, changeInfo)); res.put(in.getRevision().get(), toRevisionInfo(cd, in, repo, rw, false, changeInfo));
} }
} }
@ -1318,7 +1317,6 @@ public class ChangeJson {
out.ref = in.getRefName(); out.ref = in.getRefName();
out.created = in.getCreatedOn(); out.created = in.getCreatedOn();
out.uploader = accountLoader.get(in.getUploader()); out.uploader = accountLoader.get(in.getUploader());
out.draft = in.isDraft() ? true : null;
out.fetch = makeFetchMap(cd, in); out.fetch = makeFetchMap(cd, in);
out.kind = changeKindCache.getChangeKind(rw, repo != null ? repo.getConfig() : null, cd, in); out.kind = changeKindCache.getChangeKind(rw, repo != null ? repo.getConfig() : null, cd, in);
out.description = in.getDescription(); out.description = in.getDescription();
@ -1356,9 +1354,7 @@ public class ChangeJson {
out.files.remove(Patch.MERGE_LIST); out.files.remove(Patch.MERGE_LIST);
} }
if ((out.isCurrent || (out.draft != null && out.draft)) if (out.isCurrent && has(CURRENT_ACTIONS) && userProvider.get().isIdentifiedUser()) {
&& has(CURRENT_ACTIONS)
&& userProvider.get().isIdentifiedUser()) {
actionJson.addRevisionActions( actionJson.addRevisionActions(
changeInfo, changeInfo,
@ -1423,7 +1419,7 @@ public class ChangeJson {
continue; continue;
} }
if (!scheme.isAuthSupported() && !ctl.isPatchVisible(in, db.get())) { if (!scheme.isAuthSupported() && !ctl.isVisible(db.get())) {
continue; continue;
} }

@ -37,7 +37,6 @@ import com.google.gerrit.server.ChangeMessagesUtil;
import com.google.gerrit.server.ChangeUtil; import com.google.gerrit.server.ChangeUtil;
import com.google.gerrit.server.GerritPersonIdent; import com.google.gerrit.server.GerritPersonIdent;
import com.google.gerrit.server.IdentifiedUser; import com.google.gerrit.server.IdentifiedUser;
import com.google.gerrit.server.PatchSetUtil;
import com.google.gerrit.server.ReviewerSet; import com.google.gerrit.server.ReviewerSet;
import com.google.gerrit.server.Sequences; import com.google.gerrit.server.Sequences;
import com.google.gerrit.server.git.CodeReviewCommit; import com.google.gerrit.server.git.CodeReviewCommit;
@ -96,7 +95,6 @@ public class CherryPickChange {
private final ProjectControl.GenericFactory projectControlFactory; private final ProjectControl.GenericFactory projectControlFactory;
private final ApprovalsUtil approvalsUtil; private final ApprovalsUtil approvalsUtil;
private final ChangeMessagesUtil changeMessagesUtil; private final ChangeMessagesUtil changeMessagesUtil;
private final PatchSetUtil psUtil;
private final NotifyUtil notifyUtil; private final NotifyUtil notifyUtil;
@Inject @Inject
@ -114,7 +112,6 @@ public class CherryPickChange {
ProjectControl.GenericFactory projectControlFactory, ProjectControl.GenericFactory projectControlFactory,
ApprovalsUtil approvalsUtil, ApprovalsUtil approvalsUtil,
ChangeMessagesUtil changeMessagesUtil, ChangeMessagesUtil changeMessagesUtil,
PatchSetUtil psUtil,
NotifyUtil notifyUtil) { NotifyUtil notifyUtil) {
this.dbProvider = dbProvider; this.dbProvider = dbProvider;
this.seq = seq; this.seq = seq;
@ -129,7 +126,6 @@ public class CherryPickChange {
this.projectControlFactory = projectControlFactory; this.projectControlFactory = projectControlFactory;
this.approvalsUtil = approvalsUtil; this.approvalsUtil = approvalsUtil;
this.changeMessagesUtil = changeMessagesUtil; this.changeMessagesUtil = changeMessagesUtil;
this.psUtil = psUtil;
this.notifyUtil = notifyUtil; this.notifyUtil = notifyUtil;
} }
@ -325,12 +321,9 @@ public class CherryPickChange {
throws IOException, OrmException, BadRequestException, ConfigInvalidException { throws IOException, OrmException, BadRequestException, ConfigInvalidException {
Change destChange = destNotes.getChange(); Change destChange = destNotes.getChange();
PatchSet.Id psId = ChangeUtil.nextPatchSetId(git, destChange.currentPatchSetId()); PatchSet.Id psId = ChangeUtil.nextPatchSetId(git, destChange.currentPatchSetId());
PatchSet current = psUtil.current(dbProvider.get(), destNotes);
PatchSetInserter inserter = patchSetInserterFactory.create(destNotes, psId, cherryPickCommit); PatchSetInserter inserter = patchSetInserterFactory.create(destNotes, psId, cherryPickCommit);
inserter inserter
.setMessage("Uploaded patch set " + inserter.getPatchSetId().get() + ".") .setMessage("Uploaded patch set " + inserter.getPatchSetId().get() + ".")
.setDraft(current.isDraft())
.setNotify(input.notify) .setNotify(input.notify)
.setAccountsToNotify(notifyUtil.resolveAccounts(input.notifyDetails)); .setAccountsToNotify(notifyUtil.resolveAccounts(input.notifyDetails));
bu.addOp(destChange.getId(), inserter); bu.addOp(destChange.getId(), inserter);

@ -161,9 +161,9 @@ public class CreateMergePatchSet
rsrc.getId(), rsrc.getId(),
psInserter psInserter
.setMessage("Uploaded patch set " + nextPsId.get() + ".") .setMessage("Uploaded patch set " + nextPsId.get() + ".")
.setDraft(ps.isDraft())
.setNotify(NotifyHandling.NONE) .setNotify(NotifyHandling.NONE)
.setCheckAddPatchSetPermission(false)); .setCheckAddPatchSetPermission(false)
.setNotify(NotifyHandling.NONE));
bu.execute(); bu.execute();
} }

@ -17,18 +17,14 @@ package com.google.gerrit.server.change;
import static com.google.gerrit.extensions.conditions.BooleanCondition.and; import static com.google.gerrit.extensions.conditions.BooleanCondition.and;
import com.google.gerrit.common.TimeUtil; import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.MethodNotAllowedException; import com.google.gerrit.extensions.restapi.MethodNotAllowedException;
import com.google.gerrit.extensions.restapi.Response; import com.google.gerrit.extensions.restapi.Response;
import com.google.gerrit.extensions.restapi.RestApiException; import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.extensions.webui.UiAction; import com.google.gerrit.extensions.webui.UiAction;
import com.google.gerrit.reviewdb.client.Change; import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.server.ReviewDb; import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.change.DeleteChange.Input; import com.google.gerrit.server.change.DeleteChange.Input;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.permissions.ChangePermission; import com.google.gerrit.server.permissions.ChangePermission;
import com.google.gerrit.server.permissions.GlobalPermission;
import com.google.gerrit.server.permissions.PermissionBackend; import com.google.gerrit.server.permissions.PermissionBackend;
import com.google.gerrit.server.permissions.PermissionBackendException; import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.update.BatchUpdate; import com.google.gerrit.server.update.BatchUpdate;
@ -39,7 +35,6 @@ import com.google.gerrit.server.update.UpdateException;
import com.google.inject.Inject; import com.google.inject.Inject;
import com.google.inject.Provider; import com.google.inject.Provider;
import com.google.inject.Singleton; import com.google.inject.Singleton;
import org.eclipse.jgit.lib.Config;
@Singleton @Singleton
public class DeleteChange extends RetryingRestModifyView<ChangeResource, Input, Response<?>> public class DeleteChange extends RetryingRestModifyView<ChangeResource, Input, Response<?>>
@ -48,24 +43,13 @@ public class DeleteChange extends RetryingRestModifyView<ChangeResource, Input,
private final Provider<ReviewDb> db; private final Provider<ReviewDb> db;
private final Provider<DeleteChangeOp> opProvider; private final Provider<DeleteChangeOp> opProvider;
private final Provider<CurrentUser> user;
private final PermissionBackend permissionBackend;
private final boolean allowDrafts;
@Inject @Inject
public DeleteChange( public DeleteChange(
Provider<ReviewDb> db, Provider<ReviewDb> db, RetryHelper retryHelper, Provider<DeleteChangeOp> opProvider) {
RetryHelper retryHelper,
Provider<DeleteChangeOp> opProvider,
Provider<CurrentUser> user,
PermissionBackend permissionBackend,
@GerritServerConfig Config cfg) {
super(retryHelper); super(retryHelper);
this.db = db; this.db = db;
this.opProvider = opProvider; this.opProvider = opProvider;
this.user = user;
this.permissionBackend = permissionBackend;
this.allowDrafts = DeleteChangeOp.allowDrafts(cfg);
} }
@Override @Override
@ -74,16 +58,8 @@ public class DeleteChange extends RetryingRestModifyView<ChangeResource, Input,
throws RestApiException, UpdateException, PermissionBackendException { throws RestApiException, UpdateException, PermissionBackendException {
if (rsrc.getChange().getStatus() == Change.Status.MERGED) { if (rsrc.getChange().getStatus() == Change.Status.MERGED) {
throw new MethodNotAllowedException("delete not permitted"); throw new MethodNotAllowedException("delete not permitted");
} else if (!allowDrafts && rsrc.getChange().getStatus() == Change.Status.DRAFT) {
// If drafts are disabled, only an administrator can delete a draft.
try {
permissionBackend.user(user).check(GlobalPermission.ADMINISTRATE_SERVER);
} catch (AuthException e) {
throw new MethodNotAllowedException("Draft workflow is disabled");
}
} else {
rsrc.permissions().database(db).check(ChangePermission.DELETE);
} }
rsrc.permissions().database(db).check(ChangePermission.DELETE);
try (BatchUpdate bu = try (BatchUpdate bu =
updateFactory.create(db.get(), rsrc.getProject(), rsrc.getUser(), TimeUtil.nowTs())) { updateFactory.create(db.get(), rsrc.getProject(), rsrc.getUser(), TimeUtil.nowTs())) {
@ -115,14 +91,6 @@ public class DeleteChange extends RetryingRestModifyView<ChangeResource, Input,
case MERGED: case MERGED:
// Merged changes should never be deleted. // Merged changes should never be deleted.
return false; return false;
case DRAFT:
if (allowDrafts) {
// Drafts can only be deleted if the server has drafts enabled.
return true;
}
// If drafts are disabled, only administrators may delete.
return permissionBackend.user(user).testOrFalse(GlobalPermission.ADMINISTRATE_SERVER);
} }
return false; return false;
} }

@ -107,19 +107,6 @@ class DeleteChangeOp implements BatchUpdateOp {
id, patchSet.getPatchSetId())); id, patchSet.getPatchSetId()));
} }
} }
if (status == Change.Status.DRAFT) {
for (PatchSet ps : patchSets) {
if (!ps.isDraft()) {
throw new ResourceConflictException(
"Cannot delete draft change "
+ id
+ ": patch set "
+ ps.getPatchSetId()
+ " is not a draft");
}
}
}
} }
private boolean isPatchSetMerged(ChangeContext ctx, PatchSet patchSet) throws IOException { private boolean isPatchSetMerged(ChangeContext ctx, PatchSet patchSet) throws IOException {

@ -1,236 +0,0 @@
// Copyright (C) 2013 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.server.change;
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.common.collect.Iterables;
import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.extensions.registration.DynamicItem;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.MethodNotAllowedException;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.Response;
import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.extensions.webui.UiAction;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.PatchSetInfo;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.PatchSetUtil;
import com.google.gerrit.server.change.DeleteDraftPatchSet.Input;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.patch.PatchSetInfoFactory;
import com.google.gerrit.server.patch.PatchSetInfoNotAvailableException;
import com.google.gerrit.server.permissions.ChangePermission;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.ChangeControl;
import com.google.gerrit.server.project.NoSuchChangeException;
import com.google.gerrit.server.update.BatchUpdate;
import com.google.gerrit.server.update.BatchUpdateOp;
import com.google.gerrit.server.update.ChangeContext;
import com.google.gerrit.server.update.Order;
import com.google.gerrit.server.update.RepoContext;
import com.google.gerrit.server.update.RetryHelper;
import com.google.gerrit.server.update.RetryingRestModifyView;
import com.google.gerrit.server.update.UpdateException;
import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.Singleton;
import java.io.IOException;
import java.util.Collection;
import java.util.Map;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.lib.ObjectId;
@Singleton
public class DeleteDraftPatchSet
extends RetryingRestModifyView<RevisionResource, Input, Response<?>>
implements UiAction<RevisionResource> {
public static class Input {}
private final Provider<ReviewDb> db;
private final PatchSetInfoFactory patchSetInfoFactory;
private final PatchSetUtil psUtil;
private final Provider<DeleteChangeOp> deleteChangeOpProvider;
private final DynamicItem<AccountPatchReviewStore> accountPatchReviewStore;
private final boolean allowDrafts;
private final ChangeControl.GenericFactory changeControlFactory;
@Inject
public DeleteDraftPatchSet(
Provider<ReviewDb> db,
RetryHelper retryHelper,
PatchSetInfoFactory patchSetInfoFactory,
PatchSetUtil psUtil,
Provider<DeleteChangeOp> deleteChangeOpProvider,
DynamicItem<AccountPatchReviewStore> accountPatchReviewStore,
@GerritServerConfig Config cfg,
ChangeControl.GenericFactory changeControlFactory) {
super(retryHelper);
this.db = db;
this.patchSetInfoFactory = patchSetInfoFactory;
this.psUtil = psUtil;
this.deleteChangeOpProvider = deleteChangeOpProvider;
this.accountPatchReviewStore = accountPatchReviewStore;
this.allowDrafts = cfg.getBoolean("change", "allowDrafts", true);
this.changeControlFactory = changeControlFactory;
}
@Override
protected Response<?> applyImpl(
BatchUpdate.Factory updateFactory, RevisionResource rsrc, Input input)
throws RestApiException, UpdateException, OrmException, PermissionBackendException {
if (isDeletingOnlyPatchSet(rsrc)) {
// A change cannot have zero patch sets; the change is deleted instead.
rsrc.permissions().database(db).check(ChangePermission.DELETE);
}
try (BatchUpdate bu =
updateFactory.create(db.get(), rsrc.getProject(), rsrc.getUser(), TimeUtil.nowTs())) {
bu.setOrder(Order.DB_BEFORE_REPO);
bu.addOp(rsrc.getChange().getId(), new Op(rsrc.getPatchSet().getId()));
bu.execute();
}
return Response.none();
}
private boolean isDeletingOnlyPatchSet(RevisionResource rsrc) throws OrmException {
Collection<PatchSet> patchSets = psUtil.byChange(db.get(), rsrc.getNotes());
return patchSets.size() == 1
&& Iterables.getOnlyElement(patchSets).getId().equals(rsrc.getPatchSet().getId());
}
private class Op implements BatchUpdateOp {
private final PatchSet.Id psId;
private Collection<PatchSet> patchSetsBeforeDeletion;
private PatchSet patchSet;
private DeleteChangeOp deleteChangeOp;
private Op(PatchSet.Id psId) {
this.psId = psId;
}
@Override
public boolean updateChange(ChangeContext ctx)
throws RestApiException, OrmException, IOException, NoSuchChangeException {
Map<PatchSet.Id, PatchSet> patchSets = psUtil.byChangeAsMap(db.get(), ctx.getNotes());
patchSet = patchSets.get(psId);
if (patchSet == null) {
return false; // Nothing to do.
}
if (!patchSet.isDraft()) {
throw new ResourceConflictException("Patch set is not a draft");
}
if (!allowDrafts) {
throw new MethodNotAllowedException("Draft workflow is disabled");
}
if (!changeControlFactory
.controlFor(ctx.getNotes(), ctx.getUser())
.canDeleteDraft(ctx.getDb())) {
throw new AuthException("Not permitted to delete this draft patch set");
}
patchSetsBeforeDeletion = patchSets.values();
deleteDraftPatchSet(patchSet, ctx);
deleteOrUpdateDraftChange(ctx, patchSets);
return true;
}
@Override
public void updateRepo(RepoContext ctx) throws IOException {
if (deleteChangeOp != null) {
deleteChangeOp.updateRepo(ctx);
return;
}
ctx.addRefUpdate(
ObjectId.fromString(patchSet.getRevision().get()),
ObjectId.zeroId(),
patchSet.getRefName());
}
private void deleteDraftPatchSet(PatchSet patchSet, ChangeContext ctx) throws OrmException {
// For NoteDb itself, no need to delete these entities, as they are
// automatically filtered out when patch sets are deleted.
psUtil.delete(ctx.getDb(), ctx.getUpdate(patchSet.getId()), patchSet);
accountPatchReviewStore.get().clearReviewed(psId);
// Use the unwrap from DeleteChangeOp to handle BatchUpdateReviewDb.
ReviewDb db = DeleteChangeOp.unwrap(ctx.getDb());
db.changeMessages().delete(db.changeMessages().byPatchSet(psId));
db.patchComments().delete(db.patchComments().byPatchSet(psId));
db.patchSetApprovals().delete(db.patchSetApprovals().byPatchSet(psId));
}
private void deleteOrUpdateDraftChange(ChangeContext ctx, Map<PatchSet.Id, PatchSet> patchSets)
throws OrmException, RestApiException, IOException, NoSuchChangeException {
Change c = ctx.getChange();
if (deletedOnlyPatchSet()) {
deleteChangeOp = deleteChangeOpProvider.get();
deleteChangeOp.updateChange(ctx);
return;
}
if (c.currentPatchSetId().equals(psId)) {
c.setCurrentPatchSet(previousPatchSetInfo(ctx, patchSets));
}
}
private boolean deletedOnlyPatchSet() {
return patchSetsBeforeDeletion.size() == 1
&& patchSetsBeforeDeletion.iterator().next().getId().equals(psId);
}
private PatchSetInfo previousPatchSetInfo(
ChangeContext ctx, Map<PatchSet.Id, PatchSet> patchSets) throws OrmException {
PatchSet.Id prevPsId = null;
for (PatchSet.Id id : patchSets.keySet()) {
if (id.get() < psId.get() && (prevPsId == null || id.get() > prevPsId.get())) {
prevPsId = id;
}
}
try {
// TODO(dborowitz): Get this in a way that doesn't involve re-opening
// the repo after the updateRepo phase.
return patchSetInfoFactory.get(
ctx.getDb(),
ctx.getNotes(),
new PatchSet.Id(psId.getParentKey(), checkNotNull(prevPsId).get()));
} catch (PatchSetInfoNotAvailableException e) {
throw new OrmException(e);
}
}
}
@Override
public UiAction.Description getDescription(RevisionResource rsrc) {
try {
return new UiAction.Description()
.setLabel("Delete")
.setTitle(String.format("Delete draft revision %d", rsrc.getPatchSet().getPatchSetId()))
.setVisible(
allowDrafts
&& rsrc.getPatchSet().isDraft()
&& psUtil.byChange(db.get(), rsrc.getNotes()).size() > 1
&& changeControlFactory
.controlFor(rsrc.getNotes(), rsrc.getUser())
.canDeleteDraft(db.get()));
} catch (OrmException e) {
throw new IllegalStateException(e);
}
}
}

@ -90,7 +90,7 @@ public class GetPureRevert implements RestReadView<ChangeResource> {
throw new ResourceConflictException("current revision is missing"); throw new ResourceConflictException("current revision is missing");
} else if (!changeControlFactory } else if (!changeControlFactory
.controlFor(rsrc.getNotes(), rsrc.getUser()) .controlFor(rsrc.getNotes(), rsrc.getUser())
.isPatchVisible(currentPatchSet, dbProvider.get())) { .isVisible(dbProvider.get())) {
throw new AuthException("current revision not accessible"); throw new AuthException("current revision not accessible");
} }
return getPureRevert(rsrc.getNotes()); return getPureRevert(rsrc.getNotes());

@ -77,7 +77,6 @@ public class Module extends RestApiModule {
delete(CHANGE_KIND).to(DeleteChange.class); delete(CHANGE_KIND).to(DeleteChange.class);
post(CHANGE_KIND, "abandon").to(Abandon.class); post(CHANGE_KIND, "abandon").to(Abandon.class);
post(CHANGE_KIND, "hashtags").to(PostHashtags.class); post(CHANGE_KIND, "hashtags").to(PostHashtags.class);
post(CHANGE_KIND, "publish").to(PublishDraftPatchSet.CurrentRevision.class);
post(CHANGE_KIND, "restore").to(Restore.class); post(CHANGE_KIND, "restore").to(Restore.class);
post(CHANGE_KIND, "revert").to(Revert.class); post(CHANGE_KIND, "revert").to(Revert.class);
post(CHANGE_KIND, "submit").to(Submit.CurrentRevision.class); post(CHANGE_KIND, "submit").to(Submit.CurrentRevision.class);
@ -111,9 +110,7 @@ public class Module extends RestApiModule {
get(REVISION_KIND, "actions").to(GetRevisionActions.class); get(REVISION_KIND, "actions").to(GetRevisionActions.class);
post(REVISION_KIND, "cherrypick").to(CherryPick.class); post(REVISION_KIND, "cherrypick").to(CherryPick.class);
get(REVISION_KIND, "commit").to(GetCommit.class); get(REVISION_KIND, "commit").to(GetCommit.class);
delete(REVISION_KIND).to(DeleteDraftPatchSet.class);
get(REVISION_KIND, "mergeable").to(Mergeable.class); get(REVISION_KIND, "mergeable").to(Mergeable.class);
post(REVISION_KIND, "publish").to(PublishDraftPatchSet.class);
get(REVISION_KIND, "related").to(GetRelated.class); get(REVISION_KIND, "related").to(GetRelated.class);
get(REVISION_KIND, "review").to(GetReview.class); get(REVISION_KIND, "review").to(GetReview.class);
post(REVISION_KIND, "review").to(PostReview.class); post(REVISION_KIND, "review").to(PostReview.class);

@ -140,7 +140,7 @@ public class Move extends RetryingRestModifyView<ChangeResource, MoveInput, Chan
public boolean updateChange(ChangeContext ctx) public boolean updateChange(ChangeContext ctx)
throws OrmException, ResourceConflictException, RepositoryNotFoundException, IOException { throws OrmException, ResourceConflictException, RepositoryNotFoundException, IOException {
change = ctx.getChange(); change = ctx.getChange();
if (change.getStatus() != Status.NEW && change.getStatus() != Status.DRAFT) { if (change.getStatus() != Status.NEW) {
throw new ResourceConflictException("Change is " + ChangeUtil.status(change)); throw new ResourceConflictException("Change is " + ChangeUtil.status(change));
} }

@ -98,7 +98,6 @@ public class PatchSetInserter implements BatchUpdateOp {
private String description; private String description;
private boolean validate = true; private boolean validate = true;
private boolean checkAddPatchSetPermission = true; private boolean checkAddPatchSetPermission = true;
private boolean draft;
private List<String> groups = Collections.emptyList(); private List<String> groups = Collections.emptyList();
private boolean fireRevisionCreated = true; private boolean fireRevisionCreated = true;
private NotifyHandling notify = NotifyHandling.ALL; private NotifyHandling notify = NotifyHandling.ALL;
@ -168,11 +167,6 @@ public class PatchSetInserter implements BatchUpdateOp {
return this; return this;
} }
public PatchSetInserter setDraft(boolean draft) {
this.draft = draft;
return this;
}
public PatchSetInserter setGroups(List<String> groups) { public PatchSetInserter setGroups(List<String> groups) {
checkNotNull(groups, "groups may not be null"); checkNotNull(groups, "groups may not be null");
this.groups = groups; this.groups = groups;
@ -253,7 +247,6 @@ public class PatchSetInserter implements BatchUpdateOp {
ctx.getUpdate(psId), ctx.getUpdate(psId),
psId, psId,
commitId, commitId,
draft,
newGroups, newGroups,
null, null,
description); description);
@ -275,7 +268,7 @@ public class PatchSetInserter implements BatchUpdateOp {
patchSetInfo = patchSetInfo =
patchSetInfoFactory.get(ctx.getRevWalk(), ctx.getRevWalk().parseCommit(commitId), psId); patchSetInfoFactory.get(ctx.getRevWalk(), ctx.getRevWalk().parseCommit(commitId), psId);
if (change.getStatus() != Change.Status.DRAFT && !allowClosed) { if (!allowClosed) {
change.setStatus(Change.Status.NEW); change.setStatus(Change.Status.NEW);
} }
change.setCurrentPatchSet(patchSetInfo); change.setCurrentPatchSet(patchSetInfo);

@ -1,295 +0,0 @@
// Copyright (C) 2013 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.server.change;
import static com.google.gerrit.server.mail.MailUtil.getRecipientsFromFooters;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.common.data.LabelTypes;
import com.google.gerrit.common.errors.EmailException;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
import com.google.gerrit.extensions.restapi.Response;
import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.extensions.webui.UiAction;
import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.ChangeMessage;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.PatchSetInfo;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.ApprovalsUtil;
import com.google.gerrit.server.ChangeMessagesUtil;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.PatchSetUtil;
import com.google.gerrit.server.account.AccountResolver;
import com.google.gerrit.server.change.PublishDraftPatchSet.Input;
import com.google.gerrit.server.extensions.events.DraftPublished;
import com.google.gerrit.server.mail.MailUtil.MailRecipients;
import com.google.gerrit.server.mail.send.CreateChangeSender;
import com.google.gerrit.server.mail.send.ReplacePatchSetSender;
import com.google.gerrit.server.notedb.ChangeUpdate;
import com.google.gerrit.server.patch.PatchSetInfoFactory;
import com.google.gerrit.server.project.ChangeControl;
import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.project.ProjectState;
import com.google.gerrit.server.update.BatchUpdate;
import com.google.gerrit.server.update.BatchUpdateOp;
import com.google.gerrit.server.update.ChangeContext;
import com.google.gerrit.server.update.Context;
import com.google.gerrit.server.update.RetryHelper;
import com.google.gerrit.server.update.RetryingRestModifyView;
import com.google.gerrit.server.update.UpdateException;
import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.Singleton;
import java.io.IOException;
import java.util.Collection;
import java.util.List;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.revwalk.FooterLine;
import org.eclipse.jgit.revwalk.RevCommit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@Singleton
public class PublishDraftPatchSet
extends RetryingRestModifyView<RevisionResource, Input, Response<?>>
implements UiAction<RevisionResource> {
private static final Logger log = LoggerFactory.getLogger(PublishDraftPatchSet.class);
public static class Input {}
private final AccountResolver accountResolver;
private final ApprovalsUtil approvalsUtil;
private final CreateChangeSender.Factory createChangeSenderFactory;
private final PatchSetInfoFactory patchSetInfoFactory;
private final PatchSetUtil psUtil;
private final Provider<ReviewDb> dbProvider;
private final ReplacePatchSetSender.Factory replacePatchSetFactory;
private final DraftPublished draftPublished;
private final ProjectCache projectCache;
private final ChangeControl.GenericFactory changeControlFactory;
@Inject
public PublishDraftPatchSet(
AccountResolver accountResolver,
ApprovalsUtil approvalsUtil,
RetryHelper retryHelper,
CreateChangeSender.Factory createChangeSenderFactory,
PatchSetInfoFactory patchSetInfoFactory,
PatchSetUtil psUtil,
Provider<ReviewDb> dbProvider,
ReplacePatchSetSender.Factory replacePatchSetFactory,
DraftPublished draftPublished,
ProjectCache projectCache,
ChangeControl.GenericFactory changeControlFactory) {
super(retryHelper);
this.accountResolver = accountResolver;
this.approvalsUtil = approvalsUtil;
this.createChangeSenderFactory = createChangeSenderFactory;
this.patchSetInfoFactory = patchSetInfoFactory;
this.psUtil = psUtil;
this.dbProvider = dbProvider;
this.replacePatchSetFactory = replacePatchSetFactory;
this.draftPublished = draftPublished;
this.projectCache = projectCache;
this.changeControlFactory = changeControlFactory;
}
@Override
protected Response<?> applyImpl(
BatchUpdate.Factory updateFactory, RevisionResource rsrc, Input input)
throws RestApiException, UpdateException, IOException {
return apply(
updateFactory,
rsrc.getUser(),
rsrc.getChange(),
rsrc.getPatchSet().getId(),
rsrc.getPatchSet());
}
private Response<?> apply(
BatchUpdate.Factory updateFactory, CurrentUser u, Change c, PatchSet.Id psId, PatchSet ps)
throws RestApiException, UpdateException, IOException {
try (BatchUpdate bu =
updateFactory.create(dbProvider.get(), c.getProject(), u, TimeUtil.nowTs())) {
bu.addOp(c.getId(), new Op(psId, projectCache.checkedGet(c.getProject()), ps));
bu.execute();
}
return Response.none();
}
@Override
public UiAction.Description getDescription(RevisionResource rsrc) {
try {
return new UiAction.Description()
.setLabel("Publish")
.setTitle(String.format("Publish revision %d", rsrc.getPatchSet().getPatchSetId()))
.setVisible(
rsrc.getPatchSet().isDraft()
&& changeControlFactory
.controlFor(rsrc.getNotes(), rsrc.getUser())
.canPublish(dbProvider.get()));
} catch (OrmException e) {
throw new IllegalStateException(e);
}
}
public static class CurrentRevision
extends RetryingRestModifyView<ChangeResource, Input, Response<?>> {
private final PublishDraftPatchSet publish;
@Inject
CurrentRevision(RetryHelper retryHelper, PublishDraftPatchSet publish) {
super(retryHelper);
this.publish = publish;
}
@Override
protected Response<?> applyImpl(
BatchUpdate.Factory updateFactory, ChangeResource rsrc, Input input)
throws RestApiException, UpdateException, IOException {
return publish.apply(
updateFactory,
rsrc.getUser(),
rsrc.getChange(),
rsrc.getChange().currentPatchSetId(),
null);
}
}
private class Op implements BatchUpdateOp {
private final PatchSet.Id psId;
private final ProjectState projectState;
private PatchSet patchSet;
private Change change;
private boolean wasDraftChange;
private PatchSetInfo patchSetInfo;
private MailRecipients recipients;
private Op(PatchSet.Id psId, ProjectState projectState, @Nullable PatchSet patchSet) {
this.psId = psId;
this.projectState = projectState;
this.patchSet = patchSet;
}
@Override
public boolean updateChange(ChangeContext ctx)
throws RestApiException, OrmException, IOException {
if (!changeControlFactory.controlFor(ctx.getNotes(), ctx.getUser()).canPublish(ctx.getDb())) {
throw new AuthException("Cannot publish this draft patch set");
}
if (patchSet == null) {
patchSet = psUtil.get(ctx.getDb(), ctx.getNotes(), psId);
if (patchSet == null) {
throw new ResourceNotFoundException(psId.toString());
}
}
saveChange(ctx);
savePatchSet(ctx);
addReviewers(ctx);
return true;
}
private void saveChange(ChangeContext ctx) {
change = ctx.getChange();
ChangeUpdate update = ctx.getUpdate(psId);
wasDraftChange = change.getStatus() == Change.Status.DRAFT;
if (wasDraftChange) {
change.setStatus(Change.Status.NEW);
update.setStatus(change.getStatus());
}
}
private void savePatchSet(ChangeContext ctx) throws RestApiException, OrmException {
if (!patchSet.isDraft()) {
throw new ResourceConflictException("Patch set is not a draft");
}
psUtil.publish(ctx.getDb(), ctx.getUpdate(psId), patchSet);
}
private void addReviewers(ChangeContext ctx) throws OrmException, IOException {
LabelTypes labelTypes = projectState.getLabelTypes(ctx.getNotes(), ctx.getUser());
Collection<Account.Id> oldReviewers =
approvalsUtil.getReviewers(ctx.getDb(), ctx.getNotes()).all();
RevCommit commit =
ctx.getRevWalk().parseCommit(ObjectId.fromString(patchSet.getRevision().get()));
patchSetInfo = patchSetInfoFactory.get(ctx.getRevWalk(), commit, psId);
List<FooterLine> footerLines = commit.getFooterLines();
recipients =
getRecipientsFromFooters(ctx.getDb(), accountResolver, patchSet.isDraft(), footerLines);
recipients.remove(ctx.getAccountId());
approvalsUtil.addReviewers(
ctx.getDb(),
ctx.getUpdate(psId),
labelTypes,
change,
patchSet,
patchSetInfo,
recipients.getReviewers(),
oldReviewers);
}
@Override
public void postUpdate(Context ctx) throws OrmException {
draftPublished.fire(change, patchSet, ctx.getAccount(), ctx.getWhen());
if (patchSet.isDraft() && change.getStatus() == Change.Status.DRAFT) {
// Skip emails if the patch set is still a draft.
return;
}
try {
if (wasDraftChange) {
sendCreateChange(ctx);
} else {
sendReplacePatchSet(ctx);
}
} catch (EmailException e) {
log.error("Cannot send email for publishing draft " + psId, e);
}
}
private void sendCreateChange(Context ctx) throws EmailException {
CreateChangeSender cm = createChangeSenderFactory.create(ctx.getProject(), change.getId());
cm.setFrom(ctx.getAccountId());
cm.setPatchSet(patchSet, patchSetInfo);
cm.addReviewers(recipients.getReviewers());
cm.addExtraCC(recipients.getCcOnly());
cm.send();
}
private void sendReplacePatchSet(Context ctx) throws EmailException {
ChangeMessage msg =
ChangeMessagesUtil.newMessage(
psId,
ctx.getUser(),
ctx.getWhen(),
"Uploaded patch set " + psId.get() + ".",
ChangeMessagesUtil.TAG_UPLOADED_PATCH_SET);
ReplacePatchSetSender cm = replacePatchSetFactory.create(ctx.getProject(), change.getId());
cm.setFrom(ctx.getAccountId());
cm.setPatchSet(patchSet, patchSetInfo);
cm.setChangeMessage(msg.getMessage(), ctx.getWhen());
cm.addReviewers(recipients.getReviewers());
cm.addExtraCC(recipients.getCcOnly());
cm.send();
}
}
}

@ -111,7 +111,7 @@ public class PutMessage
throw new ResourceConflictException("current revision is missing"); throw new ResourceConflictException("current revision is missing");
} else if (!changeControlFactory } else if (!changeControlFactory
.controlFor(resource.getNotes(), resource.getUser()) .controlFor(resource.getNotes(), resource.getUser())
.isPatchVisible(ps, db.get())) { .isVisible(db.get())) {
throw new AuthException("current revision not accessible"); throw new AuthException("current revision not accessible");
} }

@ -158,7 +158,7 @@ public class Rebase extends RetryingRestModifyView<RevisionResource, RebaseInput
} }
PatchSet.Id baseId = base.patchSet().getId(); PatchSet.Id baseId = base.patchSet().getId();
ChangeControl baseCtl = changeControlFactory.controlFor(base.notes(), userProvider.get()); ChangeControl baseCtl = changeControlFactory.controlFor(base.notes(), userProvider.get());
if (!baseCtl.isPatchVisible(base.patchSet(), db)) { if (!baseCtl.isVisible(db)) {
throw new AuthException("base revision not accessible: " + str); throw new AuthException("base revision not accessible: " + str);
} else if (change.getId().equals(baseId.getParentKey())) { } else if (change.getId().equals(baseId.getParentKey())) {
throw new ResourceConflictException("cannot rebase change onto itself"); throw new ResourceConflictException("cannot rebase change onto itself");
@ -252,7 +252,7 @@ public class Rebase extends RetryingRestModifyView<RevisionResource, RebaseInput
throw new ResourceConflictException("current revision is missing"); throw new ResourceConflictException("current revision is missing");
} else if (!changeControlFactory } else if (!changeControlFactory
.controlFor(rsrc.getNotes(), rsrc.getUser()) .controlFor(rsrc.getNotes(), rsrc.getUser())
.isPatchVisible(ps, rebase.dbProvider.get())) { .isVisible(rebase.dbProvider.get())) {
throw new AuthException("current revision not accessible"); throw new AuthException("current revision not accessible");
} }
return rebase.applyImpl(updateFactory, new RevisionResource(rsrc, ps), input); return rebase.applyImpl(updateFactory, new RevisionResource(rsrc, ps), input);

@ -186,7 +186,6 @@ public class RebaseChangeOp implements BatchUpdateOp {
patchSetInserterFactory patchSetInserterFactory
.create(notes, rebasedPatchSetId, rebasedCommit) .create(notes, rebasedPatchSetId, rebasedCommit)
.setDescription("Rebase") .setDescription("Rebase")
.setDraft(originalPatchSet.isDraft())
.setNotify(NotifyHandling.NONE) .setNotify(NotifyHandling.NONE)
.setFireRevisionCreated(fireRevisionCreated) .setFireRevisionCreated(fireRevisionCreated)
.setCopyApprovals(copyApprovals) .setCopyApprovals(copyApprovals)

@ -129,11 +129,7 @@ public class ReviewerJson {
PatchSet ps = cd.currentPatchSet(); PatchSet ps = cd.currentPatchSet();
if (ps != null) { if (ps != null) {
for (SubmitRecord rec : for (SubmitRecord rec :
submitRuleEvaluatorFactory submitRuleEvaluatorFactory.create(user, cd).setFastEvalLabels(true).evaluate()) {
.create(user, cd)
.setFastEvalLabels(true)
.setAllowDraft(true)
.evaluate()) {
if (rec.labels == null) { if (rec.labels == null) {
continue; continue;
} }

@ -77,7 +77,7 @@ public class Revisions implements ChildCollection<ChangeResource, RevisionResour
throws ResourceNotFoundException, AuthException, OrmException, IOException { throws ResourceNotFoundException, AuthException, OrmException, IOException {
if (id.get().equals("current")) { if (id.get().equals("current")) {
PatchSet ps = psUtil.current(dbProvider.get(), change.getNotes()); PatchSet ps = psUtil.current(dbProvider.get(), change.getNotes());
if (ps != null && visible(change, ps)) { if (ps != null && visible(change)) {
return new RevisionResource(change, ps).doNotCache(); return new RevisionResource(change, ps).doNotCache();
} }
throw new ResourceNotFoundException(id); throw new ResourceNotFoundException(id);
@ -85,7 +85,7 @@ public class Revisions implements ChildCollection<ChangeResource, RevisionResour
List<RevisionResource> match = Lists.newArrayListWithExpectedSize(2); List<RevisionResource> match = Lists.newArrayListWithExpectedSize(2);
for (RevisionResource rsrc : find(change, id.get())) { for (RevisionResource rsrc : find(change, id.get())) {
if (visible(change, rsrc.getPatchSet())) { if (visible(change)) {
match.add(rsrc); match.add(rsrc);
} }
} }
@ -100,10 +100,10 @@ public class Revisions implements ChildCollection<ChangeResource, RevisionResour
} }
} }
private boolean visible(ChangeResource change, PatchSet ps) throws OrmException { private boolean visible(ChangeResource change) throws OrmException {
return changeControlFactory return changeControlFactory
.controlFor(change.getNotes(), change.getUser()) .controlFor(change.getNotes(), change.getUser())
.isPatchVisible(ps, dbProvider.get()); .isVisible(dbProvider.get());
} }
private List<RevisionResource> find(ChangeResource change, String id) private List<RevisionResource> find(ChangeResource change, String id)

@ -254,7 +254,6 @@ public class Submit
} }
// $FALL-THROUGH$ // $FALL-THROUGH$
case ABANDONED: case ABANDONED:
case DRAFT:
default: default:
throw new ResourceConflictException("change is " + ChangeUtil.status(change)); throw new ResourceConflictException("change is " + ChangeUtil.status(change));
} }
@ -316,7 +315,6 @@ public class Submit
Change change = resource.getChange(); Change change = resource.getChange();
if (!change.getStatus().isOpen() if (!change.getStatus().isOpen()
|| !resource.isCurrent() || !resource.isCurrent()
|| resource.getPatchSet().isDraft()
|| !resource.permissions().testOrFalse(ChangePermission.SUBMIT)) { || !resource.permissions().testOrFalse(ChangePermission.SUBMIT)) {
return null; // submit not visible return null; // submit not visible
} }
@ -534,7 +532,7 @@ public class Submit
throw new ResourceConflictException("current revision is missing"); throw new ResourceConflictException("current revision is missing");
} else if (!changeControlFactory } else if (!changeControlFactory
.controlFor(rsrc.getNotes(), rsrc.getUser()) .controlFor(rsrc.getNotes(), rsrc.getUser())
.isPatchVisible(ps, dbProvider.get())) { .isVisible(dbProvider.get())) {
throw new AuthException("current revision not accessible"); throw new AuthException("current revision not accessible");
} }

@ -41,7 +41,6 @@ import com.google.gerrit.extensions.events.ChangeMergedListener;
import com.google.gerrit.extensions.events.ChangeRestoredListener; import com.google.gerrit.extensions.events.ChangeRestoredListener;
import com.google.gerrit.extensions.events.ChangeRevertedListener; import com.google.gerrit.extensions.events.ChangeRevertedListener;
import com.google.gerrit.extensions.events.CommentAddedListener; import com.google.gerrit.extensions.events.CommentAddedListener;
import com.google.gerrit.extensions.events.DraftPublishedListener;
import com.google.gerrit.extensions.events.GarbageCollectorListener; import com.google.gerrit.extensions.events.GarbageCollectorListener;
import com.google.gerrit.extensions.events.GitReferenceUpdatedListener; import com.google.gerrit.extensions.events.GitReferenceUpdatedListener;
import com.google.gerrit.extensions.events.GroupIndexedListener; import com.google.gerrit.extensions.events.GroupIndexedListener;
@ -313,7 +312,6 @@ public class GerritGlobalModule extends FactoryModule {
DynamicSet.setOf(binder(), AssigneeChangedListener.class); DynamicSet.setOf(binder(), AssigneeChangedListener.class);
DynamicSet.setOf(binder(), ChangeAbandonedListener.class); DynamicSet.setOf(binder(), ChangeAbandonedListener.class);
DynamicSet.setOf(binder(), CommentAddedListener.class); DynamicSet.setOf(binder(), CommentAddedListener.class);
DynamicSet.setOf(binder(), DraftPublishedListener.class);
DynamicSet.setOf(binder(), HashtagsEditedListener.class); DynamicSet.setOf(binder(), HashtagsEditedListener.class);
DynamicSet.setOf(binder(), ChangeMergedListener.class); DynamicSet.setOf(binder(), ChangeMergedListener.class);
DynamicSet.setOf(binder(), ChangeRestoredListener.class); DynamicSet.setOf(binder(), ChangeRestoredListener.class);

@ -25,7 +25,6 @@ public class PatchSetAttribute {
public AccountAttribute uploader; public AccountAttribute uploader;
public Long createdOn; public Long createdOn;
public AccountAttribute author; public AccountAttribute author;
public boolean isDraft;
public ChangeKind kind; public ChangeKind kind;
public List<ApprovalAttribute> approvals; public List<ApprovalAttribute> approvals;

@ -26,7 +26,6 @@ import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.RestApiException; import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.reviewdb.client.Account; import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.Change; import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.Change.Status;
import com.google.gerrit.reviewdb.client.PatchSet; import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.reviewdb.client.RefNames; import com.google.gerrit.reviewdb.client.RefNames;
import com.google.gerrit.reviewdb.server.ReviewDb; import com.google.gerrit.reviewdb.server.ReviewDb;
@ -205,11 +204,7 @@ public class ChangeEditUtil {
try (BatchUpdate bu = try (BatchUpdate bu =
updateFactory.create(db.get(), change.getProject(), user, TimeUtil.nowTs())) { updateFactory.create(db.get(), change.getProject(), user, TimeUtil.nowTs())) {
bu.setRepository(repo, rw, oi); bu.setRepository(repo, rw, oi);
bu.addOp( bu.addOp(change.getId(), inserter.setMessage(message.toString()));
change.getId(),
inserter
.setDraft(change.getStatus() == Status.DRAFT || basePatchSet.isDraft())
.setMessage(message.toString()));
bu.addOp( bu.addOp(
change.getId(), change.getId(),
new BatchUpdateOp() { new BatchUpdateOp() {

@ -1,28 +0,0 @@
// Copyright (C) 2012 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.server.events;
import com.google.common.base.Supplier;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.server.data.AccountAttribute;
public class DraftPublishedEvent extends PatchSetEvent {
static final String TYPE = "draft-published";
public Supplier<AccountAttribute> uploader;
public DraftPublishedEvent(Change change) {
super(TYPE, change);
}
}

@ -470,7 +470,6 @@ public class EventFactory {
p.ref = patchSet.getRefName(); p.ref = patchSet.getRefName();
p.uploader = asAccountAttribute(patchSet.getUploader()); p.uploader = asAccountAttribute(patchSet.getUploader());
p.createdOn = patchSet.getCreatedOn().getTime() / 1000L; p.createdOn = patchSet.getCreatedOn().getTime() / 1000L;
p.isDraft = patchSet.isDraft();
PatchSet.Id pId = patchSet.getId(); PatchSet.Id pId = patchSet.getId();
try { try {
p.parents = new ArrayList<>(); p.parents = new ArrayList<>();

@ -28,7 +28,6 @@ public class EventTypes {
register(ChangeRestoredEvent.TYPE, ChangeRestoredEvent.class); register(ChangeRestoredEvent.TYPE, ChangeRestoredEvent.class);
register(CommentAddedEvent.TYPE, CommentAddedEvent.class); register(CommentAddedEvent.TYPE, CommentAddedEvent.class);
register(CommitReceivedEvent.TYPE, CommitReceivedEvent.class); register(CommitReceivedEvent.TYPE, CommitReceivedEvent.class);
register(DraftPublishedEvent.TYPE, DraftPublishedEvent.class);
register(HashtagsChangedEvent.TYPE, HashtagsChangedEvent.class); register(HashtagsChangedEvent.TYPE, HashtagsChangedEvent.class);
register(RefUpdatedEvent.TYPE, RefUpdatedEvent.class); register(RefUpdatedEvent.TYPE, RefUpdatedEvent.class);
register(RefReceivedEvent.TYPE, RefReceivedEvent.class); register(RefReceivedEvent.TYPE, RefReceivedEvent.class);

@ -29,7 +29,6 @@ import com.google.gerrit.extensions.events.ChangeAbandonedListener;
import com.google.gerrit.extensions.events.ChangeMergedListener; import com.google.gerrit.extensions.events.ChangeMergedListener;
import com.google.gerrit.extensions.events.ChangeRestoredListener; import com.google.gerrit.extensions.events.ChangeRestoredListener;
import com.google.gerrit.extensions.events.CommentAddedListener; import com.google.gerrit.extensions.events.CommentAddedListener;
import com.google.gerrit.extensions.events.DraftPublishedListener;
import com.google.gerrit.extensions.events.GitReferenceUpdatedListener; import com.google.gerrit.extensions.events.GitReferenceUpdatedListener;
import com.google.gerrit.extensions.events.HashtagsEditedListener; import com.google.gerrit.extensions.events.HashtagsEditedListener;
import com.google.gerrit.extensions.events.NewProjectCreatedListener; import com.google.gerrit.extensions.events.NewProjectCreatedListener;
@ -79,7 +78,6 @@ public class StreamEventsApiListener
ChangeMergedListener, ChangeMergedListener,
ChangeRestoredListener, ChangeRestoredListener,
CommentAddedListener, CommentAddedListener,
DraftPublishedListener,
GitReferenceUpdatedListener, GitReferenceUpdatedListener,
HashtagsEditedListener, HashtagsEditedListener,
NewProjectCreatedListener, NewProjectCreatedListener,
@ -98,7 +96,6 @@ public class StreamEventsApiListener
DynamicSet.bind(binder(), ChangeMergedListener.class).to(StreamEventsApiListener.class); DynamicSet.bind(binder(), ChangeMergedListener.class).to(StreamEventsApiListener.class);
DynamicSet.bind(binder(), ChangeRestoredListener.class).to(StreamEventsApiListener.class); DynamicSet.bind(binder(), ChangeRestoredListener.class).to(StreamEventsApiListener.class);
DynamicSet.bind(binder(), CommentAddedListener.class).to(StreamEventsApiListener.class); DynamicSet.bind(binder(), CommentAddedListener.class).to(StreamEventsApiListener.class);
DynamicSet.bind(binder(), DraftPublishedListener.class).to(StreamEventsApiListener.class);
DynamicSet.bind(binder(), GitReferenceUpdatedListener.class) DynamicSet.bind(binder(), GitReferenceUpdatedListener.class)
.to(StreamEventsApiListener.class); .to(StreamEventsApiListener.class);
DynamicSet.bind(binder(), HashtagsEditedListener.class).to(StreamEventsApiListener.class); DynamicSet.bind(binder(), HashtagsEditedListener.class).to(StreamEventsApiListener.class);
@ -391,24 +388,6 @@ public class StreamEventsApiListener
} }
} }
@Override
public void onDraftPublished(DraftPublishedListener.Event ev) {
try {
ChangeNotes notes = getNotes(ev.getChange());
Change change = notes.getChange();
PatchSet ps = getPatchSet(notes, ev.getRevision());
DraftPublishedEvent event = new DraftPublishedEvent(change);
event.change = changeAttributeSupplier(change);
event.patchSet = patchSetAttributeSupplier(change, ps);
event.uploader = accountAttributeSupplier(ev.getWho());
dispatcher.get().postEvent(change, event);
} catch (OrmException | PermissionBackendException e) {
log.error("Failed to dispatch event", e);
}
}
@Override @Override
public void onCommentAdded(CommentAddedListener.Event ev) { public void onCommentAdded(CommentAddedListener.Event ev) {
try { try {

@ -1,73 +0,0 @@
// Copyright (C) 2015 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.server.extensions.events;
import com.google.gerrit.extensions.api.changes.NotifyHandling;
import com.google.gerrit.extensions.common.AccountInfo;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.RevisionInfo;
import com.google.gerrit.extensions.events.DraftPublishedListener;
import com.google.gerrit.extensions.registration.DynamicSet;
import com.google.gerrit.reviewdb.client.Account;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.client.PatchSet;
import com.google.gerrit.server.GpgException;
import com.google.gerrit.server.patch.PatchListNotAvailableException;
import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject;
import java.io.IOException;
import java.sql.Timestamp;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class DraftPublished {
private static final Logger log = LoggerFactory.getLogger(DraftPublished.class);
private final DynamicSet<DraftPublishedListener> listeners;
private final EventUtil util;
@Inject
public DraftPublished(DynamicSet<DraftPublishedListener> listeners, EventUtil util) {
this.listeners = listeners;
this.util = util;
}
public void fire(Change change, PatchSet patchSet, Account accountId, Timestamp when) {
try {
Event event =
new Event(
util.changeInfo(change),
util.revisionInfo(change.getProject(), patchSet),
util.accountInfo(accountId),
when);
for (DraftPublishedListener l : listeners) {
try {
l.onDraftPublished(event);
} catch (Exception e) {
util.logEventListenerError(this, l, e);
}
}
} catch (PatchListNotAvailableException | GpgException | IOException | OrmException e) {
log.error("Couldn't fire event", e);
}
}
private static class Event extends AbstractRevisionEvent implements DraftPublishedListener.Event {
Event(ChangeInfo change, RevisionInfo revision, AccountInfo publisher, Timestamp when) {
super(change, revision, publisher, when, NotifyHandling.ALL);
}
}
}

@ -98,8 +98,6 @@ public class AbandonOp implements BatchUpdateOp {
ChangeUpdate update = ctx.getUpdate(psId); ChangeUpdate update = ctx.getUpdate(psId);
if (!change.getStatus().isOpen()) { if (!change.getStatus().isOpen()) {
throw new ResourceConflictException("change is " + ChangeUtil.status(change)); throw new ResourceConflictException("change is " + ChangeUtil.status(change));
} else if (change.getStatus() == Change.Status.DRAFT) {
throw new ResourceConflictException("draft changes cannot be abandoned");
} }
patchSet = psUtil.get(ctx.getDb(), ctx.getNotes(), psId); patchSet = psUtil.get(ctx.getDb(), ctx.getNotes(), psId);
change.setStatus(Change.Status.ABANDONED); change.setStatus(Change.Status.ABANDONED);

@ -26,9 +26,6 @@ public interface ChangeReportFormatter {
@Nullable @Nullable
public abstract String subject(); public abstract String subject();
@Nullable
public abstract Boolean isDraft();
@Nullable @Nullable
public abstract Boolean isEdit(); public abstract Boolean isEdit();
@ -48,8 +45,6 @@ public interface ChangeReportFormatter {
public abstract Builder setSubject(String val); public abstract Builder setSubject(String val);
public abstract Builder setIsDraft(Boolean val);
public abstract Builder setIsEdit(Boolean val); public abstract Builder setIsEdit(Boolean val);
public abstract Builder setIsPrivate(Boolean val); public abstract Builder setIsPrivate(Boolean val);
@ -60,8 +55,6 @@ public interface ChangeReportFormatter {
abstract String subject(); abstract String subject();
abstract Boolean isDraft();
abstract Boolean isEdit(); abstract Boolean isEdit();
abstract Boolean isPrivate(); abstract Boolean isPrivate();
@ -73,7 +66,6 @@ public interface ChangeReportFormatter {
public Input build() { public Input build() {
setChange(change()); setChange(change());
setSubject(subject() == null ? change().getSubject() : subject()); setSubject(subject() == null ? change().getSubject() : subject());
setIsDraft(isDraft() == null ? Change.Status.DRAFT == change().getStatus() : isDraft());
setIsEdit(isEdit() == null ? false : isEdit()); setIsEdit(isEdit() == null ? false : isEdit());
setIsPrivate(isPrivate() == null ? change().isPrivate() : isPrivate()); setIsPrivate(isPrivate() == null ? change().isPrivate() : isPrivate());
setIsWorkInProgress( setIsWorkInProgress(

@ -49,9 +49,6 @@ public class DefaultChangeReportFormatter implements ChangeReportFormatter {
.append(ChangeUtil.formatChangeUrl(url, input.change())) .append(ChangeUtil.formatChangeUrl(url, input.change()))
.append(" ") .append(" ")
.append(ChangeUtil.cropSubject(input.subject())); .append(ChangeUtil.cropSubject(input.subject()));
if (input.isDraft()) {
m.append(" [DRAFT]");
}
if (input.isEdit()) { if (input.isEdit()) {
m.append(" [EDIT]"); m.append(" [EDIT]");
} }

@ -58,7 +58,6 @@ import com.google.gerrit.common.Nullable;
import com.google.gerrit.common.TimeUtil; import com.google.gerrit.common.TimeUtil;
import com.google.gerrit.common.data.LabelType; import com.google.gerrit.common.data.LabelType;
import com.google.gerrit.common.data.LabelTypes; import com.google.gerrit.common.data.LabelTypes;
import com.google.gerrit.common.data.Permission;
import com.google.gerrit.common.data.PermissionRule; import com.google.gerrit.common.data.PermissionRule;
import com.google.gerrit.extensions.api.changes.HashtagsInput; import com.google.gerrit.extensions.api.changes.HashtagsInput;
import com.google.gerrit.extensions.api.changes.NotifyHandling; import com.google.gerrit.extensions.api.changes.NotifyHandling;
@ -662,7 +661,6 @@ class ReceiveCommits {
ChangeReportFormatter.Input.builder() ChangeReportFormatter.Input.builder()
.setChange(u.notes.getChange()) .setChange(u.notes.getChange())
.setSubject(subject) .setSubject(subject)
.setIsDraft(u.replaceOp != null && u.replaceOp.getPatchSet().isDraft())
.setIsEdit(edit) .setIsEdit(edit)
.setIsPrivate(isPrivate) .setIsPrivate(isPrivate)
.setIsWorkInProgress(wip) .setIsWorkInProgress(wip)
@ -1239,11 +1237,6 @@ class ReceiveCommits {
cc.add(id); cc.add(id);
} }
@Option(name = "--publish", usage = "publish new/updated changes")
void publish(boolean publish) {
draft = !publish;
}
@Option( @Option(
name = "--label", name = "--label",
aliases = {"-l"}, aliases = {"-l"},
@ -1473,21 +1466,6 @@ class ReceiveCommits {
return; return;
} }
if (magicBranch.draft) {
// TODO(xchangcheng): reject all after repo-tool supports private and wip changes.
if (!receiveConfig.allowDrafts) {
errors.put(ReceiveError.CODE_REVIEW, ref);
reject(cmd, "draft workflow is disabled");
return;
} else if (projectControl
.controlForRef(MagicBranch.NEW_DRAFT_CHANGE + ref)
.isBlocked(Permission.PUSH)) {
errors.put(ReceiveError.CODE_REVIEW, ref);
reject(cmd, "cannot upload drafts");
return;
}
}
try { try {
magicBranch.perm.check(RefPermission.CREATE_CHANGE); magicBranch.perm.check(RefPermission.CREATE_CHANGE);
} catch (AuthException denied) { } catch (AuthException denied) {
@ -1511,11 +1489,6 @@ class ReceiveCommits {
return; return;
} }
if (magicBranch.draft && magicBranch.submit) {
reject(cmd, "cannot submit draft");
return;
}
if (magicBranch.submit) { if (magicBranch.submit) {
try { try {
permissions.ref(ref).check(RefPermission.UPDATE_BY_SUBMIT); permissions.ref(ref).check(RefPermission.UPDATE_BY_SUBMIT);
@ -1539,10 +1512,6 @@ class ReceiveCommits {
String destBranch = magicBranch.dest.get(); String destBranch = magicBranch.dest.get();
try { try {
if (magicBranch.merged) { if (magicBranch.merged) {
if (magicBranch.draft) {
reject(cmd, "cannot be draft & merged");
return;
}
if (magicBranch.base != null) { if (magicBranch.base != null) {
reject(cmd, "cannot use merged with base"); reject(cmd, "cannot use merged with base");
return; return;
@ -2157,7 +2126,7 @@ class ReceiveCommits {
checkNotNull(magicBranch); checkNotNull(magicBranch);
recipients.add(magicBranch.getMailRecipients()); recipients.add(magicBranch.getMailRecipients());
approvals = magicBranch.labels; approvals = magicBranch.labels;
recipients.add(getRecipientsFromFooters(db, accountResolver, false, footerLines)); recipients.add(getRecipientsFromFooters(db, accountResolver, footerLines));
recipients.remove(me); recipients.remove(me);
StringBuilder msg = StringBuilder msg =
new StringBuilder( new StringBuilder(

@ -27,7 +27,6 @@ import org.eclipse.jgit.lib.Config;
class ReceiveConfig { class ReceiveConfig {
final boolean checkMagicRefs; final boolean checkMagicRefs;
final boolean checkReferencedObjectsAreReachable; final boolean checkReferencedObjectsAreReachable;
final boolean allowDrafts;
final int maxBatchCommits; final int maxBatchCommits;
private final int systemMaxBatchChanges; private final int systemMaxBatchChanges;
private final AccountLimits.Factory limitsFactory; private final AccountLimits.Factory limitsFactory;
@ -37,7 +36,6 @@ class ReceiveConfig {
checkMagicRefs = config.getBoolean("receive", null, "checkMagicRefs", true); checkMagicRefs = config.getBoolean("receive", null, "checkMagicRefs", true);
checkReferencedObjectsAreReachable = checkReferencedObjectsAreReachable =
config.getBoolean("receive", null, "checkReferencedObjectsAreReachable", true); config.getBoolean("receive", null, "checkReferencedObjectsAreReachable", true);
allowDrafts = config.getBoolean("change", null, "allowDrafts", true);
maxBatchCommits = config.getInt("receive", null, "maxBatchCommits", 10000); maxBatchCommits = config.getInt("receive", null, "maxBatchCommits", 10000);
systemMaxBatchChanges = config.getInt("receive", "maxBatchChanges", 0); systemMaxBatchChanges = config.getInt("receive", "maxBatchChanges", 0);
this.limitsFactory = limitsFactory; this.limitsFactory = limitsFactory;

@ -279,10 +279,6 @@ public class ReplaceOp implements BatchUpdateOp {
} }
} }
boolean draft = magicBranch != null && magicBranch.draft;
if (change.getStatus() == Change.Status.DRAFT && !draft) {
update.setStatus(Change.Status.NEW);
}
newPatchSet = newPatchSet =
psUtil.insert( psUtil.insert(
ctx.getDb(), ctx.getDb(),
@ -290,14 +286,12 @@ public class ReplaceOp implements BatchUpdateOp {
update, update,
patchSetId, patchSetId,
commitId, commitId,
draft,
groups, groups,
pushCertificate != null ? pushCertificate.toTextWithSignature() : null, pushCertificate != null ? pushCertificate.toTextWithSignature() : null,
psDescription); psDescription);
update.setPsDescription(psDescription); update.setPsDescription(psDescription);
recipients.add( recipients.add(getRecipientsFromFooters(ctx.getDb(), accountResolver, commit.getFooterLines()));
getRecipientsFromFooters(ctx.getDb(), accountResolver, draft, commit.getFooterLines()));
recipients.remove(ctx.getAccountId()); recipients.remove(ctx.getAccountId());
ChangeData cd = changeDataFactory.create(ctx.getDb(), ctx.getNotes()); ChangeData cd = changeDataFactory.create(ctx.getDb(), ctx.getNotes());
MailRecipients oldRecipients = getRecipientsFromReviewers(cd.reviewers()); MailRecipients oldRecipients = getRecipientsFromReviewers(cd.reviewers());
@ -430,11 +424,7 @@ public class ReplaceOp implements BatchUpdateOp {
if (magicBranch != null && magicBranch.topic != null) { if (magicBranch != null && magicBranch.topic != null) {
change.setTopic(magicBranch.topic); change.setTopic(magicBranch.topic);
} }
if (change.getStatus() == Change.Status.DRAFT && newPatchSet.isDraft()) { change.setStatus(Change.Status.NEW);
// Leave in draft status.
} else {
change.setStatus(Change.Status.NEW);
}
change.setCurrentPatchSet(info); change.setCurrentPatchSet(info);
List<String> idList = commit.getFooterLines(CHANGE_ID); List<String> idList = commit.getFooterLines(CHANGE_ID);

@ -157,7 +157,6 @@ public class CherryPick extends SubmitStrategy {
ctx.getUpdate(psId), ctx.getUpdate(psId),
psId, psId,
newCommit, newCommit,
false,
prevPs != null ? prevPs.getGroups() : ImmutableList.<String>of(), prevPs != null ? prevPs.getGroups() : ImmutableList.<String>of(),
null, null,
null); null);

@ -222,7 +222,6 @@ public class RebaseSubmitStrategy extends SubmitStrategy {
ctx.getUpdate(newPatchSetId), ctx.getUpdate(newPatchSetId),
newPatchSetId, newPatchSetId,
newCommit, newCommit,
false,
prevPs != null ? prevPs.getGroups() : ImmutableList.<String>of(), prevPs != null ? prevPs.getGroups() : ImmutableList.<String>of(),
null, null,
null); null);

@ -324,7 +324,6 @@ abstract class SubmitStrategyOp implements BatchUpdateOp {
ctx.getUpdate(psId), ctx.getUpdate(psId),
psId, psId,
alreadyMergedCommit, alreadyMergedCommit,
false,
groups, groups,
null, null,
null); null);

@ -566,7 +566,7 @@ public class ChangeField {
// Submit rule options in this class should never use fastEvalLabels. This // Submit rule options in this class should never use fastEvalLabels. This
// slows down indexing slightly but produces correct search results. // slows down indexing slightly but produces correct search results.
public static final SubmitRuleOptions SUBMIT_RULE_OPTIONS_LENIENT = public static final SubmitRuleOptions SUBMIT_RULE_OPTIONS_LENIENT =
SubmitRuleOptions.defaults().allowClosed(true).allowDraft(true).build(); SubmitRuleOptions.defaults().allowClosed(true).build();
public static final SubmitRuleOptions SUBMIT_RULE_OPTIONS_STRICT = public static final SubmitRuleOptions SUBMIT_RULE_OPTIONS_STRICT =
SubmitRuleOptions.defaults().build(); SubmitRuleOptions.defaults().build();

@ -90,6 +90,9 @@ public class ChangeSchemaDefinitions extends SchemaDefinitions<ChangeData> {
static final Schema<ChangeData> V46 = schema(V45); static final Schema<ChangeData> V46 = schema(V45);
// Removal of draft change workflow requires reindexing
static final Schema<ChangeData> V47 = schema(V46);
public static final String NAME = "changes"; public static final String NAME = "changes";
public static final ChangeSchemaDefinitions INSTANCE = new ChangeSchemaDefinitions(); public static final ChangeSchemaDefinitions INSTANCE = new ChangeSchemaDefinitions();

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