PostSaveDocument异步调用代理

时间:2012-06-04 10:03:33

标签: xpages asynchronous xpages-ssjs

我有一个带有单个Notes文档数据源的Xpage页面。

保存文档后,我想(有条件地)触发代理。代理需要一些时间来处理,我们不希望用户必须等待结果,所以它应该异步执行。

我已经设法通过使用XHR到代理URL从客户端JS开始工作,但我想在服务器端执行此操作以便我可以更好地控制“下一页”。使用.run().runonserver()时,客户端会等待代理完成。

在没有客户端等待结果的情况下,我怎么能在PostSaveDocument上触发代理(来自SSJS)?

4 个答案:

答案 0 :(得分:1)

尝试在OpenNTF.org上查看Thread和Jobs应用程序。在后台运行任务有很好的演示,请检查it here

答案 1 :(得分:1)

正如Martin建议我在OpenNtf上使用JobScheduler示例并对其进行修改以满足我的需要。结果代码可以在下面找到。欢迎提出任何意见或改进。

import java.security.AccessController;
import java.security.PrivilegedAction;

import lotus.domino.Agent;
import lotus.domino.Database;

import lotus.domino.NotesException;
import lotus.domino.Session;

import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.core.runtime.jobs.IJobChangeEvent;
import org.eclipse.core.runtime.jobs.Job;
import org.eclipse.core.runtime.jobs.JobChangeAdapter;

import com.ibm.domino.xsp.module.nsf.ThreadSessionExecutor;

public class JobRunner {

public static void start(String dbPath, String agentName, String paramDocId) {

    synchronized (JobRunner.class) {

        runningJob = new ISPJob(dbPath, agentName, paramDocId);
        runningJob.addJobChangeListener(new JobChangeAdapter() {
            public void done(IJobChangeEvent event) {
                System.out.println("Done event");
                runningJob = null;
            }
        });
        AccessController.doPrivileged(new PrivilegedAction<Object>() {
            public Object run() {
                runningJob.schedule();
                return null;
            }
        });

    }
}

private static ISPJob runningJob;

private static final class ISPJob extends Job {

    private ThreadSessionExecutor<IStatus> executor;

    private String docId;
    private String dbPath;
    private String agentName;

    public ISPJob(String paramDbPath, String paramAgentName, String paramDocId) {
        super(paramDocId);

        this.docId = paramDocId;
        this.dbPath = paramDbPath;
        this.agentName = paramAgentName;

        this.executor = new ThreadSessionExecutor<IStatus>() {
            @Override
            protected IStatus run(Session session) throws NotesException {

                System.out.println("Job started" + docId);
                System.out.println("   >> Session created: "
                        + session.getUserName() + ", Effective User:"
                        + session.getEffectiveUserName());

                    Database db = session.getDatabase(null,dbPath);

                    if (db != null) {
                        try {
                            if (!db.isOpen()) db.open();
                            if (db.isOpen()) {
                                System.out.println("   >> Database opened: "
                                        + db.getTitle());
                                Agent agent = db.getAgent(agentName);
                                try {
                                    System.out.println("   >> Agent Started: " + agent.getName());
                                    agent.run(docId);
                                    System.out.println("   >> Agent Ran: " + agent.getName());
                                } finally {
                                    agent.recycle();
                                }

                            }
                        } finally {
                            db.recycle();                           
                        }

                }
                System.out.println("Job completed");

                return Status.OK_STATUS;
            }
        };

    }

    protected IStatus run(IProgressMonitor monitor) {
        try {
            return executor.run();
        } catch (Exception ex) {
            return Status.CANCEL_STATUS;
        }
    }
};
}

答案 2 :(得分:0)

你可以使用一个启动Java线程的会话bean(因此不会被破坏)。或者您可以在代码中发出服务器控制台命令。或者你实现了一个DOTS监听器。

答案 3 :(得分:0)

这可能/可能不是一个选项,具体取决于您的应用程序要求,但我在onClientLoad事件中调用函数取得了很大成功,该事件基本上在XPage完全加载后启动了该过程。