Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SQL: Deletion of Assets impossible after first Contract Negotation #1403

Closed
DominikPinsel opened this issue Jun 2, 2022 · 3 comments · Fixed by #1408
Closed

SQL: Deletion of Assets impossible after first Contract Negotation #1403

DominikPinsel opened this issue Jun 2, 2022 · 3 comments · Fixed by #1408
Assignees
Labels
bug Something isn't working

Comments

@DominikPinsel
Copy link
Contributor

Bug Report

Describe the Bug

Affects all connectors that are using the PostgreSQL Asset Extension.

After the connector has done one or more contract negotiations, it is impossible to delete assets.

Expected Behavior

  • deletion of assets possible as long as its legal (not part of any agreement etc.)

Observed Behavior

  • deletion of all assets impossible after contract negotiation

Steps to Reproduce

Steps to reproduce the behavior:

  1. Use Connector with PostgreSQL Asset Extension
  2. Create Asset, etc. and do a contract negotiation with it
  3. Create a new Asset
  4. Delete the new Asset -> It's not possible to delete this new asset

Context Information

Before deleting the Asset the AssetService will check, whether a negotation/agreement exists for this asset.

 @Override
    public ServiceResult<Asset> delete(String assetId) {
        return transactionContext.execute(() -> {
            var filter = format("contractAgreement.assetId = %s", assetId);
            var query = QuerySpec.Builder.newInstance().filter(filter).build();

            // the store now returns always a negotation 
            var negotiationsOnAsset = contractNegotiationStore.queryNegotiations(query);
            if (negotiationsOnAsset.findAny().isPresent()) {
                return ServiceResult.conflict(format("Asset %s cannot be deleted as it is referenced by at least one contract agreement", assetId));
            }

            var deleted = loader.deleteById(assetId);
            if (deleted == null) {
                return ServiceResult.notFound(format("Asset %s does not exist", assetId));
            }

            return ServiceResult.success(deleted);
        });
    }

Because of an unfinished SQL Query the store will always return a negotiation if there exists at least one.

    @Override
    public String getQueryNegotiationsTemplate() {
        // todo: add WHERE ... AND ... ORDER BY... statements here
        return format("SELECT * FROM %s LEFT OUTER JOIN %s ON %s.%s = %s.%s LIMIT ? OFFSET ?;", getContractNegotiationTable(), getContractAgreementTable(),
                getContractNegotiationTable(), getContractAgreementIdFkColumn(), getContractAgreementTable(), getContractAgreementIdColumn());
    }

Detailed Description

Possible Implementation

  • resolve TODO

Dominik Pinsel dominik.pinsel@daimler.com, Mercedes-Benz Tech Innovation GmbH, legal info/Impressum

@DominikPinsel DominikPinsel added the bug Something isn't working label Jun 2, 2022
@paullatzelsperger
Copy link
Member

paullatzelsperger commented Jun 2, 2022

I'm not sure the todo has much to do with it. I think the easiest fix is to create a separate method (getNegotiationsWithAgreementOnAsset() or similar) in the ContractNegotiationStore interface and a dedicated SQL query for this. If that method can simply return a boolean ("hasAgreements") then this statement could look similar to this:

SELECT *
FROM edc_contract_negotiation
WHERE edc_contract_negotiation.contract_agreement_id in (SELECT agreement_id
                                                         FROM edc_contract_agreement
                                                         WHERE asset_id = 'asset1');

if we want that new method to return the ContractNegotiation, we probably have to enhance the statement with a JOIN to also get the ContractAgreement

SELECT *
FROM edc_contract_negotiation
INNER JOIN edc_contract_agreement eca on edc_contract_negotiation.contract_agreement_id = eca.agreement_id
WHERE edc_contract_negotiation.contract_agreement_id in (SELECT agreement_id
                                                         FROM edc_contract_agreement
                                                         WHERE asset_id = 'asset1')

Of course, equivalent implementations for the other stores would also have to be done too.
@ndr-brt what are your thoughts on this?

@ndr-brt
Copy link
Member

ndr-brt commented Jun 7, 2022

@paullatzelsperger I'm late at the party, but I think as easy-fix having a dedicated method would work, however in the long run would be better to have the postgresql implementations being able to handle the QuerySpec in its entirety to permit having a single implementation for this method across all the database implementations.

@paullatzelsperger
Copy link
Member

@paullatzelsperger I'm late at the party, but I think as easy-fix having a dedicated method would work, however in the long run would be better to have the postgresql implementations being able to handle the QuerySpec in its entirety to permit having a single implementation for this method across all the database implementations.

I agree, having an implemention that can interpret QuerySpec filter expressions would be best, but it needs careful thinking with regards to security (SQL injection), and flexibility - after all, the database schema is (theoretically) flexible. Also, this would leak the database schema through the API, which is something to be aware of.

On the other hand I am a big fan of having explicit methods for a particular use case, but they should rather be on the service than on the store.

I'll open a discussion about this.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants