/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at * trunk/opends/resource/legal-notices/OpenDS.LICENSE * or https://OpenDS.dev.java.net/OpenDS.LICENSE. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at * trunk/opends/resource/legal-notices/OpenDS.LICENSE. If applicable, * add the following below this CDDL HEADER, with the fields enclosed * by brackets "[]" replaced with your own identifying information: * Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END * * * Copyright 2008 Sun Microsystems, Inc. */ package org.opends.server.replication.protocol; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.util.ArrayList; import java.util.List; import java.util.zip.DataFormatException; import org.opends.server.replication.common.AssuredMode; import org.opends.server.replication.common.ServerStatus; /** * This message is used by DS to confirm a RS he wants to connect to him (open * a session): * Handshake sequence between DS and RS is like this: * DS --- ServerStartMsg ---> RS * DS <--- ReplServerStartMsg --- RS * DS --- StartSessionMsg ---> RS * DS <--- TopologyMsg --- RS * * This message contains: * - status: the status we are entering the topology with * - referrals URLs: the referrals URLs we allow peer DSs to use to refer to * our domain when needed. */ public class StartSessionMsg extends ReplicationMsg { // The list of referrals URLs to the sending DS private List referralsURLs = new ArrayList(); // The initial status the DS starts with private ServerStatus status = ServerStatus.INVALID_STATUS; // Assured replication enabled on DS or not private boolean assuredFlag = false; // DS assured mode (relevant if assured replication enabled) private AssuredMode assuredMode = AssuredMode.SAFE_DATA_MODE; // DS safe data level (relevant if assured mode is safe data) private byte safeDataLevel = (byte) 1; /** * Creates a new StartSessionMsg message from its encoded form. * * @param in The byte array containing the encoded form of the message. * @throws java.util.zip.DataFormatException If the byte array does not * contain a valid encoded form of the message. */ public StartSessionMsg(byte[] in) throws DataFormatException { /* * The message is stored in the form: * * * (each referral url terminates with 0) */ try { /* first byte is the type */ if (in.length < 1 || in[0] != MSG_TYPE_START_SESSION) { throw new DataFormatException( "Input is not a valid " + this.getClass().getCanonicalName()); } /* Read the status */ status = ServerStatus.valueOf(in[1]); /* Read the assured flag */ if (in[2] == 1) { assuredFlag = true; } else { assuredFlag = false; } /* Read the assured mode */ assuredMode = AssuredMode.valueOf(in[3]); /* Read the safe data level */ safeDataLevel = in[4]; /* Read the refferals URLs */ int pos = 5; referralsURLs = new ArrayList(); while (pos < in.length) { /* * Read the next URL * first calculate the length then construct the string */ int length = getNextLength(in, pos); referralsURLs.add(new String(in, pos, length, "UTF-8")); pos += length + 1; } } catch (UnsupportedEncodingException e) { throw new DataFormatException("UTF-8 is not supported by this jvm."); } catch (IllegalArgumentException e) { throw new DataFormatException(e.getMessage()); } } /** * Creates a new StartSessionMsg message with the given required parameters. * @param status Status we are starting with * @param referralsURLs Referrals URLs to be used by peer DSs * @param assuredFlag If assured mode is enabled or not * @param assuredMode Assured type * @param safeDataLevel Assured mode safe data level */ public StartSessionMsg(ServerStatus status, List referralsURLs, boolean assuredFlag, AssuredMode assuredMode, byte safeDataLevel) { this.referralsURLs = referralsURLs; this.status = status; this.assuredFlag = assuredFlag; this.assuredMode = assuredMode; this.safeDataLevel = safeDataLevel; } /** * Creates a new StartSessionMsg message with the given required parameters. * Assured mode is false. * @param status Status we are starting with * @param referralsURLs Referrals URLs to be used by peer DSs */ public StartSessionMsg(ServerStatus status, List referralsURLs) { this.referralsURLs = referralsURLs; this.status = status; this.assuredFlag = false; } /** * {@inheritDoc} */ @Override public byte[] getBytes() { /* * The message is stored in the form: * * * (each referral url terminates with 0) */ try { ByteArrayOutputStream oStream = new ByteArrayOutputStream(); /* Put the message type */ oStream.write(MSG_TYPE_START_SESSION); // Put the status oStream.write(status.getValue()); // Put the assured flag oStream.write(assuredFlag ? (byte) 1 : (byte) 0); // Put assured mode oStream.write(assuredMode.getValue()); // Put safe data level oStream.write(safeDataLevel); // Put the referrals URLs if (referralsURLs.size() >= 1) { for (String url : referralsURLs) { byte[] byteArrayURL = url.getBytes("UTF-8"); oStream.write(byteArrayURL); oStream.write(0); } } return oStream.toByteArray(); } catch (IOException e) { // never happens return null; } } /** * Get the list of referrals URLs. * * @return The list of referrals URLs. */ public List getReferralsURLs() { return referralsURLs; } /** * Get the status from this message. * @return The status. */ public ServerStatus getStatus() { return status; } /** * {@inheritDoc} */ @Override public String toString() { String urls = ""; for (String s : referralsURLs) { urls += s + " | "; } return ("StartSessionMsg content:\nstatus: " + status + "\nassuredFlag: " + assuredFlag + "\nassuredMode: " + assuredMode + "\nsafeDataLevel: " + safeDataLevel + "\nreferralsURLs: " + urls); } /** * Returns true if assured mode is enabled. * @return true if assured mode is enabled. */ public boolean isAssured() { return assuredFlag; } /** * Get the assured mode. * @return the assured mode. */ public AssuredMode getAssuredMode() { return assuredMode; } /** * Get the safe data level. * @return the safe data level. */ public byte getSafeDataLevel() { return safeDataLevel; } }