提交 27553573 编写于 作者: E Eric Blake

maint: clean up error reporting in migration

The choice of error message and category was not consistent
in the migration code; furthermore, the use of virLibConnError
is no longer necessary now that we have a generic virReportError.

* src/qemu/qemu_migration.c (virDomainMigrate*): Prefer
virReportError over virLibConnError.
Signed-off-by: NEric Blake <eblake@redhat.com>
上级 c8ed177a
...@@ -4468,8 +4468,8 @@ virDomainMigrateVersion1(virDomainPtr domain, ...@@ -4468,8 +4468,8 @@ virDomainMigrateVersion1(virDomainPtr domain,
goto done; goto done;
if (uri == NULL && uri_out == NULL) { if (uri == NULL && uri_out == NULL) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("domainMigratePrepare did not set uri")); _("domainMigratePrepare did not set uri"));
goto done; goto done;
} }
if (uri_out) if (uri_out)
...@@ -4560,7 +4560,7 @@ virDomainMigrateVersion2(virDomainPtr domain, ...@@ -4560,7 +4560,7 @@ virDomainMigrateVersion2(virDomainPtr domain,
* and pass it to Prepare2. * and pass it to Prepare2.
*/ */
if (!domain->conn->driver->domainGetXMLDesc) { if (!domain->conn->driver->domainGetXMLDesc) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, __FUNCTION__); virReportUnsupportedError();
return NULL; return NULL;
} }
...@@ -4590,8 +4590,8 @@ virDomainMigrateVersion2(virDomainPtr domain, ...@@ -4590,8 +4590,8 @@ virDomainMigrateVersion2(virDomainPtr domain,
goto done; goto done;
if (uri == NULL && uri_out == NULL) { if (uri == NULL && uri_out == NULL) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("domainMigratePrepare2 did not set uri")); _("domainMigratePrepare2 did not set uri"));
cancelled = 1; cancelled = 1;
/* Make sure Finish doesn't overwrite the error */ /* Make sure Finish doesn't overwrite the error */
orig_err = virSaveLastError(); orig_err = virSaveLastError();
...@@ -4715,7 +4715,7 @@ virDomainMigrateVersion3Full(virDomainPtr domain, ...@@ -4715,7 +4715,7 @@ virDomainMigrateVersion3Full(virDomainPtr domain,
!domain->conn->driver->domainMigrateConfirm3Params || !domain->conn->driver->domainMigrateConfirm3Params ||
!dconn->driver->domainMigratePrepare3Params || !dconn->driver->domainMigratePrepare3Params ||
!dconn->driver->domainMigrateFinish3Params))) { !dconn->driver->domainMigrateFinish3Params))) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, __FUNCTION__); virReportUnsupportedError();
return NULL; return NULL;
} }
...@@ -4797,8 +4797,8 @@ virDomainMigrateVersion3Full(virDomainPtr domain, ...@@ -4797,8 +4797,8 @@ virDomainMigrateVersion3Full(virDomainPtr domain,
} else if (!uri && } else if (!uri &&
virTypedParamsGetString(params, nparams, virTypedParamsGetString(params, nparams,
VIR_MIGRATE_PARAM_URI, &uri) <= 0) { VIR_MIGRATE_PARAM_URI, &uri) <= 0) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("domainMigratePrepare3 did not set uri")); _("domainMigratePrepare3 did not set uri"));
cancelled = 1; cancelled = 1;
orig_err = virSaveLastError(); orig_err = virSaveLastError();
goto finish; goto finish;
...@@ -5010,7 +5010,7 @@ virDomainMigratePeer2PeerFull(virDomainPtr domain, ...@@ -5010,7 +5010,7 @@ virDomainMigratePeer2PeerFull(virDomainPtr domain,
(!useParams && (!useParams &&
!domain->conn->driver->domainMigratePerform && !domain->conn->driver->domainMigratePerform &&
!domain->conn->driver->domainMigratePerform3)) { !domain->conn->driver->domainMigratePerform3)) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, __FUNCTION__); virReportUnsupportedError();
return -1; return -1;
} }
...@@ -5039,14 +5039,14 @@ virDomainMigratePeer2PeerFull(virDomainPtr domain, ...@@ -5039,14 +5039,14 @@ virDomainMigratePeer2PeerFull(virDomainPtr domain,
} else { } else {
VIR_DEBUG("Using migration protocol 2"); VIR_DEBUG("Using migration protocol 2");
if (xmlin) { if (xmlin) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML " _("Unable to change target guest XML during "
"during migration")); "migration"));
return -1; return -1;
} }
if (uri) { if (uri) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("Unable to override peer2peer migration URI")); _("Unable to override peer2peer migration URI"));
return -1; return -1;
} }
return domain->conn->driver->domainMigratePerform return domain->conn->driver->domainMigratePerform
...@@ -5132,8 +5132,8 @@ virDomainMigrateDirect(virDomainPtr domain, ...@@ -5132,8 +5132,8 @@ virDomainMigrateDirect(virDomainPtr domain,
} else { } else {
VIR_DEBUG("Using migration protocol 2"); VIR_DEBUG("Using migration protocol 2");
if (xmlin) { if (xmlin) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML during migration")); _("Unable to change target guest XML during migration"));
return -1; return -1;
} }
return domain->conn->driver->domainMigratePerform(domain, return domain->conn->driver->domainMigratePerform(domain,
...@@ -5261,16 +5261,16 @@ virDomainMigrate(virDomainPtr domain, ...@@ -5261,16 +5261,16 @@ virDomainMigrate(virDomainPtr domain,
if (flags & VIR_MIGRATE_OFFLINE) { if (flags & VIR_MIGRATE_OFFLINE) {
if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the source host")); "the source host"));
goto error; goto error;
} }
if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn, if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the destination host")); "the destination host"));
goto error; goto error;
} }
} }
...@@ -5308,14 +5308,14 @@ virDomainMigrate(virDomainPtr domain, ...@@ -5308,14 +5308,14 @@ virDomainMigrate(virDomainPtr domain,
if (flags & VIR_MIGRATE_CHANGE_PROTECTION && if (flags & VIR_MIGRATE_CHANGE_PROTECTION &&
!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, !VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) { VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("cannot enforce change protection")); _("cannot enforce change protection"));
goto error; goto error;
} }
flags &= ~VIR_MIGRATE_CHANGE_PROTECTION; flags &= ~VIR_MIGRATE_CHANGE_PROTECTION;
if (flags & VIR_MIGRATE_TUNNELLED) { if (flags & VIR_MIGRATE_TUNNELLED) {
virLibConnError(VIR_ERR_OPERATION_INVALID, "%s", virReportError(VIR_ERR_OPERATION_INVALID, "%s",
_("cannot perform tunnelled migration without using peer2peer flag")); _("cannot perform tunnelled migration without using peer2peer flag"));
goto error; goto error;
} }
...@@ -5487,16 +5487,16 @@ virDomainMigrate2(virDomainPtr domain, ...@@ -5487,16 +5487,16 @@ virDomainMigrate2(virDomainPtr domain,
if (flags & VIR_MIGRATE_OFFLINE) { if (flags & VIR_MIGRATE_OFFLINE) {
if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the source host")); "the source host"));
goto error; goto error;
} }
if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn, if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the destination host")); "the destination host"));
goto error; goto error;
} }
} }
...@@ -5531,14 +5531,14 @@ virDomainMigrate2(virDomainPtr domain, ...@@ -5531,14 +5531,14 @@ virDomainMigrate2(virDomainPtr domain,
if (flags & VIR_MIGRATE_CHANGE_PROTECTION && if (flags & VIR_MIGRATE_CHANGE_PROTECTION &&
!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, !VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) { VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("cannot enforce change protection")); _("cannot enforce change protection"));
goto error; goto error;
} }
flags &= ~VIR_MIGRATE_CHANGE_PROTECTION; flags &= ~VIR_MIGRATE_CHANGE_PROTECTION;
if (flags & VIR_MIGRATE_TUNNELLED) { if (flags & VIR_MIGRATE_TUNNELLED) {
virLibConnError(VIR_ERR_OPERATION_INVALID, "%s", virReportError(VIR_ERR_OPERATION_INVALID, "%s",
_("cannot perform tunnelled migration without using peer2peer flag")); _("cannot perform tunnelled migration without using peer2peer flag"));
goto error; goto error;
} }
...@@ -5556,8 +5556,8 @@ virDomainMigrate2(virDomainPtr domain, ...@@ -5556,8 +5556,8 @@ virDomainMigrate2(virDomainPtr domain,
VIR_DRV_FEATURE_MIGRATION_V2)) { VIR_DRV_FEATURE_MIGRATION_V2)) {
VIR_DEBUG("Using migration protocol 2"); VIR_DEBUG("Using migration protocol 2");
if (dxml) { if (dxml) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML during migration")); _("Unable to change target guest XML during migration"));
goto error; goto error;
} }
ddomain = virDomainMigrateVersion2(domain, dconn, flags, ddomain = virDomainMigrateVersion2(domain, dconn, flags,
...@@ -5568,8 +5568,8 @@ virDomainMigrate2(virDomainPtr domain, ...@@ -5568,8 +5568,8 @@ virDomainMigrate2(virDomainPtr domain,
VIR_DRV_FEATURE_MIGRATION_V1)) { VIR_DRV_FEATURE_MIGRATION_V1)) {
VIR_DEBUG("Using migration protocol 1"); VIR_DEBUG("Using migration protocol 1");
if (dxml) { if (dxml) {
virLibConnError(VIR_ERR_INTERNAL_ERROR, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML during migration")); _("Unable to change target guest XML during migration"));
goto error; goto error;
} }
ddomain = virDomainMigrateVersion1(domain, dconn, flags, ddomain = virDomainMigrateVersion1(domain, dconn, flags,
...@@ -5670,16 +5670,16 @@ virDomainMigrate3(virDomainPtr domain, ...@@ -5670,16 +5670,16 @@ virDomainMigrate3(virDomainPtr domain,
if (flags & VIR_MIGRATE_OFFLINE) { if (flags & VIR_MIGRATE_OFFLINE) {
if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the source host")); "the source host"));
goto error; goto error;
} }
if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn, if (!VIR_DRV_SUPPORTS_FEATURE(dconn->driver, dconn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the destination host")); "the destination host"));
goto error; goto error;
} }
} }
...@@ -5692,8 +5692,8 @@ virDomainMigrate3(virDomainPtr domain, ...@@ -5692,8 +5692,8 @@ virDomainMigrate3(virDomainPtr domain,
if (flags & VIR_MIGRATE_CHANGE_PROTECTION && if (flags & VIR_MIGRATE_CHANGE_PROTECTION &&
!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, !VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) { VIR_DRV_FEATURE_MIGRATE_CHANGE_PROTECTION)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("cannot enforce change protection")); _("cannot enforce change protection"));
goto error; goto error;
} }
flags &= ~VIR_MIGRATE_CHANGE_PROTECTION; flags &= ~VIR_MIGRATE_CHANGE_PROTECTION;
...@@ -5712,9 +5712,9 @@ virDomainMigrate3(virDomainPtr domain, ...@@ -5712,9 +5712,9 @@ virDomainMigrate3(virDomainPtr domain,
if (!virTypedParamsCheck(params, nparams, compatParams, if (!virTypedParamsCheck(params, nparams, compatParams,
ARRAY_CARDINALITY(compatParams))) { ARRAY_CARDINALITY(compatParams))) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Migration APIs with extensible parameters are not " _("Migration APIs with extensible parameters are not "
"supported but extended parameters were passed")); "supported but extended parameters were passed"));
goto error; goto error;
} }
...@@ -5742,9 +5742,9 @@ virDomainMigrate3(virDomainPtr domain, ...@@ -5742,9 +5742,9 @@ virDomainMigrate3(virDomainPtr domain,
VIR_DRV_FEATURE_MIGRATION_V2)) { VIR_DRV_FEATURE_MIGRATION_V2)) {
VIR_DEBUG("Using migration protocol 2"); VIR_DEBUG("Using migration protocol 2");
if (dxml) { if (dxml) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML during " _("Unable to change target guest XML during "
"migration")); "migration"));
goto error; goto error;
} }
ddomain = virDomainMigrateVersion2(domain, dconn, flags, ddomain = virDomainMigrateVersion2(domain, dconn, flags,
...@@ -5755,9 +5755,9 @@ virDomainMigrate3(virDomainPtr domain, ...@@ -5755,9 +5755,9 @@ virDomainMigrate3(virDomainPtr domain,
VIR_DRV_FEATURE_MIGRATION_V1)) { VIR_DRV_FEATURE_MIGRATION_V1)) {
VIR_DEBUG("Using migration protocol 1"); VIR_DEBUG("Using migration protocol 1");
if (dxml) { if (dxml) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Unable to change target guest XML during " _("Unable to change target guest XML during "
"migration")); "migration"));
goto error; goto error;
} }
ddomain = virDomainMigrateVersion1(domain, dconn, flags, ddomain = virDomainMigrateVersion1(domain, dconn, flags,
...@@ -5881,9 +5881,9 @@ virDomainMigrateToURI(virDomainPtr domain, ...@@ -5881,9 +5881,9 @@ virDomainMigrateToURI(virDomainPtr domain,
if (flags & VIR_MIGRATE_OFFLINE && if (flags & VIR_MIGRATE_OFFLINE &&
!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, !VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_OFFLINE)) { VIR_DRV_FEATURE_MIGRATION_OFFLINE)) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("offline migration is not supported by " _("offline migration is not supported by "
"the source host")); "the source host"));
goto error; goto error;
} }
...@@ -5908,9 +5908,9 @@ virDomainMigrateToURI(virDomainPtr domain, ...@@ -5908,9 +5908,9 @@ virDomainMigrateToURI(virDomainPtr domain,
goto error; goto error;
} else { } else {
/* Cannot do a migration with only the perform step */ /* Cannot do a migration with only the perform step */
virLibConnError(VIR_ERR_OPERATION_INVALID, "%s", virReportError(VIR_ERR_OPERATION_INVALID, "%s",
_("direct migration is not supported by the" _("direct migration is not supported by the"
" connection driver")); " connection driver"));
goto error; goto error;
} }
} }
...@@ -6056,9 +6056,9 @@ virDomainMigrateToURI2(virDomainPtr domain, ...@@ -6056,9 +6056,9 @@ virDomainMigrateToURI2(virDomainPtr domain,
goto error; goto error;
} else { } else {
/* Cannot do a migration with only the perform step */ /* Cannot do a migration with only the perform step */
virLibConnError(VIR_ERR_OPERATION_INVALID, "%s", virReportError(VIR_ERR_OPERATION_INVALID, "%s",
_("direct migration is not supported by the" _("direct migration is not supported by the"
" connection driver")); " connection driver"));
goto error; goto error;
} }
} }
...@@ -6161,9 +6161,9 @@ virDomainMigrateToURI3(virDomainPtr domain, ...@@ -6161,9 +6161,9 @@ virDomainMigrateToURI3(virDomainPtr domain,
if (flags & VIR_MIGRATE_PEER2PEER) { if (flags & VIR_MIGRATE_PEER2PEER) {
if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_P2P)) { VIR_DRV_FEATURE_MIGRATION_P2P)) {
virLibConnError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
_("Peer-to-peer migration is not supported by " _("Peer-to-peer migration is not supported by "
"the connection driver")); "the connection driver"));
goto error; goto error;
} }
...@@ -6179,26 +6179,26 @@ virDomainMigrateToURI3(virDomainPtr domain, ...@@ -6179,26 +6179,26 @@ virDomainMigrateToURI3(virDomainPtr domain,
dconnuri, uri, bandwidth) < 0) dconnuri, uri, bandwidth) < 0)
goto error; goto error;
} else { } else {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Peer-to-peer migration with extensible " _("Peer-to-peer migration with extensible "
"parameters is not supported but extended " "parameters is not supported but extended "
"parameters were passed")); "parameters were passed"));
goto error; goto error;
} }
} else { } else {
if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn, if (!VIR_DRV_SUPPORTS_FEATURE(domain->conn->driver, domain->conn,
VIR_DRV_FEATURE_MIGRATION_DIRECT)) { VIR_DRV_FEATURE_MIGRATION_DIRECT)) {
/* Cannot do a migration with only the perform step */ /* Cannot do a migration with only the perform step */
virLibConnError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
_("Direct migration is not supported by the" _("Direct migration is not supported by the"
" connection driver")); " connection driver"));
goto error; goto error;
} }
if (!compat) { if (!compat) {
virLibConnError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s", virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, "%s",
_("Direct migration does not support extensible " _("Direct migration does not support extensible "
"parameters")); "parameters"));
goto error; goto error;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册