| 
 | ||||||||||
| PREV CLASS NEXT CLASS | FRAMES NO FRAMES | |||||||||
| SUMMARY: NESTED | FIELD | CONSTR | METHOD | DETAIL: FIELD | CONSTR | METHOD | |||||||||
java.lang.Objectorg.apache.hadoop.hbase.zookeeper.ZKUtil
@InterfaceAudience.Public @InterfaceStability.Evolving public class ZKUtil
Internal HBase utility class for ZooKeeper.
Contains only static methods and constants.
Methods all throw KeeperException if there is an unexpected
 zookeeper exception, so callers of these methods must handle appropriately.
 If ZK is required for the operation, the server will need to be aborted.
| Nested Class Summary | |
|---|---|
| static class | ZKUtil.NodeAndDataDeprecated. Unused | 
| static class | ZKUtil.ZKUtilOpRepresents an action taken by ZKUtil, e.g. | 
| Field Summary | |
|---|---|
| static char | ZNODE_PATH_SEPARATOR | 
| Constructor Summary | |
|---|---|
| ZKUtil() | |
| Method Summary | |
|---|---|
| static void | applyClusterKeyToConf(org.apache.hadoop.conf.Configuration conf,
                      String key)Apply the settings in the given key to the given configuration, this is used to communicate with distant clusters | 
| static void | asyncCreate(ZooKeeperWatcher zkw,
            String znode,
            byte[] data,
            org.apache.zookeeper.AsyncCallback.StringCallback cb,
            Object ctx)Async creates the specified node with the specified data. | 
| static byte[] | blockUntilAvailable(ZooKeeperWatcher zkw,
                    String znode,
                    long timeout) | 
| static int | checkExists(ZooKeeperWatcher zkw,
            String znode)Check if the specified node exists. | 
| static RecoverableZooKeeper | connect(org.apache.hadoop.conf.Configuration conf,
        String ensemble,
        org.apache.zookeeper.Watcher watcher) | 
| static RecoverableZooKeeper | connect(org.apache.hadoop.conf.Configuration conf,
        String ensemble,
        org.apache.zookeeper.Watcher watcher,
        String identifier) | 
| static RecoverableZooKeeper | connect(org.apache.hadoop.conf.Configuration conf,
        org.apache.zookeeper.Watcher watcher)Creates a new connection to ZooKeeper, pulling settings and ensemble config from the specified configuration object using methods from ZKConfig. | 
| static org.apache.zookeeper.KeeperException | convert(DeserializationException e)Convert a DeserializationExceptionto a more palatableKeeperException. | 
| static void | createAndFailSilent(ZooKeeperWatcher zkw,
                    String znode)Creates the specified node, iff the node does not exist. | 
| static void | createAndFailSilent(ZooKeeperWatcher zkw,
                    String znode,
                    byte[] data)Creates the specified node containing specified data, iff the node does not exist. | 
| static int | createAndWatch(ZooKeeperWatcher zkw,
               String znode,
               byte[] data)Creates the specified node with the specified data and watches it. | 
| static boolean | createEphemeralNodeAndWatch(ZooKeeperWatcher zkw,
                            String znode,
                            byte[] data)Set the specified znode to be an ephemeral node carrying the specified data. | 
| static boolean | createNodeIfNotExistsAndWatch(ZooKeeperWatcher zkw,
                              String znode,
                              byte[] data)Creates the specified znode to be a persistent node carrying the specified data. | 
| static String | createNodeIfNotExistsNoWatch(ZooKeeperWatcher zkw,
                             String znode,
                             byte[] data,
                             org.apache.zookeeper.CreateMode createMode)Creates the specified znode with the specified data but does not watch it. | 
| static void | createSetData(ZooKeeperWatcher zkw,
              String znode,
              byte[] data)Set data into node creating node if it doesn't yet exist. | 
| static void | createWithParents(ZooKeeperWatcher zkw,
                  String znode)Creates the specified node and all parent nodes required for it to exist. | 
| static void | createWithParents(ZooKeeperWatcher zkw,
                  String znode,
                  byte[] data)Creates the specified node and all parent nodes required for it to exist. | 
| static void | deleteChildrenRecursively(ZooKeeperWatcher zkw,
                          String node)Delete all the children of the specified node but not the node itself. | 
| static void | deleteNode(ZooKeeperWatcher zkw,
           String node)Delete the specified node. | 
| static boolean | deleteNode(ZooKeeperWatcher zkw,
           String node,
           int version)Delete the specified node with the specified version. | 
| static void | deleteNodeFailSilent(ZooKeeperWatcher zkw,
                     String node)Deletes the specified node. | 
| static void | deleteNodeRecursively(ZooKeeperWatcher zkw,
                      String node)Delete the specified node and all of it's children. | 
| static String | dump(ZooKeeperWatcher zkw) | 
| static List<ZKUtil.NodeAndData> | getChildDataAndWatchForNewChildren(ZooKeeperWatcher zkw,
                                   String baseNode)Deprecated. Unused | 
| static byte[] | getData(ZooKeeperWatcher zkw,
        String znode)Get znode data. | 
| static byte[] | getDataAndWatch(ZooKeeperWatcher zkw,
                String znode)Get the data at the specified znode and set a watch. | 
| static byte[] | getDataAndWatch(ZooKeeperWatcher zkw,
                String znode,
                org.apache.zookeeper.data.Stat stat)Get the data at the specified znode and set a watch. | 
| static byte[] | getDataNoWatch(ZooKeeperWatcher zkw,
               String znode,
               org.apache.zookeeper.data.Stat stat)Get the data at the specified znode without setting a watch. | 
| static String | getNodeName(String path)Get the name of the current node from the specified fully-qualified path. | 
| static int | getNumberOfChildren(ZooKeeperWatcher zkw,
                    String znode)Get the number of children of the specified node. | 
| static String | getParent(String node)Returns the full path of the immediate parent of the specified node. | 
| static String[] | getServerStats(String server,
               int timeout)Gets the statistics from the given server. | 
| static String | getZooKeeperClusterKey(org.apache.hadoop.conf.Configuration conf)Get the key to the ZK ensemble for this configuration without adding a name at the end | 
| static String | getZooKeeperClusterKey(org.apache.hadoop.conf.Configuration conf,
                       String name)Get the key to the ZK ensemble for this configuration and append a name at the end | 
| static boolean | isSecureZooKeeper(org.apache.hadoop.conf.Configuration conf)Returns whether or not secure authentication is enabled (whether hbase.security.authenticationis set tokerberos. | 
| static String | joinZNode(String prefix,
          String suffix)Join the prefix znode name with the suffix znode name to generate a proper full znode name. | 
| static List<String> | listChildrenAndWatchForNewChildren(ZooKeeperWatcher zkw,
                                   String znode)Lists the children znodes of the specified znode. | 
| static List<String> | listChildrenAndWatchThem(ZooKeeperWatcher zkw,
                         String znode)List all the children of the specified znode, setting a watch for children changes and also setting a watch on every individual child in order to get the NodeCreated and NodeDeleted events. | 
| static List<String> | listChildrenNoWatch(ZooKeeperWatcher zkw,
                    String znode)Lists the children of the specified znode without setting any watches. | 
| static void | loginClient(org.apache.hadoop.conf.Configuration conf,
            String keytabFileKey,
            String userNameKey,
            String hostname)Log in the current zookeeper client using the given configuration keys for the credential file and login principal. | 
| static void | loginServer(org.apache.hadoop.conf.Configuration conf,
            String keytabFileKey,
            String userNameKey,
            String hostname)Log in the current zookeeper server process using the given configuration keys for the credential file and login principal. | 
| static void | logZKTree(ZooKeeperWatcher zkw,
          String root)Recursively print the current state of ZK (non-transactional) | 
| protected static void | logZKTree(ZooKeeperWatcher zkw,
          String root,
          String prefix)Helper method to print the current state of the ZK tree. | 
| static void | multiOrSequential(ZooKeeperWatcher zkw,
                  List<ZKUtil.ZKUtilOp> ops,
                  boolean runSequentialOnMultiFailure)If hbase.zookeeper.useMulti is true, use ZooKeeper's multi-update functionality. | 
| static boolean | nodeHasChildren(ZooKeeperWatcher zkw,
                String znode)Checks if the specified znode has any children. | 
| static long | parseHLogPositionFrom(byte[] bytes) | 
| static ZooKeeperProtos.RegionStoreSequenceIds | parseRegionStoreSequenceIds(byte[] bytes) | 
| static byte[] | positionToByteArray(long position) | 
| static byte[] | regionSequenceIdsToByteArray(Long regionLastFlushedSequenceId,
                             Map<byte[],Long> storeSequenceIds) | 
| static void | setData(ZooKeeperWatcher zkw,
        String znode,
        byte[] data)Sets the data of the existing znode to be the specified data. | 
| static boolean | setData(ZooKeeperWatcher zkw,
        String znode,
        byte[] data,
        int expectedVersion)Sets the data of the existing znode to be the specified data. | 
| static boolean | setWatchIfNodeExists(ZooKeeperWatcher zkw,
                     String znode)Watch the specified znode, but only if exists. | 
| static String[] | transformClusterKey(String key)Separate the given key into the three configurations it should contain: hbase.zookeeper.quorum, hbase.zookeeper.client.port and zookeeper.znode.parent | 
| static void | updateExistingNodeData(ZooKeeperWatcher zkw,
                       String znode,
                       byte[] data,
                       int expectedVersion)Deprecated. Unused | 
| static void | waitForBaseZNode(org.apache.hadoop.conf.Configuration conf)Waits for HBase installation's base (parent) znode to become available. | 
| static boolean | watchAndCheckExists(ZooKeeperWatcher zkw,
                    String znode)Watch the specified znode for delete/create/change events. | 
| Methods inherited from class java.lang.Object | 
|---|
| clone, equals, finalize, getClass, hashCode, notify, notifyAll, toString, wait, wait, wait | 
| Field Detail | 
|---|
public static final char ZNODE_PATH_SEPARATOR
| Constructor Detail | 
|---|
public ZKUtil()
| Method Detail | 
|---|
public static RecoverableZooKeeper connect(org.apache.hadoop.conf.Configuration conf,
                                           org.apache.zookeeper.Watcher watcher)
                                    throws IOException
ZKConfig.
 Sets the connection status monitoring watcher to the specified watcher.
conf - configuration to pull ensemble and other settings fromwatcher - watcher to monitor connection changes
IOException - if unable to connect to zk or config problem
public static RecoverableZooKeeper connect(org.apache.hadoop.conf.Configuration conf,
                                           String ensemble,
                                           org.apache.zookeeper.Watcher watcher)
                                    throws IOException
IOException
public static RecoverableZooKeeper connect(org.apache.hadoop.conf.Configuration conf,
                                           String ensemble,
                                           org.apache.zookeeper.Watcher watcher,
                                           String identifier)
                                    throws IOException
IOException
public static void loginServer(org.apache.hadoop.conf.Configuration conf,
                               String keytabFileKey,
                               String userNameKey,
                               String hostname)
                        throws IOException
This is only applicable when running on secure hbase On regular HBase (without security features), this will safely be ignored.
conf - The configuration data to usekeytabFileKey - Property key used to configure the path to the credential fileuserNameKey - Property key used to configure the login principalhostname - Current hostname to use in any credentials
IOException - underlying exception from SecurityUtil.login() call
public static void loginClient(org.apache.hadoop.conf.Configuration conf,
                               String keytabFileKey,
                               String userNameKey,
                               String hostname)
                        throws IOException
This is only applicable when running on secure hbase On regular HBase (without security features), this will safely be ignored.
conf - The configuration data to usekeytabFileKey - Property key used to configure the path to the credential fileuserNameKey - Property key used to configure the login principalhostname - Current hostname to use in any credentials
IOException - underlying exception from SecurityUtil.login() call
public static String joinZNode(String prefix,
                               String suffix)
prefix - beginning of znode namesuffix - ending of znode name
public static String getParent(String node)
node - path to get parent of
public static String getNodeName(String path)
path - fully-qualified path
public static String getZooKeeperClusterKey(org.apache.hadoop.conf.Configuration conf)
conf - Configuration to use to build the key
public static String getZooKeeperClusterKey(org.apache.hadoop.conf.Configuration conf,
                                            String name)
conf - Configuration to use to build the keyname - Name that should be appended at the end if not empty or null
public static void applyClusterKeyToConf(org.apache.hadoop.conf.Configuration conf,
                                         String key)
                                  throws IOException
conf - configuration object to configurekey - string that contains the 3 required configuratins
IOException
public static String[] transformClusterKey(String key)
                                    throws IOException
key - 
IOException
public static boolean watchAndCheckExists(ZooKeeperWatcher zkw,
                                          String znode)
                                   throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to watch
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static boolean setWatchIfNodeExists(ZooKeeperWatcher zkw,
                                           String znode)
                                    throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to watch
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static int checkExists(ZooKeeperWatcher zkw,
                              String znode)
                       throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to watch
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static List<String> listChildrenAndWatchForNewChildren(ZooKeeperWatcher zkw,
                                                              String znode)
                                                       throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to list and watch children of
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static List<String> listChildrenAndWatchThem(ZooKeeperWatcher zkw,
                                                    String znode)
                                             throws org.apache.zookeeper.KeeperException
zkw - zookeeper referenceznode - node to get children of and watch
org.apache.zookeeper.KeeperException
public static List<String> listChildrenNoWatch(ZooKeeperWatcher zkw,
                                               String znode)
                                        throws org.apache.zookeeper.KeeperException
zkw - zookeeper referenceznode - node to get children
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static boolean nodeHasChildren(ZooKeeperWatcher zkw,
                                      String znode)
                               throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to check for children of
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static int getNumberOfChildren(ZooKeeperWatcher zkw,
                                      String znode)
                               throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node to count children of
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static byte[] getData(ZooKeeperWatcher zkw,
                             String znode)
                      throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperException
public static byte[] getDataAndWatch(ZooKeeperWatcher zkw,
                                     String znode)
                              throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static byte[] getDataAndWatch(ZooKeeperWatcher zkw,
                                     String znode,
                                     org.apache.zookeeper.data.Stat stat)
                              throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodestat - object to populate the version of the znode
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static byte[] getDataNoWatch(ZooKeeperWatcher zkw,
                                    String znode,
                                    org.apache.zookeeper.data.Stat stat)
                             throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodestat - node status to get if node exists
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static List<ZKUtil.NodeAndData> getChildDataAndWatchForNewChildren(ZooKeeperWatcher zkw,
                                                                          String baseNode)
                                                                   throws org.apache.zookeeper.KeeperException
zkw - zk referencebaseNode - path of node to list and watch children of
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static void updateExistingNodeData(ZooKeeperWatcher zkw,
                                          String znode,
                                          byte[] data,
                                          int expectedVersion)
                                   throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - data - expectedVersion - 
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
KeeperException.BadVersionException - if version mismatch
public static boolean setData(ZooKeeperWatcher zkw,
                              String znode,
                              byte[] data,
                              int expectedVersion)
                       throws org.apache.zookeeper.KeeperException,
                              org.apache.zookeeper.KeeperException.NoNodeException
If the node does not exist, a KeeperException.NoNodeException will be thrown.
 
If their is a version mismatch, method returns null.
No watches are set but setting data will trigger other watchers of this node.
If there is another problem, a KeeperException will be thrown.
zkw - zk referenceznode - path of nodedata - data to set for nodeexpectedVersion - version expected when setting data
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
org.apache.zookeeper.KeeperException.NoNodeException
public static void createSetData(ZooKeeperWatcher zkw,
                                 String znode,
                                 byte[] data)
                          throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodedata - data to set for node
org.apache.zookeeper.KeeperException
public static void setData(ZooKeeperWatcher zkw,
                           String znode,
                           byte[] data)
                    throws org.apache.zookeeper.KeeperException,
                           org.apache.zookeeper.KeeperException.NoNodeException
If the node does not exist, a KeeperException.NoNodeException will be thrown.
 
No watches are set but setting data will trigger other watchers of this node.
If there is another problem, a KeeperException will be thrown.
zkw - zk referenceznode - path of nodedata - data to set for node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
org.apache.zookeeper.KeeperException.NoNodeExceptionpublic static boolean isSecureZooKeeper(org.apache.hadoop.conf.Configuration conf)
hbase.security.authentication is set to
 kerberos.
public static boolean createEphemeralNodeAndWatch(ZooKeeperWatcher zkw,
                                                  String znode,
                                                  byte[] data)
                                           throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodedata - data of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static boolean createNodeIfNotExistsAndWatch(ZooKeeperWatcher zkw,
                                                    String znode,
                                                    byte[] data)
                                             throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodedata - data of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static String createNodeIfNotExistsNoWatch(ZooKeeperWatcher zkw,
                                                  String znode,
                                                  byte[] data,
                                                  org.apache.zookeeper.CreateMode createMode)
                                           throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodedata - data of nodecreateMode - specifying whether the node to be created is ephemeral and/or sequential
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static int createAndWatch(ZooKeeperWatcher zkw,
                                 String znode,
                                 byte[] data)
                          throws org.apache.zookeeper.KeeperException,
                                 org.apache.zookeeper.KeeperException.NodeExistsException
Throws an exception if the node already exists.
The node created is persistent and open access.
Returns the version number of the created node if successful.
zkw - zk referenceznode - path of node to createdata - data of node to create
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
KeeperException.NodeExistsException - if node already exists
public static void asyncCreate(ZooKeeperWatcher zkw,
                               String znode,
                               byte[] data,
                               org.apache.zookeeper.AsyncCallback.StringCallback cb,
                               Object ctx)
Throws an exception if the node already exists.
The node created is persistent and open access.
zkw - zk referenceznode - path of node to createdata - data of node to createcb - ctx - 
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
KeeperException.NodeExistsException - if node already exists
public static void createAndFailSilent(ZooKeeperWatcher zkw,
                                       String znode)
                                throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static void createAndFailSilent(ZooKeeperWatcher zkw,
                                       String znode,
                                       byte[] data)
                                throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of nodedata - a byte array data to store in the znode
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static void createWithParents(ZooKeeperWatcher zkw,
                                     String znode)
                              throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static void createWithParents(ZooKeeperWatcher zkw,
                                     String znode,
                                     byte[] data)
                              throws org.apache.zookeeper.KeeperException
zkw - zk referenceznode - path of node
org.apache.zookeeper.KeeperException - if unexpected zookeeper exception
public static void deleteNode(ZooKeeperWatcher zkw,
                              String node)
                       throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperException
public static boolean deleteNode(ZooKeeperWatcher zkw,
                                 String node,
                                 int version)
                          throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperException
public static void deleteNodeFailSilent(ZooKeeperWatcher zkw,
                                        String node)
                                 throws org.apache.zookeeper.KeeperException
zkw - node - 
org.apache.zookeeper.KeeperException
public static void deleteNodeRecursively(ZooKeeperWatcher zkw,
                                         String node)
                                  throws org.apache.zookeeper.KeeperException
If the node does not exist, just returns.
Sets no watches. Throws all exceptions besides dealing with deletion of children.
org.apache.zookeeper.KeeperException
public static void deleteChildrenRecursively(ZooKeeperWatcher zkw,
                                             String node)
                                      throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperException
public static void multiOrSequential(ZooKeeperWatcher zkw,
                                     List<ZKUtil.ZKUtilOp> ops,
                                     boolean runSequentialOnMultiFailure)
                              throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperExceptionpublic static String dump(ZooKeeperWatcher zkw)
public static String[] getServerStats(String server,
                                      int timeout)
                               throws IOException
server - The server to get the statistics from.timeout - The socket timeout to use.
IOException - When the socket communication fails.
public static void waitForBaseZNode(org.apache.hadoop.conf.Configuration conf)
                             throws IOException
IOException - on ZK errors
public static byte[] blockUntilAvailable(ZooKeeperWatcher zkw,
                                         String znode,
                                         long timeout)
                                  throws InterruptedException
InterruptedExceptionpublic static org.apache.zookeeper.KeeperException convert(DeserializationException e)
DeserializationException to a more palatable KeeperException.
 Used when can't let a DeserializationException out w/o changing public API.
e - Exception to convert
public static void logZKTree(ZooKeeperWatcher zkw,
                             String root)
root - name of the root directory in zk to print
org.apache.zookeeper.KeeperException
protected static void logZKTree(ZooKeeperWatcher zkw,
                                String root,
                                String prefix)
                         throws org.apache.zookeeper.KeeperException
org.apache.zookeeper.KeeperException - if an unexpected exception occurslogZKTree(ZooKeeperWatcher, String)public static byte[] positionToByteArray(long position)
position - 
position with pb magic prefix prepended suitable
         for use as content of an hlog position in a replication queue.
public static long parseHLogPositionFrom(byte[] bytes)
                                  throws DeserializationException
bytes - - Content of a HLog position znode.
DeserializationException
public static byte[] regionSequenceIdsToByteArray(Long regionLastFlushedSequenceId,
                                                  Map<byte[],Long> storeSequenceIds)
regionLastFlushedSequenceId - the flushed sequence id of a region which is the min of its
          store max seq idsstoreSequenceIds - column family to sequence Id map
RegionSequenceIds with pb magic prefix prepended
         suitable for use to filter wal edits in distributedLogReplay mode
public static ZooKeeperProtos.RegionStoreSequenceIds parseRegionStoreSequenceIds(byte[] bytes)
                                                                          throws DeserializationException
bytes - Content of serialized data of RegionStoreSequenceIds
DeserializationException| 
 | ||||||||||
| PREV CLASS NEXT CLASS | FRAMES NO FRAMES | |||||||||
| SUMMARY: NESTED | FIELD | CONSTR | METHOD | DETAIL: FIELD | CONSTR | METHOD | |||||||||