Pioreactor dev blog #10 - happy little algae
🦠The past few weeks, we've been testing out the Pioreactor's ability to grow micro-aglae. Specifically, this involves using the Pioreactor as a photo-bioreactor. Outcome: success!
Algae have a very different set of living conditions compared to other organisms that we have tested in the Pioreactor. Algae require both light and COâ‚‚ to grow. We had the foresight to think of algae's unique growing requirements during development, hence the Pioreactor has additional pockets for LEDs in the main body. When these pockets are occupied by white-light LEDs, the Pioreactor turns from a bioreactor into a photo-bioreactor.
We are lucky to be working nearby an algae research centre: the Canadian Phycological Culture Centre (CPCC). They supplied us with a high quality culture, Chlorella Vulgaris, and BBM media for our experiments.
In phototrophic cultures (cultures that use light for energy), it's common to use a cyclic light source: often 16h with light, followed by 8h in dark. To implement this in the Pioreactor, we created a new LED automation. Just like how a user can write Python code to automate when and how dosing occurs, users can write Python code to automate how the LEDs should behave during an experiment. In our case, the light cycle was very simple: on for 16h, and then off for 8h. However, there are a few ways to make this automation smarter, each of these is possible with the Python API:
- gradual dimming and brightening of LEDs (instead of abrupt on/off)
- as the cell density increase, increasing the luminosity of the LEDs to try to keep light-per-cell approximately constant.
- changing the light cycle's period as the culture acclimatizes. Some cultures can be exposed to light constantly, but they need to be gradually introduced to these conditions.
Anyways, our experiment was going to use the simple 16h on / 8h off cycle only, with constant luminosity. The Python code we used to script the LED automation is below:
# -*- coding: utf-8 -*- | |
from __future__ import annotations | |
import signal | |
from pioreactor.automations import LEDAutomationContrib | |
from pioreactor.automations import events | |
from pioreactor.actions.led_intensity import LED_Channel | |
__plugin_summary__ = "An LED automation for ON/OFF cycles" | |
__plugin_version__ = "0.0.2" | |
__plugin_name__ = "Light Dark Cycle LED automation" | |
class LightDarkCycle(LEDAutomationContrib): | |
""" | |
Follows as h light / h dark cycle. Starts with light on. | |
""" | |
key: str = "light_dark_cycle" | |
published_settings: dict[str, dict] = { | |
"duration": {"datatype": "float", "settable": False, "unit": "min"}, # doesn't make sense to change duration. | |
"light_intensity": {"datatype": "float", "settable": True, "unit": "%"}, | |
"light_duration_hours": {"datatype": "int", "settable": True, "unit": "h"}, | |
"dark_duration_hours": {"datatype": "int", "settable": True, "unit": "h"}, | |
} | |
def __init__(self, light_intensity: float, light_duration_hours: int, dark_duration_hours: int, **kwargs): | |
super().__init__(**kwargs) | |
self.hours_online: int = -1 | |
self.light_active: bool = False | |
self.channels: list[LED_Channel] = [LED_Channel("B"), LED_Channel("C")] | |
self.set_light_intensity(light_intensity) | |
self.light_duration_hours = float(light_duration_hours) | |
self.dark_duration_hours = float(dark_duration_hours) | |
def trigger_leds(self, hours: int) -> events.Event: | |
cycle_duration: int = self.light_duration_hours + self.dark_duration_hours | |
if ((hours % cycle_duration) < self.light_duration_hours) and (not self.light_active): | |
self.light_active = True | |
for channel in self.channels: | |
self.set_led_intensity(channel, self.light_intensity) | |
return events.ChangedLedIntensity(f"{hours}h: turned on LEDs") | |
elif ((hours % cycle_duration) >= self.light_duration_hours) and (self.light_active): | |
self.light_active = False | |
for channel in self.channels: | |
self.set_led_intensity(channel, 0) | |
return events.ChangedLedIntensity(f"{hours}h: turned off LEDs") | |
else: | |
return events.NoEvent(f"{hours}h: no change") | |
def set_light_intensity(self, intensity): | |
self.light_intensity = float(intensity) | |
if self.light_active: | |
# update now! | |
for channel in self.channels: | |
self.set_led_intensity(channel, self.light_intensity) | |
def execute(self) -> events.Event: | |
self.hours_online += 1 | |
event = self.trigger_leds(self.hours_online) | |
return event |
The other requirement for algae is COâ‚‚. For our experiment, we decided to skip have a bubbler and relied just on the continuous stirring to incorporate COâ‚‚. Another way to increase the amount of COâ‚‚ available is to leave more headroom in the vial - so our experiments only used about 10ml of media (instead of the usual 15ml).
Results
Okay, so how did things go? Great! Here's a shot of the normalized optical density:
We replicated the experiment in another Pioreactor, this time with 20h light / 4h dark cycles.
So the Pioreactor can grow algae! However, there are some caveats:
- We observed that the LEDs we were using did emit some IR light that was being picked up by our optical density photodiodes. This meant that there was a "background" signal in the OD that was an artifact. A way to eliminate this is to add optical low-pass filters in front of the LEDs. However, this is costly, so we chose the simpler solution: turn off all LEDs while the OD signal is being read. The OD signal takes less than a second to read, and the frequency of OD reading can be reduced to make the effect less noticeable.
- Algae wants to be stirred at a fairly low RPM - between 60 and 100 is sufficient. Our current stirring can only go as low as ~200 RPM (we hope we can improve the range of our stirring when we revisit stirring during upcoming redesign). I don't know what the effect of higher stirring is on algae growth.
In conclusion, we are pretty happy with how our cultures have grown in the Pioreactor. Stay tuned for more!