org.exist.client.QueryDialog Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of exist-core Show documentation
Show all versions of exist-core Show documentation
eXist-db NoSQL Database Core
/*
* eXist Open Source Native XML Database
* Copyright (C) 2001-2015 The eXist-db Project
* http://exist-db.org
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
package org.exist.client;
import java.awt.BorderLayout;
import java.awt.Cursor;
import java.awt.Dimension;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.*;
import java.net.URL;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import javax.swing.BorderFactory;
import javax.swing.Box;
import javax.swing.DefaultComboBoxModel;
import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JComboBox;
import javax.swing.JComponent;
import javax.swing.JFileChooser;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JProgressBar;
import javax.swing.JSpinner;
import javax.swing.JSplitPane;
import javax.swing.JTabbedPane;
import javax.swing.JTextField;
import javax.swing.JToolBar;
import javax.swing.SpinnerNumberModel;
import javax.swing.border.BevelBorder;
import javax.xml.transform.OutputKeys;
import org.exist.security.PermissionDeniedException;
import org.exist.xmldb.EXistXQueryService;
import org.exist.xmldb.LocalCollection;
import org.exist.xmldb.UserManagementService;
import org.exist.xmldb.XmldbURI;
import org.exist.xquery.CompiledXQuery;
import org.exist.xquery.XQueryContext;
import org.exist.xquery.XQueryWatchDog;
import org.fife.ui.rsyntaxtextarea.RSyntaxTextArea;
import org.fife.ui.rsyntaxtextarea.SyntaxConstants;
import org.fife.ui.rtextarea.RTextScrollPane;
import org.xmldb.api.base.Collection;
import org.xmldb.api.base.CompiledExpression;
import org.xmldb.api.base.Resource;
import org.xmldb.api.base.ResourceIterator;
import org.xmldb.api.base.ResourceSet;
import org.xmldb.api.base.XMLDBException;
import org.xmldb.api.modules.XMLResource;
import static java.nio.charset.StandardCharsets.UTF_8;
public class QueryDialog extends JFrame {
private static final long serialVersionUID = 1L;
private InteractiveClient client;
private Collection collection;
private Properties properties;
private RSyntaxTextArea query;
private RTextScrollPane queryScrollPane;
private JTabbedPane resultTabs;
private RSyntaxTextArea resultDisplay;
private RTextScrollPane resultDisplayScrollPane;
private RSyntaxTextArea exprDisplay;
private RTextScrollPane exprDisplayScrollPane;
private JComboBox collections = null;
private SpinnerNumberModel count;
private DefaultComboBoxModel history = new DefaultComboBoxModel();
private JTextField statusMessage;
private JTextField queryPositionDisplay;
private JProgressBar progress;
private JButton submitButton;
private JButton killButton;
private QueryRunnable queryRunnable = null;
private Resource resource = null;
private QueryDialog(final InteractiveClient client, final Collection collection, final Properties properties, boolean loadedFromDb) {
super(Messages.getString("QueryDialog.0"));
this.collection = collection;
this.properties = properties;
this.client = client;
this.setIconImage(InteractiveClient.getExistIcon(getClass()).getImage());
setupComponents(loadedFromDb);
pack();
}
public QueryDialog(final InteractiveClient client, final Collection collection, final Properties properties) {
this(client, collection, properties, false);
}
public QueryDialog(final InteractiveClient client, final Collection collection, final Resource resource, final Properties properties) throws XMLDBException {
this(client, collection, properties, true);
this.resource = resource;
addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent ev) {
try {
final UserManagementService service = (UserManagementService) collection
.getService("UserManagementService", "1.0"); //$NON-NLS-1$ //$NON-NLS-2$
service.unlockResource(resource);
} catch (final XMLDBException e) {
e.printStackTrace();
}
}
});
//set the content of the query
query.setText(new String((byte[]) resource.getContent()));
//set title
setTitle(Messages.getString("QueryDialog.0") + ": " + resource.getId());
}
private void saveToDb(final String queryText) {
try {
resource.setContent(queryText);
collection.storeResource(resource);
} catch (final XMLDBException xmldbe) {
ClientFrame.showErrorMessage(xmldbe.getMessage(), xmldbe);
}
}
private void setupComponents(boolean loadedFromDb) {
getContentPane().setLayout(new BorderLayout());
final JToolBar toolbar = new JToolBar();
URL url = getClass().getResource("icons/Open24.gif");
JButton button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.opentooltip"));
button.addActionListener(e -> open());
toolbar.add(button);
if (loadedFromDb) {
url = getClass().getResource("icons/SaveAs23.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText("Save to database");
button.addActionListener(e -> saveToDb(query.getText()));
toolbar.add(button);
}
url = getClass().getResource("icons/SaveAs24.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.saveastooltip"));
button.addActionListener(e -> save(query.getText(), "query"));
toolbar.add(button);
url = getClass().getResource("icons/SaveAs25.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.saveresultstooltip"));
button.addActionListener(e -> save(resultDisplay.getText(), "result"));
toolbar.add(button);
toolbar.addSeparator();
url = getClass().getResource("icons/Copy24.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.copytooltip"));
button.addActionListener(e -> query.copy());
toolbar.add(button);
url = getClass().getResource("icons/Cut24.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.cuttooltip"));
button.addActionListener(e -> query.cut());
toolbar.add(button);
url = getClass().getResource("icons/Paste24.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.pastetooltip"));
button.addActionListener(e -> query.paste());
toolbar.add(button);
toolbar.addSeparator();
//TODO: change icon
url = getClass().getResource("icons/Find24.gif");
button = new JButton(new ImageIcon(url));
button.setToolTipText(Messages.getString("QueryDialog.compiletooltip"));
button.addActionListener(e -> compileQuery());
toolbar.add(button);
toolbar.addSeparator();
url = getClass().getResource("icons/Find24.gif");
submitButton = new JButton(Messages.getString("QueryDialog.submitbutton"), new ImageIcon(url));
submitButton.setToolTipText(Messages.getString("QueryDialog.submittooltip"));
toolbar.add(submitButton);
submitButton.addActionListener(e -> {
submitButton.setEnabled(false);
if (collection instanceof LocalCollection) {
killButton.setEnabled(true);
}
queryRunnable = doQuery();
});
toolbar.addSeparator();
url = getClass().getResource("icons/Delete24.gif");
killButton = new JButton(Messages.getString("QueryDialog.killbutton"), new ImageIcon(url));
killButton.setToolTipText(Messages.getString("QueryDialog.killtooltip"));
toolbar.add(killButton);
killButton.setEnabled(false);
killButton.addActionListener(e -> {
if (queryRunnable != null) {
queryRunnable.killQuery();
killButton.setEnabled(false);
queryRunnable = null;
}
});
final JSplitPane split = new JSplitPane(JSplitPane.VERTICAL_SPLIT);
split.setResizeWeight(0.6);
final JComponent qbox = createQueryBox();
split.setTopComponent(qbox);
final JPanel vbox = new JPanel();
vbox.setLayout(new BorderLayout());
final JLabel label = new JLabel(Messages.getString("QueryDialog.resultslabel"));
vbox.add(label, BorderLayout.NORTH);
resultTabs = new JTabbedPane();
resultDisplay = new RSyntaxTextArea(14, 80);
resultDisplay.setEditable(false);
resultDisplay.getPopupMenu().remove(0); //remove undo
resultDisplay.getPopupMenu().remove(0); //remove redo
resultDisplay.getPopupMenu().remove(0); //remove undo/redo splitter
resultDisplay.setSyntaxEditingStyle(SyntaxConstants.SYNTAX_STYLE_XML);
resultDisplay.setCodeFoldingEnabled(true);
resultDisplayScrollPane = new RTextScrollPane(resultDisplay);
resultTabs.add(Messages.getString("QueryDialog.XMLtab"), resultDisplayScrollPane);
exprDisplay = new RSyntaxTextArea(14, 80);
exprDisplay.setEditable(false);
exprDisplay.getPopupMenu().remove(0); //remove undo
exprDisplay.getPopupMenu().remove(0); //remove redo
exprDisplay.getPopupMenu().remove(0); //remove undo/redo splitter
exprDisplayScrollPane = new RTextScrollPane(exprDisplay);
resultTabs.add(Messages.getString("QueryDialog.tracetab"), exprDisplayScrollPane);
vbox.add(resultTabs, BorderLayout.CENTER);
final Box statusbar = Box.createHorizontalBox();
statusbar.setBorder(BorderFactory.createBevelBorder(BevelBorder.LOWERED));
statusMessage = new JTextField(20);
statusMessage.setEditable(false);
statusMessage.setFocusable(true);
statusbar.add(statusMessage);
queryPositionDisplay = new JTextField(5);
queryPositionDisplay.setEditable(false);
queryPositionDisplay.setFocusable(true);
statusbar.add(queryPositionDisplay);
query.addCaretListener(e -> {
final RSyntaxTextArea txt = (RSyntaxTextArea) e.getSource();
queryPositionDisplay.setText("Line: " + (txt.getCaretLineNumber() + 1) + " Column:" + (txt.getCaretOffsetFromLineStart() + 1));
});
progress = new JProgressBar();
progress.setPreferredSize(new Dimension(200, statusbar.getHeight()));
progress.setVisible(false);
statusbar.add(progress);
vbox.add(statusbar, BorderLayout.SOUTH);
split.setBottomComponent(vbox);
split.setDividerLocation(0.6);
getContentPane().add(toolbar, BorderLayout.NORTH);
getContentPane().add(split, BorderLayout.CENTER);
}
private JComponent createQueryBox() {
final JTabbedPane tabs = new JTabbedPane();
final JPanel inputVBox = new JPanel();
inputVBox.setLayout(new BorderLayout());
tabs.add(Messages.getString("QueryDialog.inputtab"), inputVBox);
final Box historyBox = Box.createHorizontalBox();
JLabel label = new JLabel(Messages.getString("QueryDialog.historylabel"));
historyBox.add(label);
final JComboBox historyList = new JComboBox(history);
for (final String query : client.queryHistory) {
addQuery(query);
}
historyList.addActionListener(e -> {
final String item = (String) client.queryHistory.get(historyList.getSelectedIndex());
query.setText(item);
});
historyBox.add(historyList);
inputVBox.add(historyBox, BorderLayout.NORTH);
query = new RSyntaxTextArea(14, 80);
query.setSyntaxEditingStyle(SyntaxConstants.SYNTAX_STYLE_XML);
query.setCodeFoldingEnabled(true);
queryScrollPane = new RTextScrollPane(query);
inputVBox.add(queryScrollPane, BorderLayout.CENTER);
final Box optionsPanel = Box.createHorizontalBox();
label = new JLabel(Messages.getString("QueryDialog.contextlabel"));
optionsPanel.add(label);
final List data = new ArrayList();
try {
final Collection root = client.getCollection(XmldbURI.ROOT_COLLECTION);
data.add(collection.getName());
getCollections(root, collection, data);
} catch (final XMLDBException e) {
ClientFrame.showErrorMessage(
Messages.getString("QueryDialog.collectionretrievalerrormessage") + ".", e);
}
collections = new JComboBox(new java.util.Vector(data));
collections.addActionListener(e -> {
final int p = collections.getSelectedIndex();
final String context = data.get(p);
try {
collection = client.getCollection(context);
} catch (final XMLDBException e1) {
}
});
optionsPanel.add(collections);
label = new JLabel(Messages.getString("QueryDialog.maxlabel"));
optionsPanel.add(label);
count = new SpinnerNumberModel(100, 1, 10000, 50);
final JSpinner spinner = new JSpinner(count);
spinner.setMaximumSize(new Dimension(400, 100));
optionsPanel.add(spinner);
inputVBox.add(optionsPanel, BorderLayout.SOUTH);
return tabs;
}
private List getCollections(final Collection root, final Collection collection, final List collectionsList) throws XMLDBException {
if (!collection.getName().equals(root.getName())) {
collectionsList.add(root.getName());
}
final String[] childCollections = root.listChildCollections();
Collection child = null;
for (int i = 0; i < childCollections.length; i++) {
try {
child = root.getChildCollection(childCollections[i]);
} catch (final XMLDBException xmldbe) {
if (xmldbe.getCause() instanceof PermissionDeniedException) {
continue;
} else {
throw xmldbe;
}
} catch (Exception npe) {
System.out.println("Corrupted resource/collection skipped: " + child != null ? child.getName() != null ? child.getName() : "unknown" : "unknown");
continue;
}
try {
getCollections(child, collection, collectionsList);
} catch (Exception ee) {
System.out.println("Corrupted resource/collection skipped: " + child != null ? child.getName() != null ? child.getName() : "unknown" : "unknown");
continue;
}
}
return collectionsList;
}
private void open() {
final String workDir = properties.getProperty("working-dir", System.getProperty("user.dir"));
final JFileChooser chooser = new JFileChooser();
chooser.setCurrentDirectory(Paths.get(workDir).toFile());
chooser.setMultiSelectionEnabled(false);
chooser.setFileSelectionMode(JFileChooser.FILES_ONLY);
chooser.addChoosableFileFilter(new MimeTypeFileFilter("application/xquery"));
if (chooser.showDialog(this, Messages.getString("QueryDialog.opendialog")) == JFileChooser.APPROVE_OPTION) {
final Path selectedDir = chooser.getCurrentDirectory().toPath();
properties.setProperty("working-dir", selectedDir.toAbsolutePath().toString());
final Path file = chooser.getSelectedFile().toPath();
if (!Files.isReadable(file)) {
JOptionPane.showInternalMessageDialog(this, Messages.getString("QueryDialog.cannotreadmessage") + " " + file.toAbsolutePath(),
Messages.getString("QueryDialog.Error"), JOptionPane.ERROR_MESSAGE);
}
try (final BufferedReader reader = Files.newBufferedReader(file)) {
final StringBuilder buf = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
buf.append(line);
buf.append('\n');
}
query.setText(buf.toString());
} catch (final IOException e) {
ClientFrame.showErrorMessage(e.getMessage(), e);
}
}
}
private void save(String stringToSave, String fileCategory) {
if (stringToSave == null || "".equals(stringToSave)) {
return;
}
final String workDir = properties.getProperty("working-dir", System.getProperty("user.dir"));
final JFileChooser chooser = new JFileChooser();
chooser.setMultiSelectionEnabled(false);
chooser.setCurrentDirectory(Paths.get(workDir).toFile());
chooser.setFileSelectionMode(JFileChooser.FILES_ONLY);
if ("result".equals(fileCategory)) {
chooser.addChoosableFileFilter(new MimeTypeFileFilter("application/xhtml+xml"));
chooser.addChoosableFileFilter(new MimeTypeFileFilter("application/xml"));
} else {
chooser.addChoosableFileFilter(new MimeTypeFileFilter("application/xquery"));
}
if (chooser.showDialog(this, Messages.getString("QueryDialog.savedialogpre") + " " + fileCategory + " " + Messages.getString("QueryDialog.savedialogpost"))
== JFileChooser.APPROVE_OPTION) {
final Path selectedDir = chooser.getCurrentDirectory().toPath();
properties.setProperty("working-dir", selectedDir.toAbsolutePath().toString());
final Path file = chooser.getSelectedFile().toPath();
if (Files.exists(file) && (!Files.isWritable(file))) {
JOptionPane.showMessageDialog(this, Messages.getString("QueryDialog.cannotsavemessagepre") + " " + fileCategory + " " + Messages.getString("QueryDialog.cannotsavemessageinf") + " " + file.toAbsolutePath(),
Messages.getString("QueryDialog.Error"), JOptionPane.ERROR_MESSAGE);
}
if (Files.exists(file) &&
JOptionPane.showConfirmDialog(this, Messages.getString("QueryDialog.savedialogconfirm"), "Overwrite?",
JOptionPane.YES_NO_OPTION) == JOptionPane.NO_OPTION) {
return;
}
try(final Writer writer = Files.newBufferedWriter(file)) {
writer.write(stringToSave);
} catch (final IOException e) {
ClientFrame.showErrorMessage(e.getMessage(), e);
}
}
}
private static final AtomicInteger queryThreadId = new AtomicInteger();
private QueryRunnable doQuery() {
final String xpath = query.getText();
if (xpath.length() == 0) {
return null;
}
resultDisplay.setText("");
final QueryRunnable queryRunnable = new QueryRunnable(xpath);
final Thread queryThread = client.newClientThread("query-" + queryThreadId.getAndIncrement(), queryRunnable);
queryThread.start();
return queryRunnable;
}
private void compileQuery() {
final String xpath = (String) query.getText();
if (xpath.length() == 0) {
return;
}
resultDisplay.setText("");
{
statusMessage.setText(Messages.getString("QueryDialog.compilemessage"));
setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR));
final long tResult = 0;
long tCompiled = 0;
try {
final EXistXQueryService service = (EXistXQueryService) collection.getService("XQueryService", "1.0");
service.setProperty(OutputKeys.INDENT, properties.getProperty(OutputKeys.INDENT, "yes"));
final long t0 = System.currentTimeMillis();
final CompiledExpression compiled = service.compile(xpath);
final long t1 = System.currentTimeMillis();
tCompiled = t1 - t0;
// In this way we can see the parsed structure meanwhile the query is
final StringWriter writer = new StringWriter();
service.dump(compiled, writer);
exprDisplay.setText(writer.toString());
resultTabs.setSelectedComponent(exprDisplayScrollPane);
statusMessage.setText(Messages.getString("QueryDialog.Compilation") + ": " + tCompiled + "ms");
} catch (final Throwable e) {
statusMessage.setText(Messages.getString("QueryDialog.Error") + ": " + InteractiveClient.getExceptionMessage(e) + ". " + Messages.getString("QueryDialog.Compilation") + ": " + tCompiled + "ms, " + Messages.getString("QueryDialog.Execution") + ": " + tResult + "ms");
ClientFrame.showErrorMessageQuery(
Messages.getString("QueryDialog.compilationerrormessage") + ": "
+ InteractiveClient.getExceptionMessage(e), e);
}
setCursor(Cursor.getDefaultCursor());
}
}
private class QueryRunnable implements Runnable {
private final String xpath;
private final AtomicReference runningContext = new AtomicReference<>();
public QueryRunnable(final String query) {
this.xpath = query;
}
public boolean killQuery() {
final XQueryContext contextRef = runningContext.get();
if (contextRef != null) {
final XQueryWatchDog xwd = contextRef.getWatchDog();
final boolean retval = !xwd.isTerminating();
if (retval) {
xwd.kill(0);
}
runningContext.compareAndSet(contextRef, null);
return retval;
}
return false;
}
@Override
public void run() {
statusMessage.setText(Messages.getString("QueryDialog.processingquerymessage"));
progress.setVisible(true);
progress.setIndeterminate(true);
setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR));
long tResult = 0;
long tCompiled = 0;
ResourceSet result = null;
XQueryContext context = null;
try {
final EXistXQueryService service = (EXistXQueryService) collection.getService("XQueryService", "1.0");
service.setProperty(OutputKeys.INDENT, properties.getProperty(OutputKeys.INDENT, "yes"));
final long t0 = System.currentTimeMillis();
if (resource != null) {
service.setModuleLoadPath(XmldbURI.EMBEDDED_SERVER_URI_PREFIX + resource.getParentCollection().getName());
}
final CompiledExpression compiled = service.compile(xpath);
final long t1 = System.currentTimeMillis();
// Check could also be collection instanceof LocalCollection
if (compiled instanceof CompiledXQuery) {
context = ((CompiledXQuery) compiled).getContext();
runningContext.set(context);
}
tCompiled = t1 - t0;
// In this way we can see the parsed structure meanwhile the query is
StringWriter writer = new StringWriter();
service.dump(compiled, writer);
exprDisplay.setText(writer.toString());
result = service.execute(compiled);
tResult = System.currentTimeMillis() - t1;
runningContext.set(null);
// jmfg: Is this still needed? I don't think so
writer = new StringWriter();
service.dump(compiled, writer);
exprDisplay.setText(writer.toString());
statusMessage.setText(Messages.getString("QueryDialog.retrievingmessage"));
Resource resource;
final int howmany = count.getNumber().intValue();
progress.setIndeterminate(false);
progress.setMinimum(1);
progress.setMaximum(howmany);
int j = 0;
int select = -1;
final StringBuilder contents = new StringBuilder();
for (final ResourceIterator i = result.getIterator(); i.hasMoreResources() && j < howmany; j++) {
resource = i.nextResource();
progress.setValue(j);
try {
if (resource.getResourceType() == XMLResource.RESOURCE_TYPE) {
contents.append((String) resource.getContent());
} else {
contents.append(new String((byte[])resource.getContent(), UTF_8));
}
contents.append("\n");
} catch (final XMLDBException e) {
select = ClientFrame.showErrorMessageQuery(
Messages.getString("QueryDialog.retrievalerrormessage") + ": "
+ InteractiveClient.getExceptionMessage(e), e);
if (select == 3) {
break;
}
}
}
resultTabs.setSelectedComponent(resultDisplayScrollPane);
resultDisplay.setText(contents.toString());
resultDisplay.setCaretPosition(0);
statusMessage.setText(Messages.getString("QueryDialog.Found") + " " + result.getSize() + " " + Messages.getString("QueryDialog.items") + "." +
" " + Messages.getString("QueryDialog.Compilation") + ": " + tCompiled + "ms, " + Messages.getString("QueryDialog.Execution") + ": " + tResult + "ms");
} catch (final XMLDBException e) {
statusMessage.setText(Messages.getString("QueryDialog.Error") + ": " + InteractiveClient.getExceptionMessage(e) + ". " + Messages.getString("QueryDialog.Compilation") + ": " + tCompiled + "ms, " + Messages.getString("QueryDialog.Execution") + ": " + tResult + "ms");
progress.setVisible(false);
ClientFrame.showErrorMessageQuery(
Messages.getString("QueryDialog.queryrunerrormessage") + ": "
+ InteractiveClient.getExceptionMessage(e), e);
} finally {
if (context != null) {
context.runCleanupTasks();
}
context = null;
if (result != null)
try {
result.clear();
} catch (final XMLDBException e) {
}
}
if (client.queryHistory.isEmpty() || !client.queryHistory.getLast().equals(xpath)) {
client.addToHistory(xpath);
client.writeQueryHistory();
addQuery(xpath);
}
setCursor(Cursor.getDefaultCursor());
progress.setVisible(false);
killButton.setEnabled(false);
submitButton.setEnabled(true);
}
}
private void addQuery(String query) {
if (query.length() > 40) {
query = query.substring(0, 40);
}
history.addElement(Integer.toString(history.getSize() + 1) + ". " + query);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy