Skip to content

Commit

Permalink
ZOOKEEPER-2693: DOS attack on wchp/wchc four letter words (4lw).
Browse files Browse the repository at this point in the history
Introduce a new configuration option 4lw.commands.whitelist that disables all 4lw (except srvr) by default to avoid the exploits reported on wchp/wchc commands.
  • Loading branch information
hanm committed Mar 2, 2017
1 parent fd211a5 commit b4c421d
Show file tree
Hide file tree
Showing 7 changed files with 453 additions and 15 deletions.
57 changes: 55 additions & 2 deletions src/docs/src/documentation/content/xdocs/zookeeperAdmin.xml
Original file line number Diff line number Diff line change
Expand Up @@ -1155,6 +1155,40 @@ server.3=zoo3:2888:3888</programlisting>
</listitem>
</varlistentry>

<varlistentry>
<term>4lw.commands.whitelist</term>

<listitem>
<para>(Java system property: <emphasis
role="bold">zookeeper.4lw.commands.whitelist</emphasis>)</para>

<para><emphasis role="bold">New in 3.5.3:</emphasis>
A list of comma separated <ulink url="#sc_4lw">Four Letter Words</ulink>
commands that user wants to use. A valid Four Letter Words
command must be put in this list else ZooKeeper server will
not enable the command.
By default the whitelist only contains "srvr" command
which zkServer.sh uses. The rest of four letter word commands are disabled
by default.
</para>

<para>Here's an example of the configuration that enables stat, ruok, conf, and isro
command while disabling the rest of Four Letter Words command:</para>
<programlisting>
4lw.commands.whitelist=stat, ruok, conf, isro
</programlisting>

<para>If you really need enable all four letter word commands by default, you can use
the asterisk option so you don't have to include every command one by one in the list.
As an example, this will enable all four letter word commands:
</para>
<programlisting>
4lw.commands.whitelist=*
</programlisting>

</listitem>
</varlistentry>

</variablelist>
<para></para>
</section>
Expand Down Expand Up @@ -1639,7 +1673,7 @@ server.3=zoo3:2888:3888</programlisting>
<section id="sc_zkCommands">
<title>ZooKeeper Commands</title>

<section>
<section id="sc_4lw">
<title>The Four Letter Words</title>
<para>ZooKeeper responds to a small set of commands. Each command is
composed of four letters. You issue the commands to ZooKeeper via telnet
Expand All @@ -1650,7 +1684,16 @@ server.3=zoo3:2888:3888</programlisting>
while "srvr" and "cons" give extended details on server and
connections respectively.</para>

<variablelist>
<para><emphasis role="bold">New in 3.5.3:</emphasis>
Four Letter Words need to be explicitly white listed before using.
Please refer <emphasis role="bold">4lw.commands.whitelist</emphasis>
described in <ulink url="#sc_clusterOptions">
cluster configuration section</ulink> for details.
Moving forward, Four Letter Words will be deprecated, please use
<ulink url="#sc_adminserver">AdminServer</ulink> instead.
</para>

<variablelist>
<varlistentry>
<term>conf</term>

Expand Down Expand Up @@ -2125,6 +2168,16 @@ server.3=zoo3:2888:3888</programlisting>
usage limit that would cause the system to swap.</para>
</listitem>
</varlistentry>

<varlistentry>
<term>Publicly accessible deployment</term>
<listitem>
<para>
A ZooKeeper ensemble is expected to operate in a trusted computing environment.
It is thus recommended to deploy ZooKeeper behind a firewall.
</para>
</listitem>
</varlistentry>
</variablelist>
</section>

Expand Down
21 changes: 17 additions & 4 deletions src/java/main/org/apache/zookeeper/server/NIOServerCnxn.java
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@
import org.apache.zookeeper.server.command.CommandExecutor;
import org.apache.zookeeper.server.command.FourLetterCommands;
import org.apache.zookeeper.server.command.SetTraceMaskCommand;
import org.apache.zookeeper.server.command.NopCommand;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

Expand Down Expand Up @@ -478,12 +479,11 @@ private boolean checkFourLetterWord(final SelectionKey k, final int len)
{
// We take advantage of the limited size of the length to look
// for cmds. They are all 4-bytes which fits inside of an int
String cmd = FourLetterCommands.getCmdMapView().get(len);
if (cmd == null) {
if (!FourLetterCommands.isKnown(len)) {
return false;
}
LOG.info("Processing " + cmd + " command from "
+ sock.socket().getRemoteSocketAddress());

String cmd = FourLetterCommands.getCommandString(len);
packetReceived();

/** cancel the selection key to remove the socket handling
Expand All @@ -505,6 +505,19 @@ private boolean checkFourLetterWord(final SelectionKey k, final int len)

final PrintWriter pwriter = new PrintWriter(
new BufferedWriter(new SendBufferWriter()));

// ZOOKEEPER-2693: don't execute 4lw if it's not enabled.
if (!FourLetterCommands.isEnabled(cmd)) {
LOG.debug("Command {} is not executed because it is not in the whitelist.", cmd);
NopCommand nopCmd = new NopCommand(pwriter, this, cmd +
" is not executed because it is not in the whitelist.");
nopCmd.start();
return true;
}

LOG.info("Processing " + cmd + " command from "
+ sock.socket().getRemoteSocketAddress());

if (len == FourLetterCommands.setTraceMaskCmd) {
incomingBuffer = ByteBuffer.allocate(8);
int rc = sock.read(incomingBuffer);
Expand Down
22 changes: 18 additions & 4 deletions src/java/main/org/apache/zookeeper/server/NettyServerCnxn.java
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@
import org.apache.zookeeper.proto.WatcherEvent;
import org.apache.zookeeper.server.command.CommandExecutor;
import org.apache.zookeeper.server.command.FourLetterCommands;
import org.apache.zookeeper.server.command.NopCommand;
import org.apache.zookeeper.server.command.SetTraceMaskCommand;
import org.jboss.netty.buffer.ChannelBuffer;
import org.jboss.netty.buffer.ChannelBuffers;
Expand Down Expand Up @@ -267,17 +268,30 @@ private boolean checkFourLetterWord(final Channel channel,
{
// We take advantage of the limited size of the length to look
// for cmds. They are all 4-bytes which fits inside of an int
String cmd = FourLetterCommands.getCmdMapView().get(len);
if (cmd == null) {
if (!FourLetterCommands.isKnown(len)) {
return false;
}

String cmd = FourLetterCommands.getCommandString(len);

channel.setInterestOps(0).awaitUninterruptibly();
LOG.info("Processing " + cmd + " command from "
+ channel.getRemoteAddress());
packetReceived();

final PrintWriter pwriter = new PrintWriter(
new BufferedWriter(new SendBufferWriter()));

// ZOOKEEPER-2693: don't execute 4lw if it's not enabled.
if (!FourLetterCommands.isEnabled(cmd)) {
LOG.debug("Command {} is not executed because it is not in the whitelist.", cmd);
NopCommand nopCmd = new NopCommand(pwriter, this, cmd +
" is not executed because it is not in the whitelist.");
nopCmd.start();
return true;
}

LOG.info("Processing " + cmd + " command from "
+ channel.getRemoteAddress());

if (len == FourLetterCommands.setTraceMaskCmd) {
ByteBuffer mask = ByteBuffer.allocate(8);
message.readBytes(mask);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,15 @@

package org.apache.zookeeper.server.command;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.nio.ByteBuffer;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.HashSet;
import java.util.Set;
import java.util.Arrays;

/**
* This class contains constants for all the four letter commands
Expand Down Expand Up @@ -153,11 +158,82 @@ public class FourLetterCommands {
*/
public final static int telnetCloseCmd = 0xfff4fffd;

final static HashMap<Integer, String> cmd2String =
new HashMap<Integer, String>();
private static final String ZOOKEEPER_4LW_COMMANDS_WHITELIST = "zookeeper.4lw.commands.whitelist";

private static final Logger LOG = LoggerFactory.getLogger(FourLetterCommands.class);

private static final Map<Integer, String> cmd2String = new HashMap<Integer, String>();

private static final Set<String> whiteListedCommands = new HashSet<String>();

private static boolean whiteListInitialized = false;

// @VisibleForTesting
public static void resetWhiteList() {
whiteListInitialized = false;
whiteListedCommands.clear();
}

/**
* Return the string representation of the specified command code.
*/
public static String getCommandString(int command) {
return cmd2String.get(command);
}

/**
* Check if the specified command code is from a known command.
*
* @param command The integer code of command.
* @return true if the specified command is known, false otherwise.
*/
public static boolean isKnown(int command) {
return cmd2String.containsKey(command);
}

/**
* Check if the specified command is enabled.
*
* In ZOOKEEPER-2693 we introduce a configuration option to only
* allow a specific set of white listed commands to execute.
* A command will only be executed if it is also configured
* in the white list.
*
* @param command The command string.
* @return true if the specified command is enabled
*/
public static boolean isEnabled(String command) {
if (whiteListInitialized) {
return whiteListedCommands.contains(command);
}

String commands = System.getProperty(ZOOKEEPER_4LW_COMMANDS_WHITELIST);
if (commands != null) {
String[] list = commands.split(",");
for (String cmd : list) {
if (cmd.trim().equals("*")) {
for (Map.Entry<Integer, String> entry : cmd2String.entrySet()) {
whiteListedCommands.add(entry.getValue());
}
break;
}
if (!cmd.trim().isEmpty()) {
whiteListedCommands.add(cmd.trim());
}
}
}

public static Map<Integer, String> getCmdMapView() {
return Collections.unmodifiableMap(cmd2String);
// It is sad that isro and srvr are used by ZooKeeper itself. Need fix this
// before deprecating 4lw.
if (System.getProperty("readonlymode.enabled", "false").equals("true")) {
whiteListedCommands.add("isro");
}
// zkServer.sh depends on "srvr".
whiteListedCommands.add("srvr");
whiteListInitialized = true;
LOG.info("The list of known four letter word commands is : {}", Arrays.asList(cmd2String));
LOG.info("The list of enabled four letter word commands is : {}", Arrays.asList(whiteListedCommands));
return whiteListedCommands.contains(command);
}

// specify all of the commands that are available
Expand Down
41 changes: 41 additions & 0 deletions src/java/main/org/apache/zookeeper/server/command/NopCommand.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package org.apache.zookeeper.server.command;

import java.io.PrintWriter;

import org.apache.zookeeper.server.ServerCnxn;

/**
* A command that does not do anything except reply to client with predefined message.
* It is used to inform clients who execute none white listed four letter word commands.
*/
public class NopCommand extends AbstractFourLetterCommand {
private String msg;

public NopCommand(PrintWriter pw, ServerCnxn serverCnxn, String msg) {
super(pw, serverCnxn);
this.msg = msg;
}

@Override
public void commandRun() {
pw.println(msg);
}
}
6 changes: 6 additions & 0 deletions src/java/test/org/apache/zookeeper/ZKTestCase.java
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,12 @@ public void starting(FrameworkMethod method) {
// accidentally attempting to start multiple admin servers on the
// same port.
System.setProperty("zookeeper.admin.enableServer", "false");
// ZOOKEEPER-2693 disables all 4lw by default.
// Here we enable the 4lw which ZooKeeper tests depends.
System.setProperty("zookeeper.4lw.commands.whitelist",
"ruok, envi, conf, stat, srvr, cons, dump," +
"wchs, wchp, wchc, srst, crst, " +
"dirs, mntr, gtmk, isro, stmk");
testName = method.getName();
LOG.info("STARTING " + testName);
}
Expand Down
Loading

0 comments on commit b4c421d

Please sign in to comment.