Reading yml file based lesson configuration
This commit is contained in:
parent
966e5b9e0a
commit
f12c06fc55
@ -31,6 +31,7 @@
|
|||||||
package org.owasp.webgoat.controller;
|
package org.owasp.webgoat.controller;
|
||||||
|
|
||||||
import org.owasp.webgoat.lessons.RandomLessonAdapter;
|
import org.owasp.webgoat.lessons.RandomLessonAdapter;
|
||||||
|
import org.owasp.webgoat.plugins.YmlBasedLesson;
|
||||||
import org.owasp.webgoat.session.WebSession;
|
import org.owasp.webgoat.session.WebSession;
|
||||||
import org.springframework.stereotype.Controller;
|
import org.springframework.stereotype.Controller;
|
||||||
import org.springframework.web.bind.annotation.RequestMapping;
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
@ -63,7 +64,8 @@ public class StartLesson {
|
|||||||
model.addObject("lesson", ws.getCurrentLesson());
|
model.addObject("lesson", ws.getCurrentLesson());
|
||||||
model.addObject("message", ws.getMessage());
|
model.addObject("message", ws.getMessage());
|
||||||
model.addObject("instructions", ws.getInstructions());
|
model.addObject("instructions", ws.getInstructions());
|
||||||
model.addObject("migrated", refactored.contains(ws.getCurrentLesson().getClass().getSimpleName())); //remove after ECS removal otherwise you will see the lesson twice
|
boolean isMigrated = ws.getCurrentLesson() instanceof YmlBasedLesson;
|
||||||
|
model.addObject("migrated", isMigrated); //remove after ECS removal otherwise you will see the lesson twice
|
||||||
model.setViewName("lesson_content");
|
model.setViewName("lesson_content");
|
||||||
return model;
|
return model;
|
||||||
}
|
}
|
||||||
|
@ -998,6 +998,4 @@ public abstract class AbstractLesson extends Screen implements Comparable<Object
|
|||||||
}
|
}
|
||||||
return labelManager;
|
return labelManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,36 @@
|
|||||||
|
package org.owasp.webgoat.plugins;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ************************************************************************************************
|
||||||
|
* This file is part of WebGoat, an Open Web Application Security Project utility. For details,
|
||||||
|
* please see http://www.owasp.org/
|
||||||
|
* <p>
|
||||||
|
* Copyright (c) 2002 - 20014 Bruce Mayhew
|
||||||
|
* <p>
|
||||||
|
* This program is free software; you can redistribute it and/or modify it under the terms of the
|
||||||
|
* GNU General Public License as published by the Free Software Foundation; either version 2 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
* <p>
|
||||||
|
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
|
||||||
|
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
* <p>
|
||||||
|
* You should have received a copy of the GNU General Public License along with this program; if
|
||||||
|
* not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
||||||
|
* 02111-1307, USA.
|
||||||
|
* <p>
|
||||||
|
* Getting Source ==============
|
||||||
|
* <p>
|
||||||
|
* Source for this application is maintained at https://github.com/WebGoat/WebGoat, a repository for free software
|
||||||
|
* projects.
|
||||||
|
* <p>
|
||||||
|
*
|
||||||
|
* @author WebGoat
|
||||||
|
* @version $Id: $Id
|
||||||
|
* @since June 28, 2016
|
||||||
|
*/
|
||||||
|
public class LessonConfiguration {
|
||||||
|
|
||||||
|
private String title;
|
||||||
|
|
||||||
|
}
|
@ -1,27 +1,29 @@
|
|||||||
package org.owasp.webgoat.plugins;
|
package org.owasp.webgoat.plugins;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import com.fasterxml.jackson.dataformat.yaml.YAMLFactory;
|
||||||
import com.google.common.base.Optional;
|
import com.google.common.base.Optional;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.owasp.webgoat.lessons.AbstractLesson;
|
import org.owasp.webgoat.lessons.AbstractLesson;
|
||||||
import org.springframework.util.StringUtils;
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import static org.owasp.webgoat.plugins.PluginFileUtils.fileEndsWith;
|
import static org.owasp.webgoat.plugins.PluginFileUtils.fileEndsWith;
|
||||||
import static org.owasp.webgoat.plugins.PluginFileUtils.hasParentDirectoryWithName;
|
import static org.owasp.webgoat.plugins.PluginFileUtils.hasParentDirectoryWithName;
|
||||||
import static org.owasp.webgoat.plugins.PluginFileUtils.replaceInFiles;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Plugin class.</p>
|
* <p>Plugin class.</p>
|
||||||
*
|
*
|
||||||
* @version $Id: $Id
|
|
||||||
* @author dm
|
* @author dm
|
||||||
|
* @version $Id: $Id
|
||||||
*/
|
*/
|
||||||
public class Plugin {
|
public class Plugin {
|
||||||
|
|
||||||
@ -30,6 +32,7 @@ public class Plugin {
|
|||||||
|
|
||||||
private PluginClassLoader classLoader;
|
private PluginClassLoader classLoader;
|
||||||
private Class<AbstractLesson> lesson;
|
private Class<AbstractLesson> lesson;
|
||||||
|
private YmlBasedLesson ymlBasedLesson;
|
||||||
private Map<String, File> solutionLanguageFiles = new HashMap<>();
|
private Map<String, File> solutionLanguageFiles = new HashMap<>();
|
||||||
private Map<String, File> lessonPlansLanguageFiles = new HashMap<>();
|
private Map<String, File> lessonPlansLanguageFiles = new HashMap<>();
|
||||||
private List<File> pluginFiles = Lists.newArrayList();
|
private List<File> pluginFiles = Lists.newArrayList();
|
||||||
@ -51,6 +54,7 @@ public class Plugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void findLesson(String name) {
|
private void findLesson(String name) {
|
||||||
|
//Old code remove after we migrated the lessons
|
||||||
String realClassName = StringUtils.trimLeadingCharacter(name, '/').replaceAll("/", ".").replaceAll(".class", "");
|
String realClassName = StringUtils.trimLeadingCharacter(name, '/').replaceAll("/", ".").replaceAll(".class", "");
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -62,6 +66,33 @@ public class Plugin {
|
|||||||
} catch (ClassNotFoundException ce) {
|
} catch (ClassNotFoundException ce) {
|
||||||
throw new PluginLoadingFailure("Class " + realClassName + " listed in jar but unable to load the class.", ce);
|
throw new PluginLoadingFailure("Class " + realClassName + " listed in jar but unable to load the class.", ce);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//New code all lessons should work as below
|
||||||
|
if (this.lesson == null) {
|
||||||
|
readYmlLessonConfiguration();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void readYmlLessonConfiguration() {
|
||||||
|
java.util.Optional<File> ymlFile = this.pluginFiles.stream().filter(f -> f.getName().endsWith(".yml")).findFirst();
|
||||||
|
if (ymlFile.isPresent()) {
|
||||||
|
try {
|
||||||
|
String ymlStr = FileUtils.readFileToString(ymlFile.get());
|
||||||
|
ObjectMapper mapper = new ObjectMapper(new YAMLFactory());
|
||||||
|
Map<String, Object> ymlAsMap = mapper.readValue(ymlStr, new TypeReference<Map<String, Object>>() {
|
||||||
|
});
|
||||||
|
Map<String, Object> lessonYml = (Map<String, Object>) ymlAsMap.get("lesson");
|
||||||
|
final String category = (String) lessonYml.get("category");
|
||||||
|
final List<String> hints = (List<String>) lessonYml.get("hints");
|
||||||
|
final String title = (String) lessonYml.get("title");
|
||||||
|
final String html = (String) lessonYml.get("html");
|
||||||
|
this.ymlBasedLesson = new YmlBasedLesson(category, hints, title, html);
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new PluginLoadingFailure("Unable to read yml file", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -80,7 +111,7 @@ public class Plugin {
|
|||||||
lessonSourceFile = file.toFile();
|
lessonSourceFile = file.toFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fileEndsWith(file, ".css", ".jsp", ".js")) {
|
if (fileEndsWith(file, ".css", ".jsp", ".js", ".yml")) {
|
||||||
pluginFiles.add(file.toFile());
|
pluginFiles.add(file.toFile());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -91,33 +122,33 @@ public class Plugin {
|
|||||||
* @param pluginTarget a {@link java.nio.file.Path} object.
|
* @param pluginTarget a {@link java.nio.file.Path} object.
|
||||||
*/
|
*/
|
||||||
public void rewritePaths(Path pluginTarget) {
|
public void rewritePaths(Path pluginTarget) {
|
||||||
try {
|
// try {
|
||||||
replaceInFiles(this.lesson.getSimpleName() + "_files",
|
// replaceInFiles(this.lesson.getSimpleName() + "_files",
|
||||||
"plugin_lessons/plugin/" + this.lesson
|
// "plugin_lessons/plugin/" + this.lesson
|
||||||
.getSimpleName() + "/lessonSolutions/en/" + this.lesson.getSimpleName() + "_files",
|
// .getSimpleName() + "/lessonSolutions/en/" + this.lesson.getSimpleName() + "_files",
|
||||||
solutionLanguageFiles.values());
|
// solutionLanguageFiles.values());
|
||||||
replaceInFiles(this.lesson.getSimpleName() + "_files",
|
// replaceInFiles(this.lesson.getSimpleName() + "_files",
|
||||||
"plugin_lessons/plugin/" + this.lesson
|
// "plugin_lessons/plugin/" + this.lesson
|
||||||
.getSimpleName() + "/lessonPlans/en/" + this.lesson.getSimpleName() + "_files",
|
// .getSimpleName() + "/lessonPlans/en/" + this.lesson.getSimpleName() + "_files",
|
||||||
lessonPlansLanguageFiles.values());
|
// lessonPlansLanguageFiles.values());
|
||||||
|
//
|
||||||
String[] replacements = {"jsp", "js"};
|
// String[] replacements = {"jsp", "js"};
|
||||||
for (String replacement : replacements) {
|
// for (String replacement : replacements) {
|
||||||
String s = String.format("plugin/%s/%s/", this.lesson.getSimpleName(), replacement);
|
// String s = String.format("plugin/%s/%s/", this.lesson.getSimpleName(), replacement);
|
||||||
String r = String.format("plugin_lessons/plugin/s/%s/", this.lesson.getSimpleName(), replacement);
|
// String r = String.format("plugin_lessons/plugin/s/%s/", this.lesson.getSimpleName(), replacement);
|
||||||
replaceInFiles(s, r, pluginFiles);
|
// replaceInFiles(s, r, pluginFiles);
|
||||||
replaceInFiles(s, r, Arrays.asList(lessonSourceFile));
|
// replaceInFiles(s, r, Arrays.asList(lessonSourceFile));
|
||||||
}
|
// }
|
||||||
|
//
|
||||||
//CSS with url('/plugin/images') should not begin with / otherwise image cannot be found
|
// //CSS with url('/plugin/images') should not begin with / otherwise image cannot be found
|
||||||
String s = String.format("/plugin/%s/images/", this.lesson.getSimpleName());
|
// String s = String.format("/plugin/%s/images/", this.lesson.getSimpleName());
|
||||||
String r = String
|
// String r = String
|
||||||
.format("plugin_lessons/plugin/%s/images/", this.lesson.getSimpleName());
|
// .format("plugin_lessons/plugin/%s/images/", this.lesson.getSimpleName());
|
||||||
replaceInFiles(s, r, pluginFiles);
|
// replaceInFiles(s, r, pluginFiles);
|
||||||
replaceInFiles(s, r, Arrays.asList(lessonSourceFile));
|
// replaceInFiles(s, r, Arrays.asList(lessonSourceFile));
|
||||||
} catch (IOException e) {
|
// } catch (IOException e) {
|
||||||
throw new PluginLoadingFailure("Unable to rewrite the paths in the solutions", e);
|
// throw new PluginLoadingFailure("Unable to rewrite the paths in the solutions", e);
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -127,9 +158,13 @@ public class Plugin {
|
|||||||
*/
|
*/
|
||||||
public Optional<AbstractLesson> getLesson() {
|
public Optional<AbstractLesson> getLesson() {
|
||||||
try {
|
try {
|
||||||
|
if (ymlBasedLesson != null) {
|
||||||
|
return Optional.of(ymlBasedLesson);
|
||||||
|
}
|
||||||
if (lesson != null) {
|
if (lesson != null) {
|
||||||
return Optional.of(lesson.newInstance());
|
return Optional.of(lesson.newInstance());
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (IllegalAccessException | InstantiationException e) {
|
} catch (IllegalAccessException | InstantiationException e) {
|
||||||
throw new PluginLoadingFailure("Unable to instantiate the lesson " + lesson.getName(), e);
|
throw new PluginLoadingFailure("Unable to instantiate the lesson " + lesson.getName(), e);
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,15 @@ package org.owasp.webgoat.plugins;
|
|||||||
*/
|
*/
|
||||||
public class PluginLoadingFailure extends RuntimeException {
|
public class PluginLoadingFailure extends RuntimeException {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Constructor for PluginLoadingFailure.</p>
|
||||||
|
*
|
||||||
|
* @param message a {@link java.lang.String} object.
|
||||||
|
*/
|
||||||
|
public PluginLoadingFailure(String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Constructor for PluginLoadingFailure.</p>
|
* <p>Constructor for PluginLoadingFailure.</p>
|
||||||
*
|
*
|
||||||
|
@ -0,0 +1,78 @@
|
|||||||
|
package org.owasp.webgoat.plugins;
|
||||||
|
|
||||||
|
import org.owasp.webgoat.lessons.Category;
|
||||||
|
import org.owasp.webgoat.lessons.LessonAdapter;
|
||||||
|
import org.owasp.webgoat.session.WebSession;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ************************************************************************************************
|
||||||
|
* This file is part of WebGoat, an Open Web Application Security Project utility. For details,
|
||||||
|
* please see http://www.owasp.org/
|
||||||
|
* <p>
|
||||||
|
* Copyright (c) 2002 - 20014 Bruce Mayhew
|
||||||
|
* <p>
|
||||||
|
* This program is free software; you can redistribute it and/or modify it under the terms of the
|
||||||
|
* GNU General Public License as published by the Free Software Foundation; either version 2 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
* <p>
|
||||||
|
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
|
||||||
|
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
* <p>
|
||||||
|
* You should have received a copy of the GNU General Public License along with this program; if
|
||||||
|
* not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
||||||
|
* 02111-1307, USA.
|
||||||
|
* <p>
|
||||||
|
* Getting Source ==============
|
||||||
|
* <p>
|
||||||
|
* Source for this application is maintained at https://github.com/WebGoat/WebGoat, a repository for free software
|
||||||
|
* projects.
|
||||||
|
* <p>
|
||||||
|
*
|
||||||
|
* @author WebGoat
|
||||||
|
* @version $Id: $Id
|
||||||
|
* @since June 28, 2016
|
||||||
|
*/
|
||||||
|
public class YmlBasedLesson extends LessonAdapter {
|
||||||
|
|
||||||
|
private final static Integer DEFAULT_RANKING = new Integer(10);
|
||||||
|
private final String category;
|
||||||
|
private final List<String> hints;
|
||||||
|
private final String title;
|
||||||
|
private final String html;
|
||||||
|
|
||||||
|
public YmlBasedLesson(String category, List<String> hints, String title, String html) {
|
||||||
|
this.category = category;
|
||||||
|
this.hints = hints;
|
||||||
|
this.title = title;
|
||||||
|
this.html = html;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Category getDefaultCategory() {
|
||||||
|
return Category.getCategory(category);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected List<String> getHints(WebSession s) {
|
||||||
|
return hints;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Integer getDefaultRanking() {
|
||||||
|
return DEFAULT_RANKING;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTitle() {
|
||||||
|
return title;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getHtml() {
|
||||||
|
return html;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
<div th:switch="${migrated}">
|
<div th:switch="${migrated}">
|
||||||
<div th:case="false" th:utext="${lesson.content}"></div>
|
<div th:case="false" th:utext="${lesson.content}"></div>
|
||||||
<div th:case="true" th:replace="lesson:__${lesson.class.simpleName}__"></div>
|
<div th:case="true" th:replace="lesson:__${lesson.html}__"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user