org.jcryptool.crypto.xml.ui.sign.PageResource.java Source code

Java tutorial

Introduction

Here is the source code for org.jcryptool.crypto.xml.ui.sign.PageResource.java

Source

/*******************************************************************************
 * Copyright (c) 2013 Dominik Schadow - http://www.xml-sicherheit.de All rights reserved. This program and the
 * accompanying materials are made available under the terms of the Eclipse Public License v1.0 which accompanies this
 * distribution, and is available at http://www.eclipse.org/legal/epl-v10.html
 * 
 * Contributors: Dominik Schadow - initial API and implementation
 *******************************************************************************/
package org.jcryptool.crypto.xml.ui.sign;

import java.io.File;
import java.io.InputStream;

import org.eclipse.jface.dialogs.IDialogSettings;
import org.eclipse.jface.dialogs.IMessageProvider;
import org.eclipse.jface.viewers.LabelProvider;
import org.eclipse.jface.wizard.WizardPage;
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.ModifyEvent;
import org.eclipse.swt.events.ModifyListener;
import org.eclipse.swt.layout.FormAttachment;
import org.eclipse.swt.layout.FormData;
import org.eclipse.swt.layout.FormLayout;
import org.eclipse.swt.widgets.Button;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.FileDialog;
import org.eclipse.swt.widgets.Group;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Text;
import org.eclipse.ui.PlatformUI;
import org.jcryptool.core.util.constants.IConstants;
import org.jcryptool.core.util.directories.DirectoryService;
import org.jcryptool.crypto.xml.core.sign.Signature;
import org.jcryptool.crypto.xml.core.utils.IGlobals;
import org.jcryptool.crypto.xml.core.utils.Utils;
import org.jcryptool.crypto.xml.ui.dialogs.XpathDialog;
import org.jcryptool.crypto.xml.ui.utils.IContextHelpIds;
import org.w3c.dom.Document;

/**
 * <p>
 * First wizard page to create an XML signature. Lets the user select the resource to sign (<i>document</i>,
 * <i>selection</i>, <i>XPath</i>) and the signature type (<i>enveloping</i>, <i>enveloped</i>, <i>detached</i>).<br>
 * The option to create a <i>Basic Security Profile</i> compliant signature can be activated on this page too.
 * </p>
 * 
 * @author Dominik Schadow
 * @version 1.0.0
 */
public class PageResource extends WizardPage implements Listener {
    /** Wizard page name. */
    public static final String PAGE_NAME = "SignPageResource"; //$NON-NLS-1$
    /** Select detached file button. */
    private Button selectDetachedFile = null;
    /** Browse XPath button. */
    private Button selectXPath = null;
    /** Activate BSP checkbox. */
    private Button bBsp = null;
    /** Radio to sign the complete document. */
    private Button bDocument = null;
    /** Radio to create a detached signature. */
    private Button bDetached = null;
    /** Radio to create an enveloped signature. */
    private Button bEnveloped = null;
    /** Radio to create an enveloping signature. */
    private Button bEnveloping = null;
    /** Radio to sign the selected document fragment. */
    private Button bSelection = null;
    /** Radio to sign with the XPath expression. */
    private Button bXpath = null;
    /** Textfield for the detached file to sign. */
    private Text tDetachedFile = null;
    /** Textfield for the XPath expression to sign. */
    private Text tXpath = null;
    /** Text selection in document flag. */
    private boolean textSelection;
    /** Selected XML document to sign. */
    private Document doc = null;
    /** Empty text field. */
    private static final String EMPTY = "";
    /** Stored setting for the BSP signature selection. */
    private static final String SETTING_BSP_COMPLIANT_SIGNATURE = "sign_bsp_compliant";
    /** Model for the XML Signature Wizard. */
    private Signature signature = null;
    /** All signature IDs in the current XML document. */
    private String[] ids;
    private boolean globalError = false;

    /**
     * Constructor for the resource page of the wizard.
     * 
     * @param signature The signature wizard model
     * @param data The selected file
     * @param textSelection Status of text selection in editor
     */
    public PageResource(final Signature signature, final InputStream data, final boolean textSelection) {
        super(PAGE_NAME);
        setTitle(Messages.signatureTitle);
        setDescription(Messages.resourceDescription);

        this.signature = signature;
        this.textSelection = textSelection;

        try {
            doc = Utils.parse(data);
            ids = Utils.getAllIds(doc);
        } catch (Exception ex) {
            updateStatus(Messages.documentInvalid, IMessageProvider.ERROR);
            globalError = true;
        }
    }

