Hallo,
ich hätte eine Frage zu Java Spezialisten...
Nachdem ein Kollege den u.g. Java Agenten in einer Datenbank programmiert hat, entstehen dort fast täglich Replizier- und Speicherkonflikte... :(
Kommt evtl. was verdächtig vor?
Vielen Dank im Voraus.
Gruß
Leo
import java.io.IOException;
import java.nio.charset.Charset;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.methods.PutMethod;
import java.io.ByteArrayInputStream;
public class BestellAkte {
private final String narsUrl;
public BestellAkte(String narsUrl) {
this.narsUrl = narsUrl;
}
public void syncAgorumWithNotes(EinkaufsTicket ekTicket) throws Exception {
ekTicket.createLock();
if (!isCommunicationWithAgorumNecessary(ekTicket.getBestellStatus())) {
ekTicket.setAgorumStatus("Keine Bestellakte vorhanden.");
} else {
try {
if (!exists(ekTicket.getBestellNummer())) {
createAgorumBestellAkte(ekTicket);
} else {
updateAgorumBestellAkte(ekTicket);
}
ekTicket.setAgorumUrl(getAgorumUrl(ekTicket.getBestellNummer()));
ekTicket.setAgorumStatus("OK");
} catch (IOException e) {
ekTicket.setAgorumStatus("Fehler bei der Kommunikation mit NARS! "+e.getMessage());
}
}
ekTicket.saveAndReleaseLock();
if (ekTicket.errorsOnSave()) {
throw new Exception("Errors while saving document in notes! " + ekTicket.getBestellNummer());
}
}
private ByteArrayInputStream generateRequestBody(EinkaufsTicket ekTicket) {
return new ByteArrayInputStream(("{\"bestellNummer\": \"" + ekTicket.getBestellNummer() + "\", "
+ "\"bestellStatus\": \"" + ekTicket.getBestellStatus() + "\", "
+ "\"bestellGenehmiger\": \"" + ekTicket.getBestellGenehmiger() + "\", "
+ "\"bestellSumme\": \"" + ekTicket.getBestellSumme() + "\", "
+ "\"bestellGenehmigungsDatum\": \"" + ekTicket.getBestellGenehmigungsDatum() + "\", "
+ "\"bestellDatum\": \"" + ekTicket.getBestellDatum() + "\", "
+ "\"projektnummer\": \"" + ekTicket.getProjektnummer() + "\", "
+ "\"projektnummerJahr\": \"" + ekTicket.getProjektnummerJahr()+ "\", "
+ "\"kostenstelle\": \"" + ekTicket.getKostenstelle() + "\", "
+ "\"kostenstelleJahr\": \"" + ekTicket.getKostenstelleJahr()+ "\", "
+ "\"notesLink\": \"" + ekTicket.getNotesUrl() + "\" "
+ "}").getBytes(Charset.forName("UTF-8")));
}
private String createAgorumBestellAkte(EinkaufsTicket ekTicket) throws HttpException, IOException {
PostMethod method = new PostMethod(narsUrl + "bestellakte/");
HttpClient client = new HttpClient();
method.setRequestBody(generateRequestBody(ekTicket));
method.addRequestHeader("Accept", "text/plain;charset=UTF-8");
method.addRequestHeader("Content-Type", "application/json");
int statusCode = client.executeMethod(method);
if (statusCode != 201) {
throw new HttpException("Fehler beim Anlegen der Bestellakte: createAgorumBestellAkte gibt StatusCode ["+statusCode+"]");
}
byte[] responseBody = method.getResponseBody();
return new String(responseBody);
}
private void updateAgorumBestellAkte(EinkaufsTicket ekTicket) throws HttpException, IOException {
PutMethod method = new PutMethod(narsUrl + "bestellakte/"+ekTicket.getBestellNummer());
HttpClient client = new HttpClient();
method.setRequestBody(generateRequestBody(ekTicket));
method.addRequestHeader("Accept", "text/plain;charset=UTF-8");
method.addRequestHeader("Content-Type", "application/json");
int statusCode = client.executeMethod(method);
if (statusCode != 204) {
throw new HttpException("Fehler beim Aktualisieren der Bestellakte: updateAgorumBestellAkte gibt StatusCode ["+statusCode+"]");
}
}
private boolean exists(String bestellNummer) throws HttpException, IOException {
GetMethod method = new GetMethod(narsUrl + "bestellakte/" + bestellNummer);
HttpClient client = new HttpClient();
int statusCode = client.executeMethod(method);
return statusCode == 200;
}
private String getAgorumUrl(String bestellNummer) throws HttpException, IOException {
GetMethod method = new GetMethod(narsUrl + "bestellakte/" + bestellNummer + "/link");
HttpClient client = new HttpClient();
int statusCode = client.executeMethod(method);
if (statusCode != 200) {
throw new HttpException("Fehler bei der Kommunikation mit NARS! getAgorumUrl gibt StatusCode ["+statusCode+"]");
}
byte[] responseBody = method.getResponseBody();
return new String(responseBody);
}
private boolean isCommunicationWithAgorumNecessary(String bestellStatus) {
return bestellStatus.equals("Bestellung")
|| bestellStatus.equals("Bestätigung") || bestellStatus.equals("Teillieferung")
|| bestellStatus.equals("Lieferung") || bestellStatus.equals("Rechnung") || bestellStatus.equals("obsolet");
}
}
public class NotesDateConverter {
public static String convertNotesDateToJavaFormat(final String eingabeValue) {
if (eingabeValue.length() == 0) {
return eingabeValue;
}
try {
String rueckgabeValue = eingabeValue;
int pos = eingabeValue.indexOf(" ");
if (pos > 0) {
rueckgabeValue = eingabeValue.substring(0, pos);
}
pos = rueckgabeValue.indexOf("/");
if (pos > 0) {
rueckgabeValue = rueckgabeValue.replace("/", ".");
}
pos = rueckgabeValue.indexOf(".");
if (pos == 1) {
rueckgabeValue = "0"+rueckgabeValue;
}
pos = rueckgabeValue.indexOf(".", 3);
if (pos == 4) {
rueckgabeValue = rueckgabeValue.substring(0, 3) + "0" + rueckgabeValue.substring(3);
}
if (rueckgabeValue.substring(6).length() == 2) {
rueckgabeValue = rueckgabeValue.substring(0, 6) + "20" + rueckgabeValue.substring(6);
}
return rueckgabeValue;
} catch (StringIndexOutOfBoundsException e) {
return "Error: "+e.toString();
}
}
public static String entfernePunkt(String eingabeValue) {
String rueckgabeValue = eingabeValue;
int pos = eingabeValue.indexOf(".");
if (pos > 0) {
rueckgabeValue = eingabeValue.substring(0, pos);
}
return rueckgabeValue;
}
}
import lotus.domino.*;
public class BestellAkteInAgorumPflegen extends AgentBase {
public void NotesMain() {
try {
Session session = getSession();
AgentContext agentContext = session.getAgentContext();
EinkaufsTicketNotesImpl ekTicket = new EinkaufsTicketNotesImpl(agentContext.getDocumentContext());
new BestellAkte("http://nars-prod-1.osl.MyDomain.de:4380/").syncAgorumWithNotes(ekTicket);
} catch(Exception e) {
e.printStackTrace();
}
}
}
public interface EinkaufsTicket {
String getBestellNummer();
String getBestellStatus();
String getBestellGenehmiger();
String getBestellSumme();
String getBestellGenehmigungsDatum();
String getNotesUrl();
String getBestellDatum();
String getKostenstelle();
String getKostenstelleJahr();
String getProjektnummer();
String getProjektnummerJahr();
void createLock();
void saveAndReleaseLock();
void setAgorumUrl(String agorumUrl);
void setAgorumStatus(String agorumStatus);
boolean errorsOnSave();
}
import lotus.domino.*;
public class EinkaufsTicketNotesImpl implements EinkaufsTicket {
private final Document notesDocument;
private final String bestellNummer;
private final String bestellStatus;
private final String bestellGenehmiger;
private final String bestellSumme;
private final String bestellGenehmigungsDatum;
private final String notesUrl;
private final String bestellDatum;
private boolean exceptionOnSave = false;
private final String kostenstelle;
private final String kostenstelleJahr;
private final String projektnummer;
private final String projektnummerJahr;
public EinkaufsTicketNotesImpl(Document notesDocument) throws NotesException {
this.notesDocument = notesDocument;
this.bestellNummer = (String) notesDocument.getItemValue("RequestNumber").elementAt(0);
this.bestellStatus = (String) notesDocument.getItemValue("bStatus").elementAt(0);
this.bestellGenehmiger = (String) notesDocument.getItemValue("approval_control_authorized_name").elementAt(0);
this.bestellSumme = notesDocument.getItemValue("authorized_betrag").elementAt(0).toString();
this.bestellGenehmigungsDatum = NotesDateConverter.convertNotesDateToJavaFormat((String) notesDocument.getItemValue("approval_control_authorized_date").elementAt(0));
this.bestellDatum = NotesDateConverter.convertNotesDateToJavaFormat(notesDocument.getItemValueDateTimeArray("vDat_3").elementAt(0).toString());
this.kostenstelle = NotesDateConverter.entfernePunkt(notesDocument.getItemValue("numK").elementAt(0).toString());
this.kostenstelleJahr = NotesDateConverter.entfernePunkt(notesDocument.getItemValue("jahrK").elementAt(0).toString());
this.projektnummer = NotesDateConverter.entfernePunkt(notesDocument.getItemValue("numP").elementAt(0).toString());
this.projektnummerJahr = NotesDateConverter.entfernePunkt(notesDocument.getItemValue("jahrP").elementAt(0).toString());
this.notesUrl = notesDocument.getNotesURL();
}
public String getBestellNummer() {
return bestellNummer;
}
public String getBestellStatus() {
return bestellStatus;
}
public String getBestellGenehmiger() {
return this.bestellGenehmiger;
}
public String getBestellSumme() {
return this.bestellSumme.trim();
}
public String getBestellGenehmigungsDatum() {
return this.bestellGenehmigungsDatum;
}
public String getNotesUrl() {
return this.notesUrl;
}
public String getBestellDatum() {
return this.bestellDatum;
}
public void createLock() {
try {
notesDocument.lock();
} catch (NotesException e) {
exceptionOnSave = true;
}
}
public void saveAndReleaseLock() {
try {
notesDocument.computeWithForm(false, false);
notesDocument.save();
notesDocument.unlock();
} catch (NotesException e) {
exceptionOnSave = true;
}
}
public void setAgorumUrl(String agorumUrl) {
try {
notesDocument.replaceItemValue("agorumBestellakteUrl", agorumUrl);
} catch (Exception e) {
exceptionOnSave = true;
}
}
public void setAgorumStatus(String agorumStatus) {
try {
notesDocument.replaceItemValue("agorumStatus", agorumStatus);
} catch (NotesException e) {
exceptionOnSave = true;
}
}
public boolean errorsOnSave() {
return exceptionOnSave;
}
public String getKostenstelle() {
return this.kostenstelle;
}
public String getKostenstelleJahr() {
return this.kostenstelleJahr;
}
public String getProjektnummer() {
return this.projektnummer;
}
public String getProjektnummerJahr() {
return this.projektnummerJahr;
}
}
Noch eine Frage hinterher...
Der Kollege hatte den Agenten-Aufruf in PostSave eingetragen:
@Command([ToolsRunMacro];"bestellAkteInAgorumPflegen");
Ich würde es gerne in QuerySave z.B. so verschieben
'01.03.2018 Agorum Agent wird hier statt POSTSAVE Event augerufen!!!
Dim s As New NotesSession
Dim db As NotesDatabase
Dim agent2 As NotesAgent
Set db = s.CurrentDatabase
Set agent2 = db.GetAgent("bestellAkteInAgorumPflegen")
If agent2.Run = 0 Then
Print "Agorum-Agent erfolgreich gelaufen"
Else
MessageBox "Agorum-Agent konnte nicht gestartet werden",64, "Bitte Ihrem Notes Admin melden"
End If
Aber bekomme die Fehlermeldung "Could not execute Macro".
Auch dann, wenn ich den Agenten per Button mit LS aufrufe.
Außerdem, wenn ich den Agenten per Button mit FS aufrufe, bekomme ich die Fehlermeldung "Dokument noch nicht gespeichert".
Was ist da noch faul?
OK, da hat der Agent den Bezug zum Dokument verloren.
Kann man also einen bzw. so einen Java Agenten nur über FS aber nicht über LS aufrufen, da beim Aufruf über LS (egal ob in einem Event oder Button) die gleiche o.g. Fehlermeldung kommt?
Dann die Note-ID des Dokumentes dem run-Befehl mitgeben und in Java das Dokument mittels getParameterDocID holen, dann noch das Ziel beim Agenten ändern, das sollte gehen ;)
Es hört sich einfach an. ;)
Kannst Du bitte den o.g. Java-Code um die ein paar benötigten Zeilen entsprechend erweitern?
Den LS Code könnte ich so erweitern, oder?
Dim s As New NotesSession
Dim db As NotesDatabase
Dim agent2 As NotesAgent
Dim doc As NotesDocument
Set db = s.CurrentDatabase
Set agent2 = db.GetAgent("bestellAkteInAgorumPflegen")
Set doc = Source.Document
If agent.Run(doc.NoteID) = 0 Then
Print "Agorum-Agent erfolgreich gelaufen"
Else
Messagebox "Agorum-Agent konnte nicht gestartet werden",64, "Bitte Ihrem Notes Admin melden"
End If
Ich habe das jetzt mal so runter geschrieben, also ohne Gewähr ...
Sub PostSave(Source As NotesUIDocument)
Dim doc As NotesDocument
Dim db As NotesDatabase
Dim agent As NotesAgent
Dim ret
Set doc = Source.Document
Set db = doc.ParentDatabase
Set agent = db.GetAgent("<agentname>")
'// der Agent sollte im Feld _retValue einen Returncode zurückliefern ...
doc.ReplaceItemValue("_retValue", "").SaveToDisk = False
ret = agent.runWithDocumentContext(doc)
If ret <> 0 Then Error 32000, "Allgemeiner Fehler beim Aufruf des Agenten <agentname>"
ret = doc.GetItemValue("_retValue")(0)
Select Case ret
Case ...
...
Call doc.Save(True, True)
...
End Select
End Sub
Im Java-Agenten muss man dann über AgentContext.getDocumentContext() auf das Dokument zugreifen.
Hoffe, das hilft weiter?
-Werner