158 lines
6.1 KiB
Java
158 lines
6.1 KiB
Java
package space.fediverse.graph;
|
|
|
|
import org.gephi.graph.api.GraphController;
|
|
import org.gephi.graph.api.GraphModel;
|
|
import org.gephi.graph.api.Node;
|
|
import org.gephi.graph.api.UndirectedGraph;
|
|
import org.gephi.io.database.drivers.PostgreSQLDriver;
|
|
import org.gephi.io.database.drivers.SQLUtils;
|
|
import org.gephi.io.exporter.api.ExportController;
|
|
import org.gephi.io.importer.api.Container;
|
|
import org.gephi.io.importer.api.EdgeDirectionDefault;
|
|
import org.gephi.io.importer.api.ImportController;
|
|
import org.gephi.io.importer.plugin.database.EdgeListDatabaseImpl;
|
|
import org.gephi.io.importer.plugin.database.ImporterEdgeList;
|
|
import org.gephi.io.processor.plugin.DefaultProcessor;
|
|
import org.gephi.layout.plugin.AutoLayout;
|
|
import org.gephi.layout.plugin.forceAtlas2.ForceAtlas2;
|
|
import org.gephi.project.api.ProjectController;
|
|
import org.gephi.project.api.Workspace;
|
|
import org.openide.util.Lookup;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.sql.Connection;
|
|
import java.sql.PreparedStatement;
|
|
import java.sql.SQLException;
|
|
import java.sql.Statement;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
public class GraphBuilder {
|
|
|
|
private static final String nodeQuery = String.join(""
|
|
, "SELECT"
|
|
, " scraper_instance.name AS id,"
|
|
, " scraper_instance.name AS label,"
|
|
, " scraper_instance.user_count"
|
|
, " FROM scraper_instance WHERE status = 'success'"
|
|
);
|
|
|
|
private static final String edgeQuery = String.join(""
|
|
, "SELECT"
|
|
, " scraper_edge.source_id AS source,"
|
|
, " scraper_edge.target_id AS target"
|
|
, " FROM scraper_edge"
|
|
);
|
|
|
|
|
|
public static void main(String[] args) {
|
|
|
|
Path currentRelativePath = Paths.get(".");
|
|
|
|
// Init project & workspace; required to do things w/ gephi
|
|
ProjectController pc = Lookup.getDefault().lookup(ProjectController.class);
|
|
pc.newProject();
|
|
Workspace workspace = pc.getCurrentWorkspace();
|
|
|
|
// Get controllers and models
|
|
ImportController importController = Lookup.getDefault().lookup(ImportController.class);
|
|
GraphModel graphModel = Lookup.getDefault().lookup(GraphController.class).getGraphModel();
|
|
// AttributeModel?
|
|
|
|
// Get config variables
|
|
String postgresDb = System.getenv("POSTGRES_DB");
|
|
String postgresUser = System.getenv("POSTGRES_USER");
|
|
String postgresPassword = System.getenv("POSTGRES_PASSWORD");
|
|
if (postgresDb == null || postgresUser == null || postgresPassword == null) {
|
|
throw new RuntimeException(String.format("Incomplete config, canceling. DB: %s, user: %s, pass: %s",
|
|
postgresDb, postgresUser, postgresPassword));
|
|
}
|
|
|
|
// Import from database
|
|
EdgeListDatabaseImpl db = new EdgeListDatabaseImpl();
|
|
db.setSQLDriver(new PostgreSQLDriver());
|
|
db.setHost("db");
|
|
db.setPort(5432);
|
|
db.setDBName(postgresDb);
|
|
db.setUsername(postgresUser);
|
|
db.setPasswd(postgresPassword);
|
|
db.setNodeQuery(nodeQuery);
|
|
db.setEdgeQuery(edgeQuery);
|
|
|
|
ImporterEdgeList edgeListImporter = new ImporterEdgeList();
|
|
Container container = importController.importDatabase(db, edgeListImporter);
|
|
// If a node is in the edge list, but not node list, we don't want to create it automatically
|
|
container.getLoader().setAllowAutoNode(false);
|
|
container.getLoader().setAllowSelfLoop(false);
|
|
container.getLoader().setEdgeDefault(EdgeDirectionDefault.UNDIRECTED); // This is an undirected graph
|
|
|
|
// Add imported data to graph
|
|
importController.process(container, new DefaultProcessor(), workspace);
|
|
|
|
// Layout
|
|
AutoLayout autoLayout = new AutoLayout(2, TimeUnit.MINUTES);
|
|
autoLayout.setGraphModel(graphModel);
|
|
// YifanHuLayout firstLayout = new YifanHuLayout(null, new StepDisplacement(1f));
|
|
ForceAtlas2 forceAtlas2Layout = new ForceAtlas2(null);
|
|
forceAtlas2Layout.setLinLogMode(true);
|
|
autoLayout.addLayout(forceAtlas2Layout, 1f);
|
|
autoLayout.execute();
|
|
|
|
// Update coordinates in database
|
|
// First, connect
|
|
String dbUrl = SQLUtils.getUrl(db.getSQLDriver(), db.getHost(), db.getPort(), db.getDBName());
|
|
Connection conn = null;
|
|
try {
|
|
conn = db.getSQLDriver().getConnection(dbUrl, db.getUsername(), db.getPasswd());
|
|
} catch (SQLException e) {
|
|
if (conn != null) {
|
|
try {
|
|
conn.close();
|
|
} catch (Exception e2) {
|
|
// Closing failed; ah well
|
|
}
|
|
}
|
|
throw new RuntimeException(e);
|
|
}
|
|
// Update
|
|
UndirectedGraph graph = graphModel.getUndirectedGraph();
|
|
for (Node node: graph.getNodes()) {
|
|
String id = node.getId().toString();
|
|
float x = node.x();
|
|
float y = node.y();
|
|
|
|
try {
|
|
PreparedStatement statement = conn.prepareStatement(
|
|
"UPDATE scraper_instance SET x_coord=?, y_coord=? WHERE name=?");
|
|
statement.setFloat(1, x);
|
|
statement.setFloat(2, y);
|
|
statement.setString(3, id);
|
|
statement.executeUpdate();
|
|
} catch (SQLException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
// Close connection
|
|
try {
|
|
conn.close();
|
|
} catch (SQLException e) {
|
|
// Closing failed; ah well
|
|
}
|
|
|
|
|
|
// Also export to gexf
|
|
ExportController exportController = Lookup.getDefault().lookup(ExportController.class);
|
|
try {
|
|
exportController.exportFile(new File("fediverse.gexf"));
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
|
|
// Gephi doesn't seem to provide a good way to close its postgres connection, so we have to force close the
|
|
// program. This'll leave a hanging connection for some period ¯\_(ツ)_/¯
|
|
System.exit(0);
|
|
}
|
|
}
|