Randomization - Basic Usage
In this section you will get to know how to use some basic concepts of Randomization.
Agenda:
- Using Sky Engine's Randomization vs the usual random()
- Distribution: Uniform vs Gaussian
- Synchronization: Unrestricted vs Equal
Scene setup
Let's use custom scene composer to set up the scene.
from skyrenderer.cases.utils import RandomizationBasicUsageSceneComposer
scene_composer = RandomizationBasicUsageSceneComposer(antialiasing_level=64)
scene_composer.setup_scene()
scene_composer.visualize()
2025-04-02 16:21:07,601 | skyrenderer.scene.renderer_context | INFO: Root paths: - root path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer - assets path: /dli/mount/assets - config path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer/config - gpu sources path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer/optix_sources/sources - cache path: /dli/mount/cache - ptx cache path: compiled_ptx/ptx - ocio path: ocio_configs/aces_1.2/config.ocio 2025-04-02 16:21:07,779 | skyrenderer.scene.renderer_context | WARNING: Light with light_id=light_LIGHT_NUL already exists in the scene and will be replace with a new one.There can only be a single light for a single node. 2025-04-02 16:21:07,781 | skyrenderer.scene.renderer_context | WARNING: Light with light_id=light2_LIGHT_NUL already exists in the scene and will be replace with a new one.There can only be a single light for a single node. 2025-04-02 16:21:10,763 | skyrenderer.utils.time_measurement | INFO: Setup time: 2.98 seconds 2025-04-02 16:21:10,872 | skyrenderer.utils.time_measurement | INFO: Setup time: 102 ms 2025-04-02 16:21:12,480 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.61 seconds 2025-04-02 16:21:30,886 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:21:30,888 | skyrenderer.utils.time_measurement | INFO: Render time: 18.41 seconds
This is our initial setup - a milk-filled bowl, with floating yummy Danish butter scones. The chef has baked them
to be identical, and arranged them with peculiar precision - to help further examples illustrate what's happening
more clearly.
Seriously though, they are on a grid and have 0,0,0 rotation applied.
We will now shake the bowl - randomize their positions, move them slightly off their grid spots, plunge them into
milk to a varying degree, and flip them around.
scene_composer.randomize_positions_classic()
scene_composer.visualize()
2025-04-02 16:21:31,711 | skyrenderer.utils.time_measurement | INFO: Setup time: 117 ms 2025-04-02 16:21:33,249 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.54 seconds 2025-04-02 16:21:49,826 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:21:49,827 | skyrenderer.utils.time_measurement | INFO: Render time: 16.58 seconds
That looks more natural.
Using Sky Engine's Randomization vs the usual random()
The example above and the collection of 3 examples below are randomized using Python's built-in random.random()
function - that's why we named this method "classic". Every time you invoke such randomization, results will be
totally different - as expected.
scene_composer.reduced_image_size()
renders = {}
scene_composer.randomize_positions_classic()
renders["classic #1"] = scene_composer.get_render()
scene_composer.randomize_positions_classic()
renders["classic #2"] = scene_composer.get_render()
scene_composer.randomize_positions_classic()
renders["classic #3"] = scene_composer.get_render()
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-02 16:21:50,544 | skyrenderer.utils.time_measurement | INFO: Setup time: 104 ms 2025-04-02 16:21:52,141 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.60 seconds 2025-04-02 16:21:57,608 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:21:57,609 | skyrenderer.utils.time_measurement | INFO: Render time: 5.47 seconds 2025-04-02 16:21:57,723 | skyrenderer.utils.time_measurement | INFO: Setup time: 103 ms 2025-04-02 16:21:59,252 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.53 seconds 2025-04-02 16:22:03,917 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:22:03,918 | skyrenderer.utils.time_measurement | INFO: Render time: 4.66 seconds 2025-04-02 16:22:04,027 | skyrenderer.utils.time_measurement | INFO: Setup time: 101 ms 2025-04-02 16:22:05,602 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.57 seconds 2025-04-02 16:22:10,922 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:22:10,923 | skyrenderer.utils.time_measurement | INFO: Render time: 5.32 seconds
These images will be also different when you re-run this entire tutorial file next time.
This "classic" method has been implemented for this tutorial only, to visualize the difference in behavior vs the
preferred way to randomize things, presented below. So don't worry about the specifics of the "classic" method.
Sky Engine relies on centralized control over everything that appears to be random. This is important to be able
to recreate renders with pixel-perfect precision between different runs of your rendering process, as long as the
scene setup remains unchanged. Sky Engine will make sure to draw the same exact "random" values as it did the
first time around, in the same order, in all the places you specified as random-driven, no matter how elaborate
your scene is.
Here we will create 3 images, each randomized with the preferred method.
scene_composer.reset_positions()
renders = {}
scene_composer.randomize_positions()
renders["preferred #1"] = scene_composer.get_render()
scene_composer.randomize_positions()
renders["preferred #2"] = scene_composer.get_render()
scene_composer.randomize_positions()
renders["preferred #3"] = scene_composer.get_render()
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-02 16:22:11,393 | skyrenderer.utils.time_measurement | INFO: Setup time: 108 ms 2025-04-02 16:22:12,935 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.54 seconds 2025-04-02 16:22:18,384 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:22:18,385 | skyrenderer.utils.time_measurement | INFO: Render time: 5.45 seconds 2025-04-02 16:22:18,501 | skyrenderer.utils.time_measurement | INFO: Setup time: 109 ms 2025-04-02 16:22:20,037 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.54 seconds 2025-04-02 16:22:24,757 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:22:24,758 | skyrenderer.utils.time_measurement | INFO: Render time: 4.72 seconds 2025-04-02 16:22:24,963 | skyrenderer.utils.time_measurement | INFO: Setup time: 199 ms 2025-04-02 16:22:26,512 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.55 seconds 2025-04-02 16:22:31,216 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-02 16:22:31,217 | skyrenderer.utils.time_measurement | INFO: Render time: 4.70 seconds
Randomization - Basic Usage
In this section you will get to know how to use some basic concepts of Randomization.
Agenda:
- Using Sky Engine's Randomization vs the usual random()
- Distribution: Uniform vs Gaussian
- Synchronization: Unrestricted vs Equal
Scene setup
Let's use custom scene composer to set up the scene.
from skyrenderer.cases.utils import RandomizationBasicUsageSceneComposer
scene_composer = RandomizationBasicUsageSceneComposer(antialiasing_level=64)
scene_composer.setup_scene()
scene_composer.visualize()
2025-04-03 08:37:46,178 | skyrenderer.scene.renderer_context | INFO: Root paths: - root path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer - assets path: /dli/mount/assets - config path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer/config - gpu sources path: /home/skyengine/anaconda/lib/python3.6/site-packages/skyrenderer/optix_sources/sources - cache path: /dli/mount/cache - ptx cache path: compiled_ptx/ptx - ocio path: ocio_configs/aces_1.2/config.ocio 2025-04-03 08:37:46,471 | skyrenderer.scene.renderer_context | WARNING: Light with light_id=light_LIGHT_NUL already exists in the scene and will be replace with a new one.There can only be a single light for a single node. 2025-04-03 08:37:46,476 | skyrenderer.scene.renderer_context | WARNING: Light with light_id=light2_LIGHT_NUL already exists in the scene and will be replace with a new one.There can only be a single light for a single node. 2025-04-03 08:37:50,167 | skyrenderer.utils.time_measurement | INFO: Setup time: 3.69 seconds 2025-04-03 08:37:50,388 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:37:52,340 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.95 seconds 2025-04-03 08:38:09,772 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:38:09,774 | skyrenderer.utils.time_measurement | INFO: Render time: 17.43 seconds
This is our initial setup - a milk-filled bowl, with floating yummy Danish butter scones. The chef has baked them
to be identical, and arranged them with peculiar precision - to help further examples illustrate what's happening
more clearly.
Seriously though, they are on a grid and have 0,0,0 rotation applied.
We will now shake the bowl - randomize their positions, move them slightly off their grid spots, plunge them into
milk to a varying degree, and flip them around.
scene_composer.randomize_positions_classic()
scene_composer.visualize()
2025-04-03 08:38:10,717 | skyrenderer.utils.time_measurement | INFO: Setup time: 210 ms 2025-04-03 08:38:12,426 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:38:29,953 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:38:29,955 | skyrenderer.utils.time_measurement | INFO: Render time: 17.53 seconds
That looks more natural.
Using Sky Engine's Randomization vs the usual random()
The example above and the collection of 3 examples below are randomized using Python's built-in random.random()
function - that's why we named this method "classic". Every time you invoke such randomization, results will be
totally different - as expected.
scene_composer.reduced_image_size()
renders = {}
scene_composer.randomize_positions_classic()
renders["classic #1"] = scene_composer.get_render()
scene_composer.randomize_positions_classic()
renders["classic #2"] = scene_composer.get_render()
scene_composer.randomize_positions_classic()
renders["classic #3"] = scene_composer.get_render()
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-03 08:38:30,802 | skyrenderer.utils.time_measurement | INFO: Setup time: 190 ms 2025-04-03 08:38:32,479 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.68 seconds 2025-04-03 08:38:37,828 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:38:37,830 | skyrenderer.utils.time_measurement | INFO: Render time: 5.35 seconds 2025-04-03 08:38:38,029 | skyrenderer.utils.time_measurement | INFO: Setup time: 188 ms 2025-04-03 08:38:39,694 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.66 seconds 2025-04-03 08:38:44,169 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:38:44,171 | skyrenderer.utils.time_measurement | INFO: Render time: 4.48 seconds 2025-04-03 08:38:44,374 | skyrenderer.utils.time_measurement | INFO: Setup time: 190 ms 2025-04-03 08:38:46,039 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.66 seconds 2025-04-03 08:38:50,503 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:38:50,504 | skyrenderer.utils.time_measurement | INFO: Render time: 4.46 seconds
These images will be also different when you re-run this entire tutorial file next time.
This "classic" method has been implemented for this tutorial only, to visualize the difference in behavior vs the
preferred way to randomize things, presented below. So don't worry about the specifics of the "classic" method.
Sky Engine relies on centralized control over everything that appears to be random. This is important to be able
to recreate renders with pixel-perfect precision between different runs of your rendering process, as long as the
scene setup remains unchanged. Sky Engine will make sure to draw the same exact "random" values as it did the
first time around, in the same order, in all the places you specified as random-driven, no matter how elaborate
your scene is.
Here we will create 3 images, each randomized with the preferred method.
scene_composer.reset_positions()
renders = {}
scene_composer.randomize_positions()
renders["preferred #1"] = scene_composer.get_render()
scene_composer.randomize_positions()
renders["preferred #2"] = scene_composer.get_render()
scene_composer.randomize_positions()
renders["preferred #3"] = scene_composer.get_render()
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-03 08:38:51,067 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:38:52,926 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.86 seconds 2025-04-03 08:38:58,550 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:38:58,552 | skyrenderer.utils.time_measurement | INFO: Render time: 5.62 seconds 2025-04-03 08:38:58,761 | skyrenderer.utils.time_measurement | INFO: Setup time: 201 ms 2025-04-03 08:39:00,514 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.75 seconds 2025-04-03 08:39:05,224 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:05,227 | skyrenderer.utils.time_measurement | INFO: Render time: 4.71 seconds 2025-04-03 08:39:05,437 | skyrenderer.utils.time_measurement | INFO: Setup time: 203 ms 2025-04-03 08:39:07,133 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.69 seconds 2025-04-03 08:39:12,638 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:12,640 | skyrenderer.utils.time_measurement | INFO: Render time: 5.50 seconds
You can see all images are identical, despite us having called the randomization 3 separate times. If you're
feeling adventurous, you can save these images to disk and analyze them in your favorite graphics editor to
confirm they're perfect copies, down to RGB values of each pixel. (Except for the label fonts, of course.)
In order to trigger different randomization states, you can use the frame number param, which more or less
directly affects the seeding of the entire randomization system.
We will render 3 pairs of images to demonstrate how randomized positions change with each frame number, while at
the same remain identical if you re-use the same frame number again.
renders = {}
scene_composer.randomize_positions()
renders["frame=1"] = scene_composer.get_render(frame=1)
renders["frame=2"] = scene_composer.get_render(frame=2)
renders["frame=3"] = scene_composer.get_render(frame=3)
renders["frame=1 again"] = scene_composer.get_render(frame=1)
renders["frame=2 again"] = scene_composer.get_render(frame=2)
renders["frame=3 again"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(1000, 1500), font_scale=2)
2025-04-03 08:39:13,080 | skyrenderer.utils.time_measurement | INFO: Setup time: 214 ms 2025-04-03 08:39:14,777 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.70 seconds 2025-04-03 08:39:20,288 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:39:20,290 | skyrenderer.utils.time_measurement | INFO: Render time: 5.51 seconds 2025-04-03 08:39:20,496 | skyrenderer.utils.time_measurement | INFO: Setup time: 199 ms 2025-04-03 08:39:22,192 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.70 seconds 2025-04-03 08:39:27,433 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:27,435 | skyrenderer.utils.time_measurement | INFO: Render time: 5.24 seconds 2025-04-03 08:39:27,643 | skyrenderer.utils.time_measurement | INFO: Setup time: 201 ms 2025-04-03 08:39:29,361 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.72 seconds 2025-04-03 08:39:33,846 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:33,848 | skyrenderer.utils.time_measurement | INFO: Render time: 4.49 seconds 2025-04-03 08:39:34,060 | skyrenderer.utils.time_measurement | INFO: Setup time: 204 ms 2025-04-03 08:39:35,807 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.75 seconds 2025-04-03 08:39:41,538 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:41,539 | skyrenderer.utils.time_measurement | INFO: Render time: 5.73 seconds 2025-04-03 08:39:41,750 | skyrenderer.utils.time_measurement | INFO: Setup time: 203 ms 2025-04-03 08:39:43,461 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:39:48,917 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:48,919 | skyrenderer.utils.time_measurement | INFO: Render time: 5.46 seconds 2025-04-03 08:39:49,130 | skyrenderer.utils.time_measurement | INFO: Setup time: 204 ms 2025-04-03 08:39:50,799 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.67 seconds 2025-04-03 08:39:56,143 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:39:56,144 | skyrenderer.utils.time_measurement | INFO: Render time: 5.34 seconds
Note that we called the "randomize" method only once before the first image. You don't need to manually call it
before each intended state change, just changing the frame number alone will trigger a cascade of re-randomization
throughout the system.
How to achieve this outside of this tutorial?
Behind the scenes we put all scones into a dict. Feel free to inspect its contents.
all_scones = scene_composer.all_scones
One example of a Provider which will do all the randomizations for you, is SimpleTransformProvider. Its params are
self-explanatory. We will iterate over all scone nodes and apply various random value ranges to their
transformations. Try your own values:
from skyrenderer.basic_types.provider import SimpleTransformProvider
from skyrenderer.basic_types.transform_path.transform_path import RotationMode
stp = SimpleTransformProvider(
scene_composer.renderer_context,
translation_range_x=(-0.015, 0.015),
translation_range_y=(-0.02, 0.05),
translation_range_z=(-0.015, 0.015),
rotation_range_x=(-180, 180),
rotation_range_y=(-180, 180),
rotation_range_z=(-180, 180),
rotation_mode=RotationMode.DEGREES,
scale_range_x=(0.3, 1.5),
scale_range_y=(0.4, 1.5),
scale_range_z=(0.4, 2),
)
for (scone, _) in all_scones.values():
scone.modify_locus_definition(transform_provider=stp)
renders = {}
renders["frame=1"] = scene_composer.get_render(frame=1)
renders["frame=2"] = scene_composer.get_render(frame=2)
renders["frame=3"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-03 08:39:57,392 | skyrenderer.utils.time_measurement | INFO: Setup time: 303 ms 2025-04-03 08:39:59,085 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.69 seconds 2025-04-03 08:40:04,205 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:04,210 | skyrenderer.utils.time_measurement | INFO: Render time: 5.12 seconds 2025-04-03 08:40:04,446 | skyrenderer.utils.time_measurement | INFO: Setup time: 229 ms 2025-04-03 08:40:06,277 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.83 seconds 2025-04-03 08:40:12,222 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:12,224 | skyrenderer.utils.time_measurement | INFO: Render time: 5.95 seconds 2025-04-03 08:40:12,432 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:40:14,119 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.69 seconds 2025-04-03 08:40:19,664 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:19,666 | skyrenderer.utils.time_measurement | INFO: Render time: 5.55 seconds
Distribution: Uniform vs Gaussian
When Sky Engine randomizes various aspects of your scene, it consults a Distribution Strategy - ability to control
the probability of certain values being drawn more likely than others from a given range. In the examples above we
have used Uniform Distribution - all values in all ranges were equally likely to be picked.
Let's see what happens when we change it to Gaussian.
from skyrenderer.randomization.strategy.distribution import UniformDistribution, RelativeGaussianDistribution
renders = {}
scene_composer.randomize_with_distribution(distribution=UniformDistribution())
renders["uniform #1"] = scene_composer.get_render(frame=1)
renders["uniform #2"] = scene_composer.get_render(frame=2)
renders["uniform #3"] = scene_composer.get_render(frame=3)
scene_composer.randomize_with_distribution(
distribution=RelativeGaussianDistribution(relative_mu=0.5, relative_sigma=0.07)
)
renders["gaussian #1"] = scene_composer.get_render(frame=1)
renders["gaussian #2"] = scene_composer.get_render(frame=2)
renders["gaussian #3"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(1000, 1500), font_scale=2)
2025-04-03 08:40:20,395 | skyrenderer.utils.time_measurement | INFO: Setup time: 204 ms 2025-04-03 08:40:22,094 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.70 seconds 2025-04-03 08:40:26,810 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:26,812 | skyrenderer.utils.time_measurement | INFO: Render time: 4.72 seconds 2025-04-03 08:40:27,021 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:40:28,739 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.72 seconds 2025-04-03 08:40:34,404 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:34,406 | skyrenderer.utils.time_measurement | INFO: Render time: 5.67 seconds 2025-04-03 08:40:34,616 | skyrenderer.utils.time_measurement | INFO: Setup time: 203 ms 2025-04-03 08:40:36,292 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.67 seconds 2025-04-03 08:40:41,098 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:41,099 | skyrenderer.utils.time_measurement | INFO: Render time: 4.81 seconds 2025-04-03 08:40:41,310 | skyrenderer.utils.time_measurement | INFO: Setup time: 204 ms 2025-04-03 08:40:43,064 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.75 seconds 2025-04-03 08:40:48,011 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:48,012 | skyrenderer.utils.time_measurement | INFO: Render time: 4.95 seconds 2025-04-03 08:40:48,222 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:40:49,930 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:40:55,762 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:40:55,763 | skyrenderer.utils.time_measurement | INFO: Render time: 5.83 seconds 2025-04-03 08:40:55,973 | skyrenderer.utils.time_measurement | INFO: Setup time: 202 ms 2025-04-03 08:40:57,654 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.68 seconds 2025-04-03 08:41:02,592 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:02,594 | skyrenderer.utils.time_measurement | INFO: Render time: 4.94 seconds
You can see the Uniform row shows all kinds of rotations, while in the Gaussian row most of the scones tend to be
raised upward. We prepared this example to allow only one axis of rotation, and within range of 0-180 degrees,
hence the Gaussian curve concentrated (sigma=0.07 - thin spike) in the middle of that range (mu=0.5), so it picked
values closer to 90 degrees. You can experiment with "mu" above to change the scones' tilt to the left or right,
and "sigma" to increase the variability in the scene.
How to achieve this outside of this tutorial?
You need to prepare a DrawingStrategy which consists of InputStrategies - one per each affected property that is
to be modified for a given node. In this case the DrawingStrategy will be passed into modify_locus_definition()
from skyrenderer.randomization.strategy.input_drawing_strategy import (
InputDrawingStrategy,
DrawingRule,
RelativeGaussianRandomInput,
UniformRandomInput,
)
from skyrenderer.randomization.strategy.drawing_strategy import DrawingStrategy
input_strategy = InputDrawingStrategy(drawing_rule=DrawingRule.RANDOM, distribution=UniformDistribution()) # or
input_strategy = UniformRandomInput() # shorthand # or
input_strategy = InputDrawingStrategy(
drawing_rule=DrawingRule.RANDOM,
distribution=RelativeGaussianDistribution(relative_mu=0.75, relative_sigma=0.2),
) # or
input_strategy = RelativeGaussianRandomInput(relative_mu=0.75, relative_sigma=0.2) # shorthand
drawing_strategy = DrawingStrategy(
scene_composer.renderer_context,
inputs_strategies={"rotation_y": input_strategy, "translation_y": input_strategy},
)
In this example the images will show 3 different takes on such scenario:
- most scones will be elevated to about 1cm above the milk surface (mu=0.75 on the 4cm range); however, wider
sigma allows some of the scones to sink, and others to rise even higher - most scones will be rotated horizontally to about -135 degrees away from their starting position - 0 degrees is
when serrated side faces the camera ("rotation_range_y" 0~-180deg * "relative_mu" 0.75 = -135deg); while wider
sigma allows significant +/- deviation from -135, still the scones are mostly facing away from us
stp = SimpleTransformProvider(
scene_composer.renderer_context,
translation_range_y=(-0.02, 0.02),
rotation_range_y=(0, -180),
rotation_mode=RotationMode.DEGREES,
)
for (scone, _) in all_scones.values():
scone.modify_locus_definition(
transform_provider=stp,
strategy=drawing_strategy,
)
renders = {}
renders["frame=1"] = scene_composer.get_render(frame=1)
renders["frame=2"] = scene_composer.get_render(frame=2)
renders["frame=3"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-03 08:41:03,828 | skyrenderer.utils.time_measurement | INFO: Setup time: 300 ms 2025-04-03 08:41:05,541 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:41:10,867 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 0 ms 2025-04-03 08:41:10,868 | skyrenderer.utils.time_measurement | INFO: Render time: 5.32 seconds 2025-04-03 08:41:11,076 | skyrenderer.utils.time_measurement | INFO: Setup time: 201 ms 2025-04-03 08:41:12,739 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.66 seconds 2025-04-03 08:41:17,814 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:17,815 | skyrenderer.utils.time_measurement | INFO: Render time: 5.08 seconds 2025-04-03 08:41:18,035 | skyrenderer.utils.time_measurement | INFO: Setup time: 210 ms 2025-04-03 08:41:19,768 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.73 seconds 2025-04-03 08:41:24,997 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:24,999 | skyrenderer.utils.time_measurement | INFO: Render time: 5.23 seconds
More Distributions
Learn about other types of Distributions available in Sky Engine in the following tutorial:
RANDOMIZATION_RandomizationAdvancedDistribution
Synchronization: Unrestricted vs Equal
You can synchronize object properties if you want multiple objects to appear "interlocked" to each other's state.
You can synchronize some properties, while keeping other properties freely randomized. See the example below where
we compare free rotation with synchronized rotation:
from skyrenderer.randomization.strategy.synchronization import Synchronization
renders = {}
scene_composer.randomize_with_synchronization(sync_type=Synchronization.UNRESTRICTED)
renders["unrestricted #1"] = scene_composer.get_render(frame=1)
renders["unrestricted #2"] = scene_composer.get_render(frame=2)
renders["unrestricted #3"] = scene_composer.get_render(frame=3)
scene_composer.randomize_with_synchronization(sync_type=Synchronization.EQUAL)
renders["equal #1"] = scene_composer.get_render(frame=1)
renders["equal #2"] = scene_composer.get_render(frame=2)
renders["equal #3"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(1000, 1500), font_scale=2)
2025-04-03 08:41:25,695 | skyrenderer.utils.time_measurement | INFO: Setup time: 205 ms 2025-04-03 08:41:27,377 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.68 seconds 2025-04-03 08:41:33,555 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:33,557 | skyrenderer.utils.time_measurement | INFO: Render time: 6.18 seconds 2025-04-03 08:41:33,767 | skyrenderer.utils.time_measurement | INFO: Setup time: 201 ms 2025-04-03 08:41:35,508 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.74 seconds 2025-04-03 08:41:41,003 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:41,005 | skyrenderer.utils.time_measurement | INFO: Render time: 5.50 seconds 2025-04-03 08:41:41,217 | skyrenderer.utils.time_measurement | INFO: Setup time: 204 ms 2025-04-03 08:41:42,923 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:41:49,182 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:49,184 | skyrenderer.utils.time_measurement | INFO: Render time: 6.26 seconds 2025-04-03 08:41:49,391 | skyrenderer.utils.time_measurement | INFO: Setup time: 198 ms 2025-04-03 08:41:51,062 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.67 seconds 2025-04-03 08:41:57,033 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:41:57,035 | skyrenderer.utils.time_measurement | INFO: Render time: 5.97 seconds 2025-04-03 08:41:57,246 | skyrenderer.utils.time_measurement | INFO: Setup time: 203 ms 2025-04-03 08:41:58,980 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.73 seconds 2025-04-03 08:42:05,022 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:42:05,024 | skyrenderer.utils.time_measurement | INFO: Render time: 6.04 seconds 2025-04-03 08:42:05,231 | skyrenderer.utils.time_measurement | INFO: Setup time: 200 ms 2025-04-03 08:42:06,964 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.73 seconds 2025-04-03 08:42:12,533 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:42:12,535 | skyrenderer.utils.time_measurement | INFO: Render time: 5.57 seconds
You can see the Unrestricted row shows all kinds of rotations, while in the Equal all scones in a given frame have
exactly the same, randomly picked rotation. At the same time, in both rows the vertical translation is freely
randomized - it was not included in the synchronization when preparing the bottom row, only rotation was. If you
look closely, in each vertical pair of images the distance of each corresponding scone from the milk's surface, is
the same. Only rotations differ. This is caused by using the same frame number for top and bottom image in each
pair - exactly the same random values have been picked for each scone's vertical translation, in the same order.
How to achieve this outside of this tutorial?
You need to prepare a DrawingStrategy which consists of InputStrategies - one per each affected property that is
to be synchronized between nodes. Only the properties indicated in inputs_strategies dict will be synchronized.
In this case the DrawingStrategy will be passed into modify_locus_definition()
from skyrenderer.randomization.strategy.synchronization import SynchronizationDescription
from skyrenderer.randomization.strategy.input_drawing_strategy import EqualInStrategyInput
synchronization = SynchronizationDescription(in_strategy=Synchronization.EQUAL)
input_strategy = InputDrawingStrategy(
drawing_rule=DrawingRule.RANDOM, distribution=UniformDistribution(), synchronization=synchronization
) # or
input_strategy = EqualInStrategyInput() # shorthand
drawing_strategy = DrawingStrategy(
scene_composer.renderer_context,
inputs_strategies={
"scale_x": input_strategy,
"scale_y": input_strategy,
"scale_z": input_strategy,
},
)
In this example each frame contains sets of identically scaled scones (all round, all elongated, etc.) while their
elevation and rotation is freely randomized.
stp = SimpleTransformProvider(
scene_composer.renderer_context,
rotation_range_x=(0, 360),
rotation_range_y=(0, 360),
rotation_range_z=(0, 360),
rotation_mode=RotationMode.DEGREES,
translation_range_x=(-0.015, 0.015),
translation_range_y=(0, 0.1),
translation_range_z=(-0.015, 0.015),
scale_range_x=(0.5, 3),
scale_range_y=(0.8, 2),
scale_range_z=(0.8, 3),
)
for (scone, _) in all_scones.values():
scone.modify_locus_definition(
transform_provider=stp,
strategy=drawing_strategy,
)
renders = {}
renders["frame=1"] = scene_composer.get_render(frame=1)
renders["frame=2"] = scene_composer.get_render(frame=2)
renders["frame=3"] = scene_composer.get_render(frame=3)
scene_composer.visualize_grid_desc(renders, shape=(500, 1500), font_scale=2)
2025-04-03 08:42:13,839 | skyrenderer.utils.time_measurement | INFO: Setup time: 315 ms 2025-04-03 08:42:15,625 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.78 seconds 2025-04-03 08:42:22,356 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:42:22,358 | skyrenderer.utils.time_measurement | INFO: Render time: 6.73 seconds 2025-04-03 08:42:22,583 | skyrenderer.utils.time_measurement | INFO: Setup time: 216 ms 2025-04-03 08:42:24,313 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.73 seconds 2025-04-03 08:42:31,130 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:42:31,132 | skyrenderer.utils.time_measurement | INFO: Render time: 6.82 seconds 2025-04-03 08:42:31,346 | skyrenderer.utils.time_measurement | INFO: Setup time: 207 ms 2025-04-03 08:42:33,061 | skyrenderer.utils.time_measurement | INFO: Context update time: 1.71 seconds 2025-04-03 08:42:38,780 | skyrenderer.utils.time_measurement | INFO: Key points calculation time: 1 ms 2025-04-03 08:42:38,782 | skyrenderer.utils.time_measurement | INFO: Render time: 5.72 seconds
Summary
In this section you have learnt:
- It is recommended to use Sky Engine -provided randomization facilities instead of typical random() functions.
- Sky Engine randomization makes it easy to control, manage, recreate and synchronize random states.