diff --git a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/api/DepositApiClient.js b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/api/DepositApiClient.js
index 0d35c2c8d..7a443698a 100644
--- a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/api/DepositApiClient.js
+++ b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/api/DepositApiClient.js
@@ -103,7 +103,12 @@ export class RDMDepositApiClient extends DepositApiClient {
);
return new DepositApiClientResponse(data, errors);
} catch (error) {
- const errorData = error.response.data;
+ let errorData = error.response.data;
+ const errors = this.recordSerializer.deserializeErrors(
+ error.response.data.errors || []
+ );
+ // this is to serialize raised error from the backend on publish
+ if (errors) errorData = errors;
throw new DepositApiClientResponse({}, errorData);
}
}
diff --git a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/controls/PublishButton/PublishButton.js b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/controls/PublishButton/PublishButton.js
index 71dafb704..25f07a2b0 100644
--- a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/controls/PublishButton/PublishButton.js
+++ b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/controls/PublishButton/PublishButton.js
@@ -18,6 +18,8 @@ import {
DepositFormSubmitContext,
} from "../../api/DepositFormSubmitContext";
import { DRAFT_PUBLISH_STARTED } from "../../state/types";
+import { scrollTop } from "../../utils";
+import { DRAFT_PUBLISH_FAILED_WITH_VALIDATION_ERRORS } from "../../state/types";
class PublishButtonComponent extends Component {
state = { isConfirmModalOpen: false };
@@ -30,14 +32,28 @@ class PublishButtonComponent extends Component {
handlePublish = (event, handleSubmit, publishWithoutCommunity) => {
const { setSubmitContext } = this.context;
-
- setSubmitContext(
- publishWithoutCommunity
- ? DepositFormSubmitActions.PUBLISH_WITHOUT_COMMUNITY
- : DepositFormSubmitActions.PUBLISH
- );
- handleSubmit(event);
- this.closeConfirmModal();
+ const { formik, raiseDOINeededButNotReserved } = this.props;
+ const noINeedOne = formik?.values?.noINeedOne;
+ if (noINeedOne && Object.keys(formik?.values?.pids).length === 0) {
+ const errors = {
+ pids: {
+ doi: i18next.t("DOI is needed. Please click on the button to reserve it."),
+ },
+ };
+ formik.setErrors(errors);
+ raiseDOINeededButNotReserved(formik?.values, errors);
+ this.closeConfirmModal();
+ // scroll top to show the global error
+ scrollTop();
+ } else {
+ setSubmitContext(
+ publishWithoutCommunity
+ ? DepositFormSubmitActions.PUBLISH_WITHOUT_COMMUNITY
+ : DepositFormSubmitActions.PUBLISH
+ );
+ handleSubmit(event);
+ this.closeConfirmModal();
+ }
};
isDisabled = (values, isSubmitting, filesState) => {
@@ -139,6 +155,7 @@ PublishButtonComponent.propTypes = {
formik: PropTypes.object.isRequired,
publishModalExtraContent: PropTypes.string,
filesState: PropTypes.object,
+ raiseDOINeededButNotReserved: PropTypes.func.isRequired,
};
PublishButtonComponent.defaultProps = {
@@ -155,7 +172,12 @@ const mapStateToProps = (state) => ({
filesState: state.files,
});
-export const PublishButton = connect(
- mapStateToProps,
- null
-)(connectFormik(PublishButtonComponent));
+export const PublishButton = connect(mapStateToProps, (dispatch) => {
+ return {
+ raiseDOINeededButNotReserved: (data, errors) =>
+ dispatch({
+ type: DRAFT_PUBLISH_FAILED_WITH_VALIDATION_ERRORS,
+ payload: { data: data, errors: errors },
+ }),
+ };
+})(connectFormik(PublishButtonComponent));
diff --git a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/fields/Identifiers/PIDField.js b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/fields/Identifiers/PIDField.js
index bd5b00cae..793941d36 100644
--- a/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/fields/Identifiers/PIDField.js
+++ b/invenio_rdm_records/assets/semantic-ui/js/invenio_rdm_records/src/deposit/fields/Identifiers/PIDField.js
@@ -33,7 +33,7 @@ const getFieldErrors = (form, fieldPath) => {
*/
class ReservePIDBtn extends Component {
render() {
- const { disabled, handleReservePID, label, loading } = this.props;
+ const { disabled, handleReservePID, label, loading, fieldError } = this.props;
return (
{({ form: formik }) => (
@@ -44,6 +44,7 @@ class ReservePIDBtn extends Component {
disabled={disabled || loading}
onClick={(e) => handleReservePID(e, formik)}
content={label}
+ error={fieldError}
/>
)}
@@ -54,6 +55,7 @@ class ReservePIDBtn extends Component {
ReservePIDBtn.propTypes = {
disabled: PropTypes.bool,
handleReservePID: PropTypes.func.isRequired,
+ fieldError: PropTypes.object,
label: PropTypes.string.isRequired,
loading: PropTypes.bool,
};
@@ -61,6 +63,7 @@ ReservePIDBtn.propTypes = {
ReservePIDBtn.defaultProps = {
disabled: false,
loading: false,
+ fieldError: null,
};
/**
@@ -110,11 +113,13 @@ class ManagedUnmanagedSwitch extends Component {
handleChange = (e, { value }) => {
const { onManagedUnmanagedChange } = this.props;
const isManagedSelected = value === "managed";
- onManagedUnmanagedChange(isManagedSelected);
+ const isNoNeedSelected = value === "notneeded";
+ onManagedUnmanagedChange(isManagedSelected, isNoNeedSelected);
};
render() {
- const { disabled, isManagedSelected, pidLabel } = this.props;
+ const { disabled, isManagedSelected, isNoNeedSelected, pidLabel, required } =
+ this.props;
return (
@@ -123,28 +128,41 @@ class ManagedUnmanagedSwitch extends Component {
pidLabel: pidLabel,
})}
-
+
-
+
+ {!required && (
+
+
+
+ )}
);
}
@@ -153,8 +171,10 @@ class ManagedUnmanagedSwitch extends Component {
ManagedUnmanagedSwitch.propTypes = {
disabled: PropTypes.bool,
isManagedSelected: PropTypes.bool.isRequired,
+ isNoNeedSelected: PropTypes.bool.isRequired,
onManagedUnmanagedChange: PropTypes.func.isRequired,
pidLabel: PropTypes.string,
+ required: PropTypes.bool.isRequired,
};
ManagedUnmanagedSwitch.defaultProps = {
@@ -198,6 +218,8 @@ class ManagedIdentifierComponent extends Component {
identifier,
pidPlaceholder,
pidType,
+ form,
+ fieldPath,
} = this.props;
const hasIdentifier = identifier !== "";
@@ -209,6 +231,7 @@ class ManagedIdentifierComponent extends Component {
actionState === RESERVE_PID_STARTED && actionStateExtra.pidType === pidType
}
handleReservePID={this.handleReservePID}
+ fieldError={getFieldErrors(form, fieldPath)}
/>
);
@@ -253,6 +276,8 @@ ManagedIdentifierComponent.propTypes = {
btnLabelDiscardPID: PropTypes.string.isRequired,
pidPlaceholder: PropTypes.string.isRequired,
pidType: PropTypes.string.isRequired,
+ form: PropTypes.object.isRequired,
+ fieldPath: PropTypes.string.isRequired,
/* from Redux */
actionState: PropTypes.string,
actionStateExtra: PropTypes.object,
@@ -307,7 +332,7 @@ class UnmanagedIdentifierCmp extends Component {
render() {
const { localIdentifier } = this.state;
- const { form, fieldPath, helpText, pidPlaceholder } = this.props;
+ const { form, fieldPath, helpText, pidPlaceholder, disabled } = this.props;
const fieldError = getFieldErrors(form, fieldPath);
return (
<>
@@ -318,6 +343,7 @@ class UnmanagedIdentifierCmp extends Component {
placeholder={pidPlaceholder}
width={16}
error={fieldError}
+ disabled={disabled}
/>
{helpText && }
@@ -333,10 +359,12 @@ UnmanagedIdentifierCmp.propTypes = {
identifier: PropTypes.string.isRequired,
onIdentifierChanged: PropTypes.func.isRequired,
pidPlaceholder: PropTypes.string.isRequired,
+ disabled: PropTypes.bool,
};
UnmanagedIdentifierCmp.defaultProps = {
helpText: null,
+ disabled: false,
};
/**
@@ -349,11 +377,16 @@ class CustomPIDField extends Component {
constructor(props) {
super(props);
- const { canBeManaged, canBeUnmanaged } = this.props;
+ const { canBeManaged, canBeUnmanaged, record, field } = this.props;
this.canBeManagedAndUnmanaged = canBeManaged && canBeUnmanaged;
-
+ const value = field?.value;
+ const isDraft = record?.is_draft;
this.state = {
- isManagedSelected: undefined,
+ isManagedSelected:
+ isDraft === true && value?.identifier && value?.provider !== PROVIDER_EXTERNAL
+ ? true
+ : undefined,
+ isNoNeedSelected: undefined,
};
}
@@ -373,7 +406,7 @@ class CustomPIDField extends Component {
};
render() {
- const { isManagedSelected } = this.state;
+ const { isManagedSelected, isNoNeedSelected } = this.state;
const {
btnLabelDiscardPID,
btnLabelGetPID,
@@ -394,6 +427,8 @@ class CustomPIDField extends Component {
record,
} = this.props;
+ let { doiDefaultSelection } = this.props;
+
const value = field.value || {};
const currentIdentifier = value.identifier || "";
const currentProvider = value.provider || "";
@@ -407,19 +442,43 @@ class CustomPIDField extends Component {
}
const hasManagedIdentifier = managedIdentifier !== "";
+ const hasUnmanagedIdentifier = unmanagedIdentifier !== "";
+ const doi = record?.pids?.doi?.identifier || "";
+ const parentDoi = record.parent?.pids?.doi?.identifier || "";
+
+ const hasDoi = doi !== "";
+ const hasParentDoi = parentDoi !== "";
+ const isDoiCreated = currentIdentifier !== "";
+
+ // check if record has parent DOI already minted and if yes, then set the doiDefaultSelection
+ // to yes
+ doiDefaultSelection = hasParentDoi ? "yes" : doiDefaultSelection;
+
+ const isDraft = record.is_draft;
+ const _isUnmanagedSelected =
+ isManagedSelected === undefined
+ ? hasUnmanagedIdentifier ||
+ (currentIdentifier === "" && doiDefaultSelection === "yes")
+ : !isManagedSelected;
const _isManagedSelected =
isManagedSelected === undefined
- ? hasManagedIdentifier || currentProvider === "" // i.e pids: {}
+ ? hasManagedIdentifier ||
+ (currentIdentifier === "" && doiDefaultSelection === "no") // i.e pids: {}
: isManagedSelected;
- const doi = record?.pids?.doi?.identifier || "";
- const hasDoi = doi !== "";
- const isDoiCreated = currentIdentifier !== "";
+ const _isNoNeedSelected =
+ isNoNeedSelected === undefined
+ ? (!_isManagedSelected && !_isUnmanagedSelected) ||
+ (isDraft !== true &&
+ currentIdentifier === "" &&
+ doiDefaultSelection === "not_needed")
+ : isNoNeedSelected;
+
const fieldError = getFieldErrors(form, fieldPath);
return (
<>
-
+
@@ -427,20 +486,28 @@ class CustomPIDField extends Component {
{
+ isNoNeedSelected={_isNoNeedSelected}
+ onManagedUnmanagedChange={(userSelectedManaged, userSelectedNoNeed) => {
if (userSelectedManaged) {
form.setFieldValue("pids", {});
+ form.setFieldValue("noINeedOne", true);
+ } else if (userSelectedNoNeed) {
+ form.setFieldValue("pids", {});
+ form.setFieldValue("noINeedOne", false);
} else {
this.onExternalIdentifierChanged("");
+ form.setFieldValue("noINeedOne", false);
}
this.setState({
isManagedSelected: userSelectedManaged,
+ isNoNeedSelected: userSelectedNoNeed,
});
}}
pidLabel={pidLabel}
+ required={required}
/>
)}
@@ -450,6 +517,7 @@ class CustomPIDField extends Component {
btnLabelDiscardPID={btnLabelDiscardPID}
btnLabelGetPID={btnLabelGetPID}
form={form}
+ fieldPath={fieldPath}
identifier={managedIdentifier}
helpText={managedHelpText}
pidPlaceholder={pidPlaceholder}
@@ -458,7 +526,7 @@ class CustomPIDField extends Component {
/>
)}
- {canBeUnmanaged && !_isManagedSelected && (
+ {canBeUnmanaged && (!_isManagedSelected || _isNoNeedSelected) && (
{
@@ -468,6 +536,7 @@ class CustomPIDField extends Component {
fieldPath={fieldPath}
pidPlaceholder={pidPlaceholder}
helpText={unmanagedHelpText}
+ disabled={_isNoNeedSelected || isEditingPublishedRecord}
/>
)}
>
@@ -493,6 +562,7 @@ CustomPIDField.propTypes = {
required: PropTypes.bool.isRequired,
unmanagedHelpText: PropTypes.string,
record: PropTypes.object.isRequired,
+ doiDefaultSelection: PropTypes.object.isRequired,
};
CustomPIDField.defaultProps = {
@@ -542,6 +612,7 @@ PIDField.propTypes = {
required: PropTypes.bool,
unmanagedHelpText: PropTypes.string,
record: PropTypes.object.isRequired,
+ doiDefaultSelection: PropTypes.object.isRequired,
};
PIDField.defaultProps = {
diff --git a/invenio_rdm_records/config.py b/invenio_rdm_records/config.py
index 9ea88dddd..fdb96df32 100644
--- a/invenio_rdm_records/config.py
+++ b/invenio_rdm_records/config.py
@@ -391,6 +391,7 @@ def always_valid(identifier):
"validator": idutils.is_doi,
"normalizer": idutils.normalize_doi,
"is_enabled": providers.DataCitePIDProvider.is_enabled,
+ "ui": {"default_selected": "yes"}, # "yes", "no" or "not_needed"
},
"oai": {
"providers": ["oai"],
diff --git a/invenio_rdm_records/records/api.py b/invenio_rdm_records/records/api.py
index 377ef6c82..37a59874e 100644
--- a/invenio_rdm_records/records/api.py
+++ b/invenio_rdm_records/records/api.py
@@ -540,6 +540,22 @@ def get_latest_published_by_parent(cls, parent):
return None
return latest_record
+ @classmethod
+ def get_previous_published_by_parent(cls, parent):
+ """Get the latest published record for the specified parent record.
+
+ It might return None if there is no latest published version i.e not
+ published yet or all versions are deleted.
+ """
+ with db.session.no_autoflush:
+ records = cls.get_records_by_parent(parent)
+ for record in records:
+ latest_version_index = record.versions.latest_index
+ if latest_version_index > 1:
+ if record.versions.index == latest_version_index - 1:
+ return record
+ return None
+
RDMFileRecord.record_cls = RDMRecord
diff --git a/invenio_rdm_records/resources/serializers/datacite/schema.py b/invenio_rdm_records/resources/serializers/datacite/schema.py
index 82d685e41..6e8c00544 100644
--- a/invenio_rdm_records/resources/serializers/datacite/schema.py
+++ b/invenio_rdm_records/resources/serializers/datacite/schema.py
@@ -418,7 +418,7 @@ def get_related_identifiers(self, obj):
params={"_source_includes": "pids.doi"},
)
for version in record_versions:
- version_doi = version["pids"]["doi"]
+ version_doi = version.get("pids", {}).get("doi")
id_scheme = get_scheme_datacite(
"doi",
"RDM_RECORDS_IDENTIFIERS_SCHEMES",
diff --git a/invenio_rdm_records/services/components/pids.py b/invenio_rdm_records/services/components/pids.py
index 8a08a67b2..0ad253e3c 100644
--- a/invenio_rdm_records/services/components/pids.py
+++ b/invenio_rdm_records/services/components/pids.py
@@ -15,8 +15,10 @@
from flask import current_app
from invenio_drafts_resources.services.records.components import ServiceComponent
from invenio_drafts_resources.services.records.uow import ParentRecordCommitOp
+from invenio_i18n import lazy_gettext as _
from invenio_records_resources.services.uow import TaskOp
+from ..errors import ValidationErrorWithMessageAsList
from ..pids.tasks import register_or_update_pid
@@ -63,6 +65,51 @@ def delete_draft(self, identity, draft=None, record=None, force=False):
def publish(self, identity, draft=None, record=None):
"""Publish handler."""
+
+ def _validate_doi_transition(new_provider, previous_published_provider):
+ """."""
+ ALLOWED_DOI_TRANSITIONS = {
+ "datacite": {
+ "allowed_schemes": ["datacite"],
+ "message": "A previous version used a DOI from this repository registered in DataCite. This version must also use a DOI from this repository.",
+ },
+ "external": {
+ "allowed_schemes": ["external", "not_needed"],
+ "message": "A previous version was published with a DOI from an external provider or without one. You cannot use a DOI from this repository for this version.",
+ },
+ "not_needed": {
+ "allowed_schemes": ["external", "not_needed"],
+ "message": "A previous version was published with a DOI from an external provider or without one. You cannot use a DOI from this repository for this version.",
+ },
+ }
+
+ valid_transitions = ALLOWED_DOI_TRANSITIONS.get(
+ previous_published_provider, {}
+ )
+ if new_provider not in valid_transitions.get("allowed_providers", []):
+ raise ValidationErrorWithMessageAsList(
+ message=[
+ {
+ "field": "pids.doi",
+ "messages": [valid_transitions.get("message")],
+ }
+ ]
+ )
+
+ def _validate_optional_doi():
+ # Validate the draft PIDs
+ previous_published = record.get_previous_published_by_parent(record.parent)
+ # Check only for previously published records
+ if previous_published:
+ # check the that pid
+ previous_published_pids = previous_published.get("pids", {})
+ doi_pid = [pid for pid in draft_pids.values() if "doi" in draft_pids]
+ previous_published_provider = previous_published_pids.get(
+ "doi", {}
+ ).get("provider", "not_needed")
+ new_provider = "not_needed" if not doi_pid else doi_pid[0]["provider"]
+ _validate_doi_transition(new_provider, previous_published_provider)
+
# ATTENTION: A draft can be for both an unpublished or published
# record. For an unpublished record, we usually simply need to create
# and reserve all PIDs. For a published record, some PIDs may allow
@@ -75,7 +122,16 @@ def publish(self, identity, draft=None, record=None):
record_schemes = set(record_pids.keys())
required_schemes = set(self.service.config.pids_required)
- # Validate the draft PIDs
+ # if DOI is not required in an instance check validate allowed providers
+ # for each record version
+ if "doi" not in required_schemes:
+ # if a doi was ever minted for the parent record then we always require one
+ # for any version of the record that will be published
+ if draft.parent.get("pids", {}).get("doi"):
+ required_schemes.add("doi")
+
+ _validate_optional_doi()
+
self.service.pids.pid_manager.validate(draft_pids, draft, raise_errors=True)
# Detect which PIDs on a published record that has been changed.
@@ -131,10 +187,7 @@ def new_version(self, identity, draft=None, record=None):
"""A new draft should not have any pids from the previous record."""
# This makes the draft use the same identifier as the previous
# version
- if record.pids.get("doi", {}).get("provider") == "external":
- draft.pids = {"doi": {"provider": "external", "identifier": ""}}
- else:
- draft.pids = {}
+ draft.pids = {}
def edit(self, identity, draft=None, record=None):
"""Add current pids from the record to the draft.
@@ -172,6 +225,14 @@ def publish(self, identity, draft=None, record=None):
current_schemes = set(current_pids.keys())
required_schemes = set(self.service.config.parent_pids_required)
+ # Check if a doi was added in the draft and create a parent DOI independently if
+ # doi is required.
+ # Note: we don't have to check explicitely to the parent DOI creation only for
+ # datacite provider because we pass a `condition_func` below that it omits the
+ # minting if the pid selected is external
+ if draft.get("pids", {}).get("doi"):
+ required_schemes.add("doi")
+
conditional_schemes = self.service.config.parent_pids_conditional
for scheme in set(required_schemes):
condition_func = conditional_schemes.get(scheme)
diff --git a/tests/services/test_rdm_service.py b/tests/services/test_rdm_service.py
index ae1158d99..2834fee8c 100644
--- a/tests/services/test_rdm_service.py
+++ b/tests/services/test_rdm_service.py
@@ -10,10 +10,15 @@
"""Service level tests for Invenio RDM Records."""
+from copy import deepcopy
+
import pytest
from invenio_rdm_records.proxies import current_rdm_records
-from invenio_rdm_records.services.errors import EmbargoNotLiftedError
+from invenio_rdm_records.services.errors import (
+ EmbargoNotLiftedError,
+ ValidationErrorWithMessageAsList,
+)
def test_minimal_draft_creation(running_app, search_clear, minimal_record):
@@ -50,7 +55,7 @@ def test_draft_w_languages_creation(running_app, search_clear, minimal_record):
def test_publish_public_record_with_default_doi(
- running_app, search_clear, minimal_record
+ running_app, search_clear, minimal_record, uploader
):
superuser_identity = running_app.superuser_identity
service = current_rdm_records.records_service
@@ -68,10 +73,115 @@ def test_publish_public_record_with_optional_doi(
draft = service.create(superuser_identity, minimal_record)
record = service.publish(id_=draft.id, identity=superuser_identity)
assert "doi" not in record._record.pids
+ assert "doi" not in record._record.parent.pids
+ # Reset the running_app config for next tests
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = True
+
+
+def test_publish_public_record_versions_no_or_external_doi_managed_doi(
+ running_app, search_clear, minimal_record, verified_user
+):
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = False
+ verified_user_identity = verified_user.identity
+ service = current_rdm_records.records_service
+ # Publish without DOI
+ draft = service.create(verified_user_identity, minimal_record)
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+ assert "doi" not in record._record.pids
+ assert "doi" not in record._record.parent.pids
+
+ # create a new version with an external DOI
+ draft = service.new_version(verified_user_identity, record.id)
+ draft_data = deepcopy(draft.data)
+ draft_data["metadata"]["publication_date"] = "2023-01-01"
+ draft_data["pids"]["doi"] = {
+ "identifier": "10.4321/test.1234",
+ "provider": "external",
+ }
+ draft = service.update_draft(verified_user_identity, draft.id, data=draft_data)
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+ assert "doi" in record._record.pids
+ assert "doi" in record._record.parent.pids
+
+ # create a new version and and try to mint a managed DOI now when you publish
+ draft = service.new_version(verified_user_identity, record.id)
+ draft = service.pids.create(verified_user_identity, draft.id, "doi")
+ draft_data = deepcopy(draft.data)
+ draft_data["metadata"]["publication_date"] = "2023-01-01"
+ draft = service.update_draft(verified_user_identity, draft.id, data=draft_data)
+
+ with pytest.raises(ValidationErrorWithMessageAsList):
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+
+ # Reset the running_app config for next tests
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = True
+
+
+def test_publish_public_record_versions_managed_doi_external_doi(
+ running_app, search_clear, minimal_record, verified_user
+):
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = False
+ verified_user_identity = verified_user.identity
+ service = current_rdm_records.records_service
+ # Publish with locally managed DOI
+ draft = service.create(verified_user_identity, minimal_record)
+ draft = service.pids.create(verified_user_identity, draft.id, "doi")
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+ assert "doi" in record._record.pids
+ assert "doi" in record._record.parent.pids
+
+ # create a new version with an external DOI
+ draft = service.new_version(verified_user_identity, record.id)
+ draft_data = deepcopy(draft.data)
+ draft_data["metadata"]["publication_date"] = "2023-01-01"
+ draft_data["pids"]["doi"] = {
+ "identifier": "10.4321/test.1234",
+ "provider": "external",
+ }
+ draft = service.update_draft(verified_user_identity, draft.id, data=draft_data)
+ with pytest.raises(ValidationErrorWithMessageAsList):
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+
# Reset the running_app config for next tests
running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = True
+def test_publish_public_record_versions_managed_doi_no_doi(
+ running_app, search_clear, minimal_record, verified_user
+):
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = False
+ verified_user_identity = verified_user.identity
+ service = current_rdm_records.records_service
+ # Publish with locally managed DOI
+ draft = service.create(verified_user_identity, minimal_record)
+ draft = service.pids.create(verified_user_identity, draft.id, "doi")
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+ assert "doi" in record._record.pids
+ assert "doi" in record._record.parent.pids
+
+ # create a new version with no DOI
+ draft = service.new_version(verified_user_identity, record.id)
+ draft_data = deepcopy(draft.data)
+ draft_data["metadata"]["publication_date"] = "2023-01-01"
+ draft_data["pids"] = {}
+ draft = service.update_draft(verified_user_identity, draft.id, data=draft_data)
+ with pytest.raises(ValidationErrorWithMessageAsList):
+ record = service.publish(id_=draft.id, identity=verified_user_identity)
+
+ # Reset the running_app config for next tests
+ running_app.app.config["RDM_PERSISTENT_IDENTIFIERS"]["doi"]["required"] = True
+
+
+def test_publish_public_record_with_default_doi(
+ running_app, search_clear, minimal_record, uploader
+):
+ superuser_identity = running_app.superuser_identity
+ service = current_rdm_records.records_service
+ draft = service.create(superuser_identity, minimal_record)
+ record = service.publish(id_=draft.id, identity=superuser_identity)
+ assert "doi" in record._record.pids
+
+
def test_publish_restricted_record_without_default_doi(
running_app, search_clear, minimal_restricted_record
):