Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Migrate vul chain finder to data processing platform #11

Open
wants to merge 40 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
871a793
add callableindex and vulchainfinder to default topics
ashkboos Feb 21, 2022
2fbe385
add vulchain finder module
ashkboos Feb 21, 2022
4377293
Implementing basic plugin that consumes from callable index output to…
ashkboos Feb 21, 2022
98da731
add more intellij files to gitignore
ashkboos Feb 24, 2022
5dbf062
Implement rest api dependency resolver and test it
ashkboos Feb 24, 2022
34eefd7
add base uri of rest api to the arguments of vulchainfinder plugin
ashkboos Feb 24, 2022
c715c4d
fix a minor checkstyle validation
ashkboos Feb 24, 2022
9588a01
Implementing vul chain finder DatabaseUtils and its test
ashkboos Feb 25, 2022
53d39f1
implement and test impact propagator
ashkboos Mar 1, 2022
9f62cb1
add test resources for propagator
ashkboos Mar 1, 2022
5379853
add NodeImpact
ashkboos Mar 2, 2022
653a286
add JsonUtils to vulchainfinder
ashkboos Mar 2, 2022
c5a0472
add callable index to args and config
ashkboos Mar 2, 2022
180107a
add callable index utils to vulchainfinder plugin
ashkboos Mar 2, 2022
70f8c44
add a method and test for querying vulnerable packages to vulchain pl…
ashkboos Mar 2, 2022
8abe3e5
use set in RestAPIResolver to be compatible with merger
ashkboos Mar 2, 2022
8091a24
Implement and test the Main class of vulchainfinder plugin
ashkboos Mar 2, 2022
791d893
add dependency to OPAL plugin in vulchain plugin
ashkboos Mar 2, 2022
7ee536a
disable test in vulchain main test
ashkboos Mar 2, 2022
5585452
Adapt versions to fix build
ashkboos Mar 2, 2022
09fd34f
Finish the implementation of Main of the plugin and its tests
ashkboos Mar 4, 2022
de0ac20
add CoreJacksonModule to vulchainfinder
ashkboos Mar 4, 2022
60d3988
update uri and vul object in the DBUtilstest
ashkboos Mar 4, 2022
c8d0a09
add expected json for vulchain integration test
ashkboos Mar 4, 2022
902d2c4
bump version of vulchainfinder
ashkboos Mar 4, 2022
e2d4c2c
remove old json utils completely
ashkboos Mar 8, 2022
59eb9de
test node reachability and add isReachableTarget method
ashkboos Mar 10, 2022
6ec7f92
add test for FastenJacksonModule and fix the build
ashkboos Mar 11, 2022
b469f90
improve testing of DBUtils
ashkboos Mar 21, 2022
f297370
improve testing of impact propagator
ashkboos Mar 21, 2022
015d95b
Disable the integration tests
ashkboos Mar 21, 2022
f525311
Adapt changes introduced in newer core release
proksch May 19, 2022
127a00b
Fix the CLI args for VulChainFinder and add asserts for them
mir-am May 13, 2022
77ecb44
Provide the baseDir arg for storing Vuln. chain repos
mir-am May 13, 2022
c843bf3
Open RocksDB in the read-only mode to bypass LOCK
mir-am May 13, 2022
e7fa3b6
Use internal REST API, not through proxy (needed for production)
mir-am May 14, 2022
1a02e29
Store all the vuln. chains regardless of whether there exists a vulne…
mir-am May 16, 2022
700a8d2
Check if a record is already processed when finding vuln. call chains.
mir-am May 16, 2022
936e533
Create `RestApiError` to stop `vulnchainfinder` when the REST API is …
mir-am May 19, 2022
bd6da37
some minor adjustments for more usability
ashkboos Mar 17, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -6,5 +6,7 @@
**/*.iml
**/*.swp
**/*.versionsBackup
**/*.ipr
**/*.iws


Original file line number Diff line number Diff line change
Expand Up @@ -24,5 +24,5 @@ private DefaultTopics() {
public static final String INGEST = "fasten.mvn.releases";
public static final String POM_ANALYZER = "fasten.POMAnalyzer";
public static final String CALLABLE_INDEXER = "fasten.CallableIndexFastenPlugin";

public static final String VUL_CHAIN_FINDER = "fasten.VulChainFinder";
}
Original file line number Diff line number Diff line change
Expand Up @@ -25,5 +25,7 @@ public class DefaultTopicsTest {
public void defaultValues() {
assertEquals("fasten.mvn.releases", DefaultTopics.INGEST);
assertEquals("fasten.POMAnalyzer", DefaultTopics.POM_ANALYZER);
assertEquals("fasten.CallableIndexFastenPlugin", DefaultTopics.CALLABLE_INDEXER);
assertEquals("fasten.VulChainFinder", DefaultTopics.VUL_CHAIN_FINDER);
}
}
9 changes: 6 additions & 3 deletions infrastructure/loader/pom.xml
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>eu.fasten-project</groupId>
Expand Down Expand Up @@ -70,7 +69,11 @@
<artifactId>ingested-artifact-completion</artifactId>
<version>0.0.8-SNAPSHOT</version>
</dependency>

<dependency>
<groupId>eu.fasten-project</groupId>
<artifactId>vulnerable-chain-finder</artifactId>
<version>0.0.8-SNAPSHOT</version>
</dependency>
<!-- actual dependencies -->
<dependency>
<groupId>org.reflections</groupId>
Expand Down
1 change: 1 addition & 0 deletions plugins/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
<packaging>pom</packaging>

<modules>
<module>vulnerable-chain-finder</module>
<module>examples</module>
<module>maven-crawler</module>
<module>pom-analyzer</module>
Expand Down
32 changes: 32 additions & 0 deletions plugins/vulnerable-chain-finder/pom.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<parent>
<groupId>eu.fasten-project</groupId>
<artifactId>plugins</artifactId>
<version>0.0.8-SNAPSHOT</version>
</parent>
<artifactId>vulnerable-chain-finder</artifactId>

<dependencies>
<dependency>
<groupId>eu.fasten-project</groupId>
<artifactId>pom-analyzer</artifactId>
<version>0.0.8-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>eu.fasten.analyzer</groupId>
<artifactId>javacg-opal</artifactId>
<version>0.0.8</version>
</dependency>
<dependency>
<groupId>eu.fasten-project</groupId>
<artifactId>infrastructure-impl</artifactId>
<version>0.0.8-SNAPSHOT</version>
<scope>test</scope>
</dependency>
</dependencies>
</project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
/*
* Copyright 2021 Delft University of Technology
*
* Licensed 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 eu.f4sten.vulchainfinder;

import eu.f4sten.infra.AssertArgs;
import eu.f4sten.infra.Plugin;
import eu.f4sten.infra.json.TRef;
import eu.f4sten.infra.kafka.Kafka;
import eu.f4sten.infra.kafka.Lane;
import eu.f4sten.infra.kafka.Message;
import eu.f4sten.infra.kafka.MessageGenerator;
import eu.f4sten.pomanalyzer.data.MavenId;
import eu.f4sten.vulchainfinder.exceptions.RestApiError;
import eu.f4sten.vulchainfinder.utils.DatabaseUtils;
import eu.f4sten.vulchainfinder.utils.ImpactPropagator;
import eu.f4sten.vulchainfinder.utils.RestAPIDependencyResolver;
import eu.fasten.core.data.callableindex.RocksDao;
import eu.fasten.core.maven.data.Pom;
import eu.fasten.core.merge.CGMerger;
import eu.fasten.core.vulchains.VulnerableCallChain;
import eu.fasten.core.vulchains.VulnerableCallChainRepository;

import java.io.File;
import java.util.HashSet;
import java.util.Set;
import javax.inject.Inject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class Main implements Plugin {

private static final Logger LOG = LoggerFactory.getLogger(Main.class);
public final RestAPIDependencyResolver resolver;
private final DatabaseUtils db;
private final RocksDao dao;
private final Kafka kafka;
private final VulChainFinderArgs args;
private final MessageGenerator msgs;
private final VulnerableCallChainRepository repo;

private MavenId curId;

@Inject
public Main(DatabaseUtils db, RocksDao dao, Kafka kafka, VulChainFinderArgs args, MessageGenerator msgs,
RestAPIDependencyResolver resolver, VulnerableCallChainRepository repo) {
this.db = db;
this.dao = dao;
this.kafka = kafka;
this.args = args;
this.msgs = msgs;
this.resolver = resolver;
this.repo = repo;
}

@Override
public void run() {
AssertArgs.assertFor(args) //
.notNull(a -> a.kafkaIn, "kafka input topic") //
.notNull(a -> a.kafkaOut, "kafka output topic");

LOG.info("Subscribing to '{}', will publish in '{}' ...", args.kafkaIn, args.kafkaOut);

final var msgClass = new TRef<Message<Message<Message<Message<MavenId, Pom>, Object>, Object>, Object>>() {};

kafka.subscribe(args.kafkaIn, msgClass, (msg, l) -> {
final var pom = msg.input.input.input.payload;
curId = extractMavenIdFrom(pom);
LOG.info("Consuming next record ...");
runOrPublishErr(this::process);
});
while (true) {
LOG.debug("Polling ...");
kafka.poll();
}
}

public void process() {
// NOTE: this can be a temporary FS-based check and can be replaced with a better approach or removed at all.
if (isCurIdProcessed()) {
LOG.info("Coordinate {} already processed!", curId.asCoordinate());
return;
}

LOG.info("Processing {}", curId.asCoordinate());

final var allDeps = resolver.resolveDependencyIds(curId);

final var vulDeps = db.selectVulnerablePackagesExistingIn(allDeps);

Set<VulnerableCallChain> vulChains = new HashSet<>();
if (curIdIsPackageLevelVulnerable(vulDeps)) {
vulChains = extractVulCallChains(allDeps, vulDeps);
}

curIdIsMethodLevelVulnerable(vulChains);
// NOTE: it stores empty vuln. chains too to avoid re-processing records.
storeInVulRepo(vulChains);
}

private boolean curIdIsMethodLevelVulnerable(final Set<VulnerableCallChain> vulChains) {
return !vulChains.isEmpty();
}

private boolean curIdIsPackageLevelVulnerable(final Set<Long> vulDeps) {
return vulDeps != null && !vulDeps.isEmpty();
}

private void storeInVulRepo(final Set<VulnerableCallChain> vulnerableCallChains) {
final var productName = String.format("%s:%s", curId.groupId, curId.artifactId);
repo.store(productName, curId.version, vulnerableCallChains);
}

private Set<VulnerableCallChain> extractVulCallChains(final Set<Long> allDeps, final Set<Long> vulDeps) {
Set<VulnerableCallChain> result = new HashSet<>();

final var merger = new CGMerger(allDeps, db.getContext(), dao);
final var mergedCG = merger.mergeAllDeps();
final var vulCallables = db.selectVulCallablesOf(vulDeps);
final var propagator = new ImpactPropagator(mergedCG, merger.getAllUrisFromDB(mergedCG));
propagator.propagateUrisImpacts(vulCallables.keySet());
LOG.info("Found {} distinct vulnerable paths", propagator.getImpacts().size());

if (!propagator.getImpacts().isEmpty()) {
result = propagator.extractApplicationVulChains(vulCallables, curId);
}

return result;
}

public static MavenId extractMavenIdFrom(final Pom pom) {
final var id = new MavenId();
id.groupId = pom.groupId;
id.artifactId = pom.artifactId;
id.version = pom.version;
return id;
}

private void runOrPublishErr(final Runnable r) {
try {
r.run();
} catch (RestApiError e) {
LOG.error("Forced to stop the plug-in as the REST API is unavailable", e);
throw e;
} catch (Exception e) {
LOG.warn("Execution failed for input: {}", curId, e);

var msg = msgs.getErr(curId, returnCause(e));
kafka.publish(msg, args.kafkaOut, Lane.ERROR);
}
}

private Throwable returnCause(final Exception e) {
final var isRunTime = RuntimeException.class.equals(e.getClass());
final var causeNotNull = e.getCause() != null;
if (isRunTime && causeNotNull) {
return e.getCause();
}
return e;
}

public MavenId getCurId() {
return curId;
}

public void setCurId(final MavenId curId) {
this.curId = curId;
}

private boolean isCurIdProcessed() {
return new File(repo.getFilePath(String.format("%s:%s", curId.groupId, curId.artifactId), curId.version)).exists();
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
/*
* Copyright 2021 Delft University of Technology
*
* Licensed 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 eu.f4sten.vulchainfinder;

import com.beust.jcommander.Parameter;
import eu.f4sten.infra.kafka.DefaultTopics;

import java.io.File;

public class VulChainFinderArgs {

@Parameter(names = "--vulchainfinder.kafkaIn", arity = 1)
public String kafkaIn = DefaultTopics.CALLABLE_INDEXER;

@Parameter(names = "--vulchainfinder.kafkaOut", arity = 1)
public String kafkaOut = DefaultTopics.VUL_CHAIN_FINDER;

@Parameter(names = "--restApiBaseUrl", arity = 1)
public String restApiBaseURL;

@Parameter(names = "--callableIndexPath", arity = 1)
public File callableIndexPath;

@Parameter(names = "--vulnChainRepoPath", arity = 1)
public File vulnChainRepoPath;
}
Loading