    /**
     * Creates the wizard page with the layout settings.
     * 
     * @param parent The parent Composite
     */
    public void createControl(final Composite parent) {
        Composite container = new Composite(parent, SWT.NULL);
        FormLayout formLayout = new FormLayout();
        container.setLayout(formLayout);

        createPageContent(container);
        addListeners();
        setControl(container);
        loadSettings();
        saveDataToModel();

        PlatformUI.getWorkbench().getHelpSystem().setHelp(getControl(), IContextHelpIds.WIZARD_SIGNATURE_RESOURCE);
    }

    /**
     * Fills this wizard page with content. Three groups (<i>Resource</i>, <i>Signature Type</i> and <i>Basic Security
     * Profile</i>) and all their widgets are inserted.
     * 
     * @param parent The parent Composite
     */
    private void createPageContent(final Composite parent) {
        FormLayout layout = new FormLayout();
        layout.marginTop = IGlobals.MARGIN / 2;
        layout.marginBottom = IGlobals.MARGIN / 2;
        layout.marginLeft = IGlobals.MARGIN / 2;
        layout.marginRight = IGlobals.MARGIN / 2;
        parent.setLayout(layout);

        // Three groups
        Group gResource = new Group(parent, SWT.SHADOW_ETCHED_IN);
        gResource.setLayout(layout);
        gResource.setText(Messages.resource);
        FormData data = new FormData();
        data.top = new FormAttachment(0, 0);
        data.left = new FormAttachment(0, 0);
        data.right = new FormAttachment(IGlobals.GROUP_NUMERATOR);
        gResource.setLayoutData(data);

        Group gType = new Group(parent, SWT.SHADOW_ETCHED_IN);
        gType.setLayout(layout);
        gType.setText(Messages.signatureType);
        data = new FormData();
        data.top = new FormAttachment(gResource, IGlobals.MARGIN, SWT.DEFAULT);
        data.left = new FormAttachment(0, 0);
        data.right = new FormAttachment(IGlobals.GROUP_NUMERATOR);
        gType.setLayoutData(data);

        Group gBsp = new Group(parent, SWT.SHADOW_ETCHED_IN);
        gBsp.setLayout(layout);
        gBsp.setText(Messages.basicSecurityProfile);
        data = new FormData();
        data.top = new FormAttachment(gType, IGlobals.MARGIN, SWT.DEFAULT);
        data.left = new FormAttachment(0, 0);
        data.right = new FormAttachment(IGlobals.GROUP_NUMERATOR);
        gBsp.setLayoutData(data);

        // Elements for group "Resource"
        bDocument = new Button(gResource, SWT.RADIO);
        bDocument.setText(Messages.document);
        bDocument.setSelection(true);
        data = new FormData();
        data.top = new FormAttachment(gResource);
        data.left = new FormAttachment(gResource);
        bDocument.setLayoutData(data);

        bSelection = new Button(gResource, SWT.RADIO);
        bSelection.setText(Messages.selection);
        bSelection.setEnabled(textSelection);
        data = new FormData();
        data.top = new FormAttachment(bDocument, IGlobals.MARGIN);
        data.left = new FormAttachment(gResource);
        bSelection.setLayoutData(data);

        bXpath = new Button(gResource, SWT.RADIO);
        bXpath.setText(Messages.xpath);
        data = new FormData();
        data.top = new FormAttachment(bSelection, IGlobals.MARGIN);
        data.left = new FormAttachment(gResource);
        bXpath.setLayoutData(data);

        tXpath = new Text(gResource, SWT.SINGLE);
        tXpath.setEnabled(false);
        data = new FormData();
        data.top = new FormAttachment(bXpath, 0, SWT.CENTER);
        data.left = new FormAttachment(gResource, 100);
        data.width = IGlobals.MEDIUM_TEXT_WIDTH;
        tXpath.setLayoutData(data);

        selectXPath = new Button(gResource, SWT.PUSH);
        selectXPath.setText(Messages.browse);
        selectXPath.setEnabled(false);
        data = new FormData();
        data.top = new FormAttachment(bXpath, 0, SWT.CENTER);
        data.left = new FormAttachment(tXpath, IGlobals.MARGIN);
        selectXPath.setLayoutData(data);

        // Elements for group "Signature Type"
        bEnveloping = new Button(gType, SWT.RADIO);
        bEnveloping.setText(Messages.enveloping);
        bEnveloping.setSelection(true);
        data = new FormData();
        data.top = new FormAttachment(gType);
        data.left = new FormAttachment(gType);
        bEnveloping.setLayoutData(data);

        bEnveloped = new Button(gType, SWT.RADIO);
        bEnveloped.setText(Messages.enveloped);
        data = new FormData();
        data.top = new FormAttachment(bEnveloping, IGlobals.MARGIN);
        data.left = new FormAttachment(gType);
        bEnveloped.setLayoutData(data);

        bDetached = new Button(gType, SWT.RADIO);
        bDetached.setText(Messages.detached);
        data = new FormData();
        data.top = new FormAttachment(bEnveloped, IGlobals.MARGIN);
        data.left = new FormAttachment(gType);
        bDetached.setLayoutData(data);

        tDetachedFile = new Text(gType, SWT.SINGLE);
        tDetachedFile.setEnabled(false);
        data = new FormData();
        data.top = new FormAttachment(bDetached, 0, SWT.CENTER);
        data.left = new FormAttachment(gType, 100);
        data.width = IGlobals.MEDIUM_TEXT_WIDTH;
        tDetachedFile.setLayoutData(data);

        selectDetachedFile = new Button(gType, SWT.PUSH);
        selectDetachedFile.setText(Messages.select);
        selectDetachedFile.setEnabled(false);
        data = new FormData();
        data.top = new FormAttachment(bDetached, 0, SWT.CENTER);
        data.left = new FormAttachment(tDetachedFile, IGlobals.MARGIN);
        selectDetachedFile.setLayoutData(data);

        // Elements for group "Basic Security Profile"
        bBsp = new Button(gBsp, SWT.CHECK);
        bBsp.setText(Messages.bspCompliant);
        data = new FormData();
        data.top = new FormAttachment(gBsp);
        data.left = new FormAttachment(gBsp);
        bBsp.setLayoutData(data);
    }

    /**
     * Adds all listeners for the current wizard page.
     */
    private void addListeners() {
        selectDetachedFile.addListener(SWT.Selection, this);
        selectXPath.addListener(SWT.Selection, this);
        bBsp.addListener(SWT.Selection, this);
        bDocument.addListener(SWT.Selection, this);
        bDetached.addListener(SWT.Selection, this);
        bEnveloped.addListener(SWT.Selection, this);
        bEnveloping.addListener(SWT.Selection, this);
        bSelection.addListener(SWT.Selection, this);
        bXpath.addListener(SWT.Selection, this);
        tDetachedFile.addModifyListener(new ModifyListener() {
            public void modifyText(final ModifyEvent e) {
                dialogChanged();
            }
        });
        tXpath.addModifyListener(new ModifyListener() {
            public void modifyText(final ModifyEvent e) {
                dialogChanged();
            }
        });
    }

    /**
     * Determines the (error) message for the missing field.
     */
    private void dialogChanged() {
        if (globalError) {
            return;
        }

        if (bXpath.getSelection() && tXpath.getText().isEmpty()) {
            updateStatus(Messages.enterXPath, IMessageProvider.INFORMATION);
            return;
        } else if (bXpath.getSelection() && !tXpath.getText().isEmpty()) {
            String xpathValidator = Utils.validateXPath(doc, tXpath.getText());
            if (xpathValidator.equals("none")) { //$NON-NLS-1$
                updateStatus(Messages.xpathNoElement, IMessageProvider.ERROR);
                return;
            } else if (xpathValidator.equals("multiple")) { //$NON-NLS-1$
                updateStatus(Messages.xpathMultipleElements, IMessageProvider.ERROR);
                return;
            } else if (xpathValidator.equals("attribute")) { //$NON-NLS-1$
                updateStatus(Messages.xpathAttribute, IMessageProvider.ERROR);
                return;
            }
        }
        if (bDetached.getSelection() && tDetachedFile.getText().isEmpty()) {
            updateStatus(Messages.detachedFile, IMessageProvider.INFORMATION);
            return;
        } else if (bDetached.getSelection() && !tDetachedFile.getText().isEmpty()) {
            File tempFile = new File(tDetachedFile.getText());
            if (!tempFile.exists()) {
                updateStatus(Messages.verifyDetachedFile, IMessageProvider.ERROR);
                return;
            }
        }

        updateStatus(null, IMessageProvider.NONE);
    }

    /**
     * Shows a message to the user to complete the fields on this page.
     * 
     * @param message The message for the user
     * @param status The status type of the message
     */
    private void updateStatus(final String message, final int status) {
        setMessage(message, status);
        if (message == null && getErrorMessage() == null) {
            setPageComplete(true);
            saveDataToModel();
        } else {
            setPageComplete(false);
        }
    }

    /**
     * Selection dialog to select the XPath for the element to sign.
     */
    private void openXPathDialog() {
        XpathDialog dialog = new XpathDialog(getShell(), new LabelProvider(), doc, Messages.xpathPopup);
        if (dialog.getReturnCode() == 0) {
            Object[] selected = dialog.getResult();
            if (selected.length == 1) {
                tXpath.setText(selected[0].toString());
            }
        }
    }

    /**
     * Opens a standard file dialog to select the detached file to sign.
     */
    private void selectDetachedFile() {
        FileDialog dialog = new FileDialog(getShell(), SWT.OPEN);
        dialog.setFilterPath(DirectoryService.getUserHomeDir());
        dialog.setFilterNames(new String[] { IConstants.XML_FILTER_NAME, IConstants.ALL_FILTER_NAME });
        dialog.setFilterExtensions(
                new String[] { IConstants.XML_FILTER_EXTENSION, IConstants.ALL_FILTER_EXTENSION });
        String filename = dialog.open();
        if (filename != null && !filename.isEmpty()) {
            tDetachedFile.setText(filename);
        }
    }

    /**
     * Handles the events from this wizard page. Affects mainly the (de-) activation state of radio buttons.
     * 
     * @param e The triggered event
     */
    public void handleEvent(final Event e) {
        if (e.widget == bDocument || e.widget == bSelection) { // Radios Complete document or Selection
            tXpath.setEnabled(false);
            tXpath.setText(EMPTY);
            selectXPath.setEnabled(false);
        } else if (e.widget == bXpath) { // Radio XPath
            tXpath.setEnabled(true);
            selectXPath.setEnabled(true);
        } else if (e.widget == selectXPath) { // Button Browse XPath
            openXPathDialog();
        } else if (e.widget == bEnveloping || e.widget == bEnveloped) { // Radio Enveloping and Enveloped
            selectDetachedFile.setEnabled(false);
            tDetachedFile.setEnabled(false);
            tDetachedFile.setText(EMPTY);
            bXpath.setEnabled(true);

            if (textSelection) {
                bSelection.setEnabled(true);
            }
        } else if (e.widget == bDetached) { // Radio Detached
            bDocument.setSelection(true);
            bSelection.setSelection(false);
            bXpath.setSelection(false);
            bSelection.setEnabled(false);
            bXpath.setEnabled(false);
            tXpath.setText(EMPTY);
            tXpath.setEnabled(false);
            selectXPath.setEnabled(false);
            tDetachedFile.setEnabled(true);
            selectDetachedFile.setEnabled(true);
        } else if (e.widget == selectDetachedFile) { // Button Select detached file
            selectDetachedFile();
        } else if (e.widget == bBsp) { // Checkbox BSP
            if (bBsp.getSelection()) {
                bDocument.setSelection(true);
                bSelection.setSelection(false);
                bXpath.setSelection(false);
                bSelection.setEnabled(false);
                bXpath.setEnabled(false);
                tXpath.setText(EMPTY);
                tXpath.setEnabled(false);
                tDetachedFile.setEnabled(true);
                selectDetachedFile.setEnabled(true);
                bEnveloping.setEnabled(false);
                bEnveloping.setSelection(false);
                bEnveloped.setSelection(false);
                bDetached.setSelection(true);
                dialogChanged();
            } else {
                if (textSelection && !bDetached.getSelection()) {
                    bSelection.setEnabled(true);
                }
                if (!bDetached.getSelection()) {
                    bXpath.setEnabled(true);
                }
                bEnveloping.setEnabled(true);
            }
        }
    }

    /**
     * Saves the selections on this wizard page to the model. Called on exit of the page.
     */
    private void saveDataToModel() {
        if (bDocument.getSelection()) {
            signature.setResource("document"); //$NON-NLS-1$
        } else if (bSelection.getSelection()) {
            signature.setResource("selection"); //$NON-NLS-1$
        } else if (bXpath.getSelection()) {
            signature.setResource("xpath"); //$NON-NLS-1$
            signature.setXpath(tXpath.getText());
        }
        if (bEnveloping.getSelection()) {
            signature.setSignatureType("enveloping"); //$NON-NLS-1$
        } else if (bEnveloped.getSelection()) {
            signature.setSignatureType("enveloped"); //$NON-NLS-1$
        } else if (bDetached.getSelection()) {
            signature.setSignatureType("detached"); //$NON-NLS-1$
            signature.setDetachedFile(new File(tDetachedFile.getText()));
        }
        signature.setBsp(bBsp.getSelection());
        signature.setIds(ids);
        signature.setDocument(doc);
    }

    /**
     * Loads the stored settings for this wizard page.
     */
    private void loadSettings() {
        bBsp.setSelection(getDialogSettings().get(SETTING_BSP_COMPLIANT_SIGNATURE) != null
                ? getDialogSettings().getBoolean(SETTING_BSP_COMPLIANT_SIGNATURE)
                : false);
    }

    /**
     * Stores some settings of this wizard page in the current workspace.
     */
    protected void storeSettings() {
        IDialogSettings settings = getDialogSettings();
        settings.put(SETTING_BSP_COMPLIANT_SIGNATURE, bBsp.getSelection());
    }
}