19#include <polyfem/embedded_spec/polyfem.hpp>
21#include <polysolve/linear/Solver.hpp>
23#include <spdlog/sinks/basic_file_sink.h>
24#include <spdlog/sinks/ostream_sink.h>
25#include <spdlog/sinks/stdout_color_sinks.h>
27#include <ipc/utils/logger.hpp>
28#ifdef POLYFEM_WITH_ITR
29#include <wmtk/utils/Logger.hpp>
43 spdlog::level::level_enum,
44 {{spdlog::level::level_enum::trace,
"trace"},
45 {spdlog::level::level_enum::debug,
"debug"},
46 {spdlog::level::level_enum::info,
"info"},
47 {spdlog::level::level_enum::warn,
"warning"},
48 {spdlog::level::level_enum::err,
"error"},
49 {spdlog::level::level_enum::critical,
"critical"},
50 {spdlog::level::level_enum::off,
"off"},
51 {spdlog::level::level_enum::trace, 0},
52 {spdlog::level::level_enum::debug, 1},
53 {spdlog::level::level_enum::info, 2},
54 {spdlog::level::level_enum::warn, 3},
55 {spdlog::level::level_enum::err, 3},
56 {spdlog::level::level_enum::critical, 4},
57 {spdlog::level::level_enum::off, 5}})
63 using namespace utils;
67 std::string root_path(
const json &args)
70 return args[
"root_path"].get<std::string>();
74 std::string resolve_input_path(
const json &args,
const std::string &path,
const bool only_if_exists =
false)
79 std::string resolve_output_path(
const std::string &output_dir,
const std::string &path)
81 if (output_dir.empty() ||
path.empty() || std::filesystem::path(path).is_absolute())
83 return std::filesystem::weakly_canonical(std::filesystem::path(output_dir) / path).string();
86 bool contact_enabled(
const json &args)
88 return args[
"contact"][
"enabled"];
91 void init_time(
json &args, Units &units)
96 const double t0 =
Units::convert(args[
"time"][
"t0"], units.time());
108 else if (num_valid == 2)
118 time_steps = int(std::ceil((tend - t0) / dt));
119 assert(time_steps > 0);
123 time_steps =
args[
"time"][
"time_steps"];
124 assert(time_steps > 0);
125 dt = (tend - t0) / time_steps;
130 throw std::runtime_error(
"This code should be unreachable!");
140 time_steps =
args[
"time"][
"time_steps"];
141 assert(time_steps > 0);
143 tend = t0 + time_steps * dt;
147 throw std::runtime_error(
"This code should be unreachable!");
150 else if (num_valid == 3)
154 time_steps =
args[
"time"][
"time_steps"];
156 if (std::abs(t0 + dt * time_steps - tend) > 1e-12)
160 args[
"time"][
"tend"] = tend;
161 args[
"time"][
"dt"] = dt;
162 args[
"time"][
"time_steps"] = time_steps;
164 units.characteristic_length() *= dt;
166 logger().info(
"t0={}, dt={}, tend={}", t0, dt, tend);
176 const std::string &log_file,
177 const spdlog::level::level_enum log_level,
178 const spdlog::level::level_enum file_log_level,
181 std::vector<spdlog::sink_ptr> sinks;
185 console_sink_ = std::make_shared<spdlog::sinks::stdout_color_sink_mt>();
189 if (!log_file.empty())
191 file_sink_ = std::make_shared<spdlog::sinks::basic_file_sink_mt>(log_file,
true);
197 spdlog::flush_every(std::chrono::seconds(3));
202 std::vector<spdlog::sink_ptr> sinks;
203 sinks.emplace_back(std::make_shared<spdlog::sinks::ostream_sink_mt>(os,
false));
208 const std::vector<spdlog::sink_ptr> &sinks,
209 const spdlog::level::level_enum log_level)
211 set_logger(std::make_shared<spdlog::logger>(
"polyfem", sinks.begin(), sinks.end()));
214 ipc::set_logger(std::make_shared<spdlog::logger>(
"ipctk", sinks.begin(), sinks.end()));
216#ifdef POLYFEM_WITH_ITR
217 wmtk::set_logger(std::make_shared<spdlog::logger>(
"wmtk", sinks.begin(), sinks.end()));
220 logger().set_level(spdlog::level::trace);
221 ipc::logger().set_level(spdlog::level::trace);
222#ifdef POLYFEM_WITH_ITR
223 wmtk::logger().set_level(spdlog::level::trace);
231 spdlog::set_level(log_level);
238 logger().set_level(log_level);
239 ipc::logger().set_level(log_level);
245 json args_in = p_args_in;
246 const bool contact_dhat_was_explicit = args_in.contains(
"/contact/dhat"_json_pointer);
253 jse.strict = strict_validation;
254 rules = jse::embed::polyfem_spec::polyfem::spec();
256 polysolve::linear::Solver::apply_default_solver(rules,
"/solver/linear");
257 polysolve::linear::Solver::apply_default_solver(rules,
"/solver/adjoint_linear");
260 polysolve::linear::Solver::select_valid_solver(args_in[
"solver"][
"linear"],
logger());
261 if (args_in[
"solver"][
"adjoint_linear"].is_null())
262 args_in[
"solver"][
"adjoint_linear"] = args_in[
"solver"][
"linear"];
264 polysolve::linear::Solver::select_valid_solver(args_in[
"solver"][
"adjoint_linear"],
logger());
266 if (args_in.contains(
"/solver/nonlinear"_json_pointer))
268 if (args_in.contains(
"/solver/augmented_lagrangian/nonlinear"_json_pointer))
270 assert(args_in[
"solver"][
"augmented_lagrangian"][
"nonlinear"].is_object());
271 json nonlinear = args_in[
"solver"][
"nonlinear"];
272 nonlinear.merge_patch(args_in[
"solver"][
"augmented_lagrangian"][
"nonlinear"]);
273 args_in[
"solver"][
"augmented_lagrangian"][
"nonlinear"] = nonlinear;
277 args_in[
"solver"][
"augmented_lagrangian"][
"nonlinear"] = args_in[
"solver"][
"nonlinear"];
281 const bool valid_input = jse.verify_json(args_in, rules);
284 logger().error(
"invalid input json:\n{}", jse.log2str());
285 throw std::runtime_error(
"Invalid input json file");
288 args = jse.inject_defaults(args_in, rules);
293 if (!args_in.contains(
"/space/advanced/bc_method"_json_pointer) &&
args[
"space"][
"basis_type"] !=
"Lagrange")
295 logger().warn(
"Setting bc method to lsq for non-Lagrange basis");
296 args[
"space"][
"advanced"][
"bc_method"] =
"lsq";
299 const std::string output_dir = resolve_input_path(
args,
args[
"output"][
"directory"].get<std::string>());
300 if (!output_dir.empty())
301 std::filesystem::create_directories(output_dir);
303 std::string out_path_log =
args[
"output"][
"log"][
"path"];
304 if (!out_path_log.empty())
305 out_path_log = resolve_output_path(output_dir, out_path_log);
307 for (
auto &path :
args[
"constraints"][
"hard"])
308 path = resolve_input_path(
args, path.get<std::string>());
310 for (
auto &path :
args[
"constraints"][
"soft"])
311 path[
"data"] = resolve_input_path(
args, path[
"data"].get<std::string>());
315 args[
"output"][
"log"][
"level"],
316 args[
"output"][
"log"][
"file_level"],
317 args[
"output"][
"log"][
"quiet"]);
319 logger().info(
"Saving output to {}", output_dir);
323 init_time(
args, units);
325 if (contact_enabled(
args))
327 if (
args[
"solver"][
"contact"][
"friction_iterations"] == 0)
329 logger().info(
"specified friction_iterations is 0; disabling friction");
330 args[
"contact"][
"friction_coefficient"] = 0.0;
332 else if (
args[
"solver"][
"contact"][
"friction_iterations"] < 0)
334 args[
"solver"][
"contact"][
"friction_iterations"] = std::numeric_limits<int>::max();
336 if (
args[
"contact"][
"friction_coefficient"] == 0.0)
338 args[
"solver"][
"contact"][
"friction_iterations"] = 0;
343 args[
"solver"][
"contact"][
"friction_iterations"] = 0;
344 args[
"contact"][
"friction_coefficient"] = 0;
345 args[
"contact"][
"periodic"] =
false;
349 if (formulation.empty())
351 logger().error(
"specify some 'materials'");
352 throw std::runtime_error(
"invalid input");
357 throw std::runtime_error(
"polyfem::State is varform-only; use polyfem::legacy::State for " + formulation +
".");
360 args[
"contact"][
"_dhat_was_explicit"] = contact_dhat_was_explicit;
362 args[
"contact"].erase(
"_dhat_was_explicit");
372 const std::function<
int(
const size_t,
const std::vector<int> &,
const RowVectorNd &,
bool)> &boundary_marker,
374 bool skip_boundary_sideset)
378 logger().info(
"Loading mesh...");
380 std::unique_ptr<Mesh> mesh =
Mesh::create(meshin, non_conforming);
383 logger().error(
"Unable to load the mesh");
388 mesh->bounding_box(min, max);
390 logger().info(
"mesh bb min [{}], max [{}]", min, max);
392 if (!skip_boundary_sideset)
393 mesh->compute_boundary_ids(boundary_marker);
396 logger().info(
" took {}s", timer.getElapsedTime());
404 const std::vector<std::string> &names,
405 const std::vector<Eigen::MatrixXi> &cells,
406 const std::vector<Eigen::MatrixXd> &vertices)
408 assert(names.size() == cells.size());
409 assert(vertices.size() == cells.size());
414 logger().info(
"Loading mesh ...");
420 args[
"geometry"],
args[
"root_path"],
421 names, vertices, cells, non_conforming);
427 mesh->bounding_box(min, max);
429 logger().info(
"mesh bb min [{}], max [{}]", min, max);
432 logger().info(
" took {}s", timer.getElapsedTime());
434#ifdef POLYFEM_WITH_BEZIER
435 if (!mesh->is_simplicial())
440 args[
"space"][
"advanced"][
"count_flipped_els_continuous"] =
false;
441 args[
"output"][
"paraview"][
"options"][
"jacobian_validity"] =
false;
442 args[
"solver"][
"advanced"][
"check_inversion"] =
"Discrete";
444 else if (
args[
"solver"][
"advanced"][
"check_inversion"] !=
"Discrete")
446 args[
"space"][
"advanced"][
"use_corner_quadrature"] =
true;
std::shared_ptr< varform::VarForm > variational_formulation
active variational formulation
void set_max_threads(const int max_threads=std::numeric_limits< int >::max())
spdlog::sink_ptr file_sink_
spdlog::sink_ptr console_sink_
logger sink to stdout
json args
main input arguments containing all defaults
void set_log_level(const spdlog::level::level_enum log_level)
change log level
std::function< void(int, int, double, double)> time_callback
Optional UI progress callback.
void init_logger(const std::string &log_file, const spdlog::level::level_enum log_level, const spdlog::level::level_enum file_log_level, const bool is_quiet)
initializing the logger
void init(const json &json)
static double convert(const json &val, const std::string &unit_type)
void init(const json &args, const bool strict_validation)
initialize the polyfem solver with a json settings
void load_mesh(bool non_conforming=false, const std::vector< std::string > &names=std::vector< std::string >(), const std::vector< Eigen::MatrixXi > &cells=std::vector< Eigen::MatrixXi >(), const std::vector< Eigen::MatrixXd > &vertices=std::vector< Eigen::MatrixXd >())
loads the mesh from the json arguments
void set_max_threads(const int max_threads=std::numeric_limits< int >::max())
void set_log_level(const spdlog::level::level_enum log_level)
change log level
void init_logger(const std::string &log_file, const spdlog::level::level_enum log_level, const spdlog::level::level_enum file_log_level, const bool is_quiet)
initializing the logger
void solve(Eigen::MatrixXd &sol, Eigen::MatrixXd &pressure, UserPostStepCallback user_post_step={}, const InitialConditionOverride *ic_override=nullptr)
solves the problem, call other methods
static std::unique_ptr< Mesh > create(const std::string &path, const bool non_conforming=false)
factory to build the proper mesh
void set_logger(spdlog::logger &logger)
static GeogramUtils & instance()
void set_num_threads(const int max_threads)
NLOHMANN_JSON_SERIALIZE_ENUM(CollisionProxyTessellation, {{CollisionProxyTessellation::REGULAR, "regular"}, {CollisionProxyTessellation::IRREGULAR, "irregular"}})
std::unique_ptr< Mesh > read_fem_geometry(const Units &units, const json &geometry, const std::string &root_path, const std::vector< std::string > &_names, const std::vector< Eigen::MatrixXd > &_vertices, const std::vector< Eigen::MatrixXi > &_cells, const bool non_conforming)
read FEM meshes from a geometry JSON array (or single)
std::string resolve_path(const std::string &path, const std::string &input_file_path, const bool only_if_exists=false)
bool is_param_valid(const json ¶ms, const std::string &key)
Determine if a key exists and is non-null in a json object.
void apply_common_params(json &args)
spdlog::logger & logger()
Retrieves the current logger.
Eigen::Matrix< double, 1, Eigen::Dynamic, Eigen::RowMajor, 1, 3 > RowVectorNd
void set_logger(std::shared_ptr< spdlog::logger > p_logger)
Setup a logger object to be used by Polyfem.
void log_and_throw_error(const std::string &msg)