Library for maximum likelihood principal component analysis for AnyBody models
Project description
sillywalk
sillywalk is a Python library for Maximum Likelihood Principal Component Analysis (ML-PCA). It allows you to build statistical models from data and predict missing values based on observed values. While it can be used with any numerical dataset, it includes special utilities for working with data from the AnyBody Modeling System™.
Installation
You can install sillywalk from PyPI,
pip install sillywalk
or as a conda package:
pixi install sillywalk
For developers
This project is managed by pixi. To set up a development environment:
git clone https://github.com/AnyBody-Research-Group/sillywalk
cd sillywalk
pixi install
pixi run test
Quick Start
Here's a quick example of how to use sillywalk to predict missing data.
1. Create a statistical model
First, you need a dataset to build the model. sillywalk works with both Pandas and Polars DataFrames.
import pandas as pd
import sillywalk
# Sample data of student measurements
data = {
"Sex": [1, 1, 2, 2, 1, 2],
"Age": [25, 30, 28, 22, 35, 29],
"Stature": [175, 180, 165, 160, 185, 170],
"Bodyweight": [70, 80, 60, 55, 85, 65],
"Shoesize": [42, 44, 39, 38, 45, 40],
}
df = pd.DataFrame(data)
# Create a PCAPredictor model from the data
model = sillywalk.PCAPredictor(df)
2. Predict missing values
Once the model is created, you can use it to predict missing values based on some known values (constraints).
# Define the known values (constraints)
constraints = {
"Stature": 180,
"Bodyweight": 65,
}
# Predict the missing values
result = model.predict(constraints)
# The result is a dictionary containing the original constraints
# and predicted values for the other variables.
print(result)
3. Save and load the model
You can save your trained model to a file and load it later to make new predictions without having to re-train it.
# Save the model to a file
model.save_npz("student_model.npz")
# Load the model from the file
loaded_model = sillywalk.PCAPredictor.from_npz("student_model.npz")
# Use the loaded model to make predictions
new_constraints = {"Age": 24, "Shoesize": 43}
prediction = loaded_model.predict(new_constraints)
print(prediction)
Working with output from AnyBody Models
sillywalk can work with time series data from AnyBody and convert it to fourier coefficients
that are compatible with AnyBody's AnyKinEqFourierDriver class:
import polars as pl
import numpy as np
import sillywalk
# Example time series data (e.g., joint angles over a gait cycle)
time_points = np.linspace(0, 1, 101) # 101 time points
hip_flexion = 30 * np.sin(2 * np.pi * time_points) + 10
knee_flexion = 60 * np.sin(2 * np.pi * time_points + np.pi/4)
# Create DataFrame with time series
time_series_df = pl.DataFrame({
'Main.HumanModel.BodyModel.Interface.Trunk.PelvisThoraxExtension': hip_flexion,
'Main.HumanModel.BodyModel.Interface.Right.KneeFlexion': knee_flexion,
})
# Convert to fourier coefficients (default n_modes=6)
fourier_df = sillywalk.anybody.compute_fourier_coefficients(time_series_df, n_modes=6)
┌────────────┬────────────┬────────────┬───────────┬───┬───────────┬───────────┬───────────┬───────────┐
│ ...tension ┆ ...tension ┆ ...tension ┆ ...tensio ┆ … ┆ ...Flexio ┆ ...Flexio ┆ ...Flexio ┆ ...Flexio │
│ _a0 ┆ _a1 ┆ _a2 ┆ n_a3 ┆ ┆ n_b2 ┆ n_b3 ┆ n_b4 ┆ n_b5 │
│ --- ┆ --- ┆ --- ┆ --- ┆ ┆ --- ┆ --- ┆ --- ┆ --- │
│ f64 ┆ f64 ┆ f64 ┆ f64 ┆ ┆ f64 ┆ f64 ┆ f64 ┆ f64 │
╞════════════╪════════════╪════════════╪═══════════╪═══╪═══════════╪═══════════╪═══════════╪═══════════╡
│ 10.0 ┆ 0.928198 ┆ -0.025044 ┆ -0.021042 ┆ … ┆ -0.550711 ┆ -0.307976 ┆ -0.218252 ┆ -0.169925 │
└────────────┴────────────┴────────────┴───────────┴───┴───────────┴───────────┴───────────┴───────────┘
The two time series columns will be decomposed into each (n_modes*2-1) fourier coefficients.
Each coefficient will have the suffix _a0 to a_5 and _b1 to _b5. The value of _b0 is not
stored as this is always 0.
Generating AnyBody models from data
Sillywalk can create anyscript include-files, from a dictionary (dataframe with one row) with fourier coefficients and/or other anthropometric data.
Fourier drivers are create if the keys/columns follow a specific format: DOF:<measure>[<idx>]_<a/b><mode>. For example:
DOF:Main.HumanModel.BodyModel.Main.HumanModel.BodyModel.Interface.Trunk.PelvisPosX[0]_a4
The dataset also needs to have a single value T, which is period of the fourier driver (until the motion repeats again).
The include file is generated by calling sillywalk.anybody.create_model_file():
# Generate an AnyBody include file from the predicted data
sillywalk.anybody.create_model_file(
predicted_data,
targetfile="predicted_motion.any"
)
The generated file will contain AnyKinEqFourierDriver entries that can be included in your AnyBody model:
AnyFolder PCA_drivers = {
AnyVar Freq = DesignVar(5.6674/(2*pi));
AnyFolder JointsAndDrivers = {
AnyKinEqFourierDriver Trunk_PelvisPosX_Pos_0 = {
Type = CosSin;
Freq = ..Freq;
AnyKinMeasure &m = Main.HumanModel.BodyModel.Main.HumanModel.BodyModel.Interface.Trunk.PelvisPosX;
AnyVar a0_offset ??= DesignVar(0.0);
A = {{ 0 + a0_offset, 0.000740, -0.00194, -1.8560e-05, 1.7458e-05, -4.8094e-05, }};
B = {{ 0, 0.00068, 0.00044, -0.000153, -0.000205, -1.02480e-07, }};
};
...
Creating Complete Human Models
You can also generate a complete stand-alone AnyBody model:
# Generate a complete model file with human model template
sillywalk.anybody.create_model_file(
predicted_data,
targetfile="complete_human_model.any",
create_human_model=True
)
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file sillywalk-1.0.0a1.tar.gz.
File metadata
- Download URL: sillywalk-1.0.0a1.tar.gz
- Upload date:
- Size: 9.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5664b7f177be4ce21652697b150223e5ecdab7a5841147a225402ecb57a6f404
|
|
| MD5 |
446eacbcea923b023010396d55d24ba4
|
|
| BLAKE2b-256 |
6c32719a55e7bc2dc0333a95c6c9df36c346789b99d4de05c9af3f0c234a961c
|
Provenance
The following attestation bundles were made for sillywalk-1.0.0a1.tar.gz:
Publisher:
build.yml on AnyBody-Research-Group/sillywalk
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
sillywalk-1.0.0a1.tar.gz -
Subject digest:
5664b7f177be4ce21652697b150223e5ecdab7a5841147a225402ecb57a6f404 - Sigstore transparency entry: 363335671
- Sigstore integration time:
-
Permalink:
AnyBody-Research-Group/sillywalk@e5ff0dfc09ccb27d3e0b546aec3d1be2063fc431 -
Branch / Tag:
refs/tags/1.0.0a1 - Owner: https://github.com/AnyBody-Research-Group
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
build.yml@e5ff0dfc09ccb27d3e0b546aec3d1be2063fc431 -
Trigger Event:
push
-
Statement type:
File details
Details for the file sillywalk-1.0.0a1-py3-none-any.whl.
File metadata
- Download URL: sillywalk-1.0.0a1-py3-none-any.whl
- Upload date:
- Size: 11.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eb353e6a2de9436261b10b4330a19ff713b11878660e6719177d50488e84d030
|
|
| MD5 |
14f6dc52c02b73429c25fb2fed412102
|
|
| BLAKE2b-256 |
898e532c83d6ab9fae35e8b120db2dd60b2d30899af8b598891d229119057ed6
|
Provenance
The following attestation bundles were made for sillywalk-1.0.0a1-py3-none-any.whl:
Publisher:
build.yml on AnyBody-Research-Group/sillywalk
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
sillywalk-1.0.0a1-py3-none-any.whl -
Subject digest:
eb353e6a2de9436261b10b4330a19ff713b11878660e6719177d50488e84d030 - Sigstore transparency entry: 363335705
- Sigstore integration time:
-
Permalink:
AnyBody-Research-Group/sillywalk@e5ff0dfc09ccb27d3e0b546aec3d1be2063fc431 -
Branch / Tag:
refs/tags/1.0.0a1 - Owner: https://github.com/AnyBody-Research-Group
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
build.yml@e5ff0dfc09ccb27d3e0b546aec3d1be2063fc431 -
Trigger Event:
push
-
Statement type: