Skip to content
Snippets Groups Projects
Commit d159be19 authored by srosse's avatar srosse
Browse files

OO-3352: better validation

parent 6304083f
No related branches found
No related tags found
No related merge requests found
...@@ -377,10 +377,13 @@ public class RestApiLoginFilter implements Filter { ...@@ -377,10 +377,13 @@ public class RestApiLoginFilter implements Filter {
Identity identity = securityBean.getIdentity(token); Identity identity = securityBean.getIdentity(token);
int loginStatus = AuthHelper.doHeadlessLogin(identity, BaseSecurityModule.getDefaultAuthProviderIdentifier(), ureq, true); int loginStatus = AuthHelper.doHeadlessLogin(identity, BaseSecurityModule.getDefaultAuthProviderIdentifier(), ureq, true);
if(loginStatus == AuthHelper.LOGIN_OK) { if(loginStatus == AuthHelper.LOGIN_OK) {
response.setHeader(RestSecurityHelper.SEC_TOKEN, securityBean.renewToken(token)); String renewedToken = securityBean.renewToken(token);
synchronized(uress) { if(renewedToken != null) {
chain.doFilter(request, response); response.setHeader(RestSecurityHelper.SEC_TOKEN, renewedToken);
} synchronized(uress) {
chain.doFilter(request, response);
}
} else response.sendError(401);
} else response.sendError(401); } else response.sendError(401);
} else response.sendError(401); } else response.sendError(401);
} }
......
...@@ -53,9 +53,9 @@ public class RestSecurityBeanImpl implements RestSecurityBean { ...@@ -53,9 +53,9 @@ public class RestSecurityBeanImpl implements RestSecurityBean {
public static final String REST_AUTH_PROVIDER = "REST"; public static final String REST_AUTH_PROVIDER = "REST";
private Map<String,Long> tokenToIdentity = new ConcurrentHashMap<String,Long>(); private Map<String,Long> tokenToIdentity = new ConcurrentHashMap<>();
private Map<String,List<String>> tokenToSessionIds = new ConcurrentHashMap<String,List<String>>(); private Map<String,List<String>> tokenToSessionIds = new ConcurrentHashMap<>();
private Map<String,String> sessionIdToTokens = new ConcurrentHashMap<String,String>(); private Map<String,String> sessionIdToTokens = new ConcurrentHashMap<>();
@Autowired @Autowired
private BaseSecurity securityManager; private BaseSecurity securityManager;
...@@ -70,7 +70,7 @@ public class RestSecurityBeanImpl implements RestSecurityBean { ...@@ -70,7 +70,7 @@ public class RestSecurityBeanImpl implements RestSecurityBean {
Authentication auth = securityManager.findAuthentication(identity, REST_AUTH_PROVIDER); Authentication auth = securityManager.findAuthentication(identity, REST_AUTH_PROVIDER);
if(auth == null) { if(auth == null) {
auth = securityManager.createAndPersistAuthentication(identity, REST_AUTH_PROVIDER, identity.getName(), token, null); securityManager.createAndPersistAuthentication(identity, REST_AUTH_PROVIDER, identity.getName(), token, null);
} else { } else {
authenticationDao.updateCredential(auth, token); authenticationDao.updateCredential(auth, token);
} }
...@@ -79,6 +79,19 @@ public class RestSecurityBeanImpl implements RestSecurityBean { ...@@ -79,6 +79,19 @@ public class RestSecurityBeanImpl implements RestSecurityBean {
@Override @Override
public String renewToken(String token) { public String renewToken(String token) {
if(token == null || token.length() > 40) {
return null;
}
// don't regex, never
for(char c:token.toCharArray()) {
if(c == '-'
|| (c >= 48 && c <= 57)
|| (c >= 65 && c <= 90)
|| (c >= 97 && c <= 122)) {
continue;
}
return null;
}
return token; return token;
} }
...@@ -119,7 +132,7 @@ public class RestSecurityBeanImpl implements RestSecurityBean { ...@@ -119,7 +132,7 @@ public class RestSecurityBeanImpl implements RestSecurityBean {
String sessionId = session.getId(); String sessionId = session.getId();
synchronized(tokenToSessionIds) {//cluster notOK -> need probably a mapping on the DB synchronized(tokenToSessionIds) {//cluster notOK -> need probably a mapping on the DB
if(!tokenToSessionIds.containsKey(token)) { if(!tokenToSessionIds.containsKey(token)) {
List<String> sessionIds = new ArrayList<String>(); List<String> sessionIds = new ArrayList<>();
sessionIds.add(session.getId()); sessionIds.add(session.getId());
tokenToSessionIds.put(token, sessionIds); tokenToSessionIds.put(token, sessionIds);
} else { } else {
......
...@@ -33,7 +33,6 @@ import java.util.List; ...@@ -33,7 +33,6 @@ import java.util.List;
import javax.ws.rs.core.MediaType; import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.UriBuilder; import javax.ws.rs.core.UriBuilder;
import org.apache.commons.io.IOUtils;
import org.apache.http.Header; import org.apache.http.Header;
import org.apache.http.HttpEntity; import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse; import org.apache.http.HttpResponse;
...@@ -44,6 +43,7 @@ import org.apache.http.client.CookieStore; ...@@ -44,6 +43,7 @@ import org.apache.http.client.CookieStore;
import org.apache.http.client.config.CookieSpecs; import org.apache.http.client.config.CookieSpecs;
import org.apache.http.client.config.RequestConfig; import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.entity.UrlEncodedFormEntity; import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpDelete; import org.apache.http.client.methods.HttpDelete;
import org.apache.http.client.methods.HttpEntityEnclosingRequestBase; import org.apache.http.client.methods.HttpEntityEnclosingRequestBase;
import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpGet;
...@@ -102,6 +102,17 @@ public class RestConnection { ...@@ -102,6 +102,17 @@ public class RestConnection {
.build(); .build();
} }
public RestConnection(boolean enableCookieStore, boolean enableCredentialProvider) {
HttpClientBuilder builder = HttpClientBuilder.create();
if(enableCookieStore) {
builder = builder.setDefaultCookieStore(cookieStore);
}
if(enableCredentialProvider) {
builder = builder.setDefaultCredentialsProvider(provider);
}
httpclient = builder.build();
}
public RestConnection(URL url) { public RestConnection(URL url) {
PORT = url.getPort(); PORT = url.getPort();
HOST = url.getHost(); HOST = url.getHost();
...@@ -151,7 +162,11 @@ public class RestConnection { ...@@ -151,7 +162,11 @@ public class RestConnection {
} }
public void shutdown() { public void shutdown() {
IOUtils.closeQuietly(httpclient); try {
httpclient.close();
} catch (IOException e) {
log.error("", e);
}
} }
public boolean login(String username, String password) throws IOException, URISyntaxException { public boolean login(String username, String password) throws IOException, URISyntaxException {
...@@ -161,17 +176,20 @@ public class RestConnection { ...@@ -161,17 +176,20 @@ public class RestConnection {
provider.setCredentials(new AuthScope(HOST, PORT), new UsernamePasswordCredentials(username, password)); provider.setCredentials(new AuthScope(HOST, PORT), new UsernamePasswordCredentials(username, password));
provider.setCredentials(new AuthScope(uri.getHost(), uri.getPort()), new UsernamePasswordCredentials(username, password)); provider.setCredentials(new AuthScope(uri.getHost(), uri.getPort()), new UsernamePasswordCredentials(username, password));
int code = -1;
HttpGet httpget = new HttpGet(uri); HttpGet httpget = new HttpGet(uri);
HttpResponse response = httpclient.execute(httpget); try(CloseableHttpResponse response = httpclient.execute(httpget)) {
Header header = response.getFirstHeader(RestSecurityHelper.SEC_TOKEN);
Header header = response.getFirstHeader(RestSecurityHelper.SEC_TOKEN); if(header != null) {
if(header != null) { securityToken = header.getValue();
securityToken = header.getValue(); }
HttpEntity entity = response.getEntity();
code = response.getStatusLine().getStatusCode();
EntityUtils.consume(entity);
} catch(IOException e) {
log.error("", e);
} }
HttpEntity entity = response.getEntity();
int code = response.getStatusLine().getStatusCode();
EntityUtils.consume(entity);
return code == 200; return code == 200;
} }
...@@ -310,8 +328,7 @@ public class RestConnection { ...@@ -310,8 +328,7 @@ public class RestConnection {
} }
public <U> U parse(HttpResponse response, Class<U> cl) { public <U> U parse(HttpResponse response, Class<U> cl) {
try { try(InputStream body = response.getEntity().getContent()) {
InputStream body = response.getEntity().getContent();
ObjectMapper mapper = new ObjectMapper(jsonFactory); ObjectMapper mapper = new ObjectMapper(jsonFactory);
U obj = mapper.readValue(body, cl); U obj = mapper.readValue(body, cl);
return obj; return obj;
......
/**
* <a href="http://www.openolat.org">
* OpenOLAT - Online Learning and Training</a><br>
* <p>
* Licensed under the Apache License, Version 2.0 (the "License"); <br>
* you may not use this file except in compliance with the License.<br>
* You may obtain a copy of the License at the
* <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a>
* <p>
* Unless required by applicable law or agreed to in writing,<br>
* software distributed under the License is distributed on an "AS IS" BASIS, <br>
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br>
* See the License for the specific language governing permissions and <br>
* limitations under the License.
* <p>
* Initial code contributed and copyrighted by<br>
* frentix GmbH, http://www.frentix.com
* <p>
*/
package org.olat.restapi.security;
import java.util.UUID;
import org.junit.Assert;
import org.junit.Test;
import org.olat.core.id.Identity;
import org.olat.test.JunitTestHelper;
import org.olat.test.OlatTestCase;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.mock.web.MockHttpSession;
/**
*
* Initial date: 5 mars 2018<br>
* @author srosse, stephane.rosse@frentix.com, http://www.frentix.com
*
*/
public class RestSecurityBeanTest extends OlatTestCase {
@Autowired
private RestSecurityBean restSecurityBean;
@Test
public void generatedToken() {
Identity id = JunitTestHelper.createAndPersistIdentityAsRndUser("rest-api");
String token = restSecurityBean.generateToken(id, new MockHttpSession());
String renewedToken = restSecurityBean.renewToken(token);
Assert.assertEquals(token, renewedToken);
}
@Test
public void renewToken() {
String token = UUID.randomUUID().toString();
String renewedToken = restSecurityBean.renewToken(token);
Assert.assertEquals(token, renewedToken);
}
@Test
public void renewToken_09() {
String uuid = "0123456789";
String renewedToken = restSecurityBean.renewToken(uuid);
Assert.assertEquals(uuid, renewedToken);
}
@Test
public void renewToken_az() {
String uuid = "abcdefghijklmnopqrstuvwxyz";
String renewedToken = restSecurityBean.renewToken(uuid);
Assert.assertEquals(uuid, renewedToken);
}
@Test
public void renewToken_AZ() {
String uuid = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
String renewedToken = restSecurityBean.renewToken(uuid);
Assert.assertEquals(uuid, renewedToken);
}
@Test
public void renewToken_tooLong() {
String uuid = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
String renewedToken = restSecurityBean.renewToken(uuid);
Assert.assertNull(renewedToken);
}
@Test
public void renewToken_notAllowed() {
String uuid = "abc:test";
String renewedToken = restSecurityBean.renewToken(uuid);
Assert.assertNull(renewedToken);
}
}
...@@ -331,6 +331,7 @@ import org.junit.runners.Suite; ...@@ -331,6 +331,7 @@ import org.junit.runners.Suite;
org.olat.restapi.RegistrationTest.class, org.olat.restapi.RegistrationTest.class,
org.olat.restapi.DocumentPoolModuleWebServiceTest.class, org.olat.restapi.DocumentPoolModuleWebServiceTest.class,
org.olat.restapi.TaxonomyWebServiceTest.class, org.olat.restapi.TaxonomyWebServiceTest.class,
org.olat.restapi.security.RestSecurityBeanTest.class,
de.bps.onyx.plugin.OnyxModuleTest.class, de.bps.onyx.plugin.OnyxModuleTest.class,
de.bps.onyx.plugin.OnyxResultManagerTest.class, de.bps.onyx.plugin.OnyxResultManagerTest.class,
de.bps.olat.portal.institution.InstitutionPortletTest.class, de.bps.olat.portal.institution.InstitutionPortletTest.class,
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment