This shows you the differences between two versions of the page.
Both sides previous revision Previous revision Next revision | Previous revision | ||
bdev:minithermocycler [2020/10/22 18:27] richard |
bdev:minithermocycler [2021/08/16 22:22] (current) |
||
---|---|---|---|
Line 1: | Line 1: | ||
+ | |||
+ | Example data demonstrating the PCR works | ||
+ | |||
+ | |||
+ | |||
+ | {{ : | ||
+ | |||
+ | |||
+ | |||
+ | {{ : | ||
+ | |||
+ | |||
+ | |||
+ | {{ : | ||
+ | |||
+ | Green channel was plotted to compare. | ||
+ | Heat conditions: | ||
+ | * 95C 15 seconds | ||
+ | * 60C 30 seconds | ||
+ | * 40 cycles | ||
+ | |||
+ | Samples | ||
+ | - beta-actin - PCR with taqman probe | ||
+ | - blank - water control with master mix | ||
+ | |||
+ | |||
+ | Note: after 30 cycles signal was visable | ||
+ | |||
+ | |||
+ | {{ : | ||
+ | |||
+ | |||
+ | |||
+ | |||
Steps for building a mini thermocycler | Steps for building a mini thermocycler | ||
Line 7: | Line 41: | ||
- | Custom Parts: | + | Custom Parts:{{ : |
|1. Milling thermoblock. It costs $100 to make this. {{ : | |1. Milling thermoblock. It costs $100 to make this. {{ : | ||
|2. Lid with 2 3mm set screws {{ : | |2. Lid with 2 3mm set screws {{ : | ||
- | |3. Block holder {{ :bdev:thermoblock_holder_1inch_half_1_.stl |}}|{{ : | + | |3. Block holder {{ :bdev:hermoblock_holder_1inch_half.stl |}}|{{ : |
|4. Extrusion connects {{ : | |4. Extrusion connects {{ : | ||
|5. Brd base {{ : | |5. Brd base {{ : | ||
|6. PCB board {{ : | |6. PCB board {{ : | ||
+ | |7. Single tip imager {{ : | ||
- | Off the shelf ($43) + assembled brd is around $20 a unit: | + | Off the shelf ($43) + assembled brd is around $20 a unit + pipetters $20: |
|1. Power supply| {{ : | |1. Power supply| {{ : | ||
|2. Peltier cooling system | {{ : | |2. Peltier cooling system | {{ : | ||
Line 24: | Line 59: | ||
|5. Heater/ | |5. Heater/ | ||
|6. Arduino Nano | {{ : | |6. Arduino Nano | {{ : | ||
+ | |||
+ | |||
+ | < | ||
+ | |||
+ | import time, | ||
+ | from datetime import datetime | ||
+ | import json,re | ||
+ | import serial | ||
+ | import subprocess | ||
+ | |||
+ | |||
+ | |||
+ | def timestamp(gg): | ||
+ | now = datetime.now() # current date and time | ||
+ | | ||
+ | | ||
+ | | ||
+ | |||
+ | def openport(prt): | ||
+ | try: | ||
+ | ser = serial.Serial(prt, | ||
+ | | ||
+ | except: | ||
+ | | ||
+ | return ser | ||
+ | |||
+ | |||
+ | def whatstheports(): | ||
+ | | ||
+ | output = subprocess.check_output(" | ||
+ | b = re.split(" | ||
+ | if len(b)> | ||
+ | for i in b: | ||
+ | if re.match(' | ||
+ | pass | ||
+ | else: | ||
+ | try: | ||
+ | dada = (re.sub(' | ||
+ | if re.match(' | ||
+ | pass | ||
+ | else: | ||
+ | #ser = serial.Serial('/ | ||
+ | ser = serial.Serial(dada, | ||
+ | time.sleep(2) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.2) | ||
+ | resp = (ser.readlines()[0].decode()) | ||
+ | ser.close() | ||
+ | if re.match(' | ||
+ | return dada | ||
+ | except: | ||
+ | pass | ||
+ | else: | ||
+ | | ||
+ | | ||
+ | print(" | ||
+ | |||
+ | |||
+ | def read(): | ||
+ | port = whatstheports() | ||
+ | ser = openport(port) | ||
+ | ser.write(b' | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | print(temp) | ||
+ | |||
+ | |||
+ | |||
+ | def incubate(stemp): | ||
+ | print(" | ||
+ | print(" | ||
+ | port = whatstheports() | ||
+ | ser = openport(port) | ||
+ | ser.write(b' | ||
+ | ser.write(b' | ||
+ | time.sleep(0.2) | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | a = 1 | ||
+ | while a >0: | ||
+ | | ||
+ | | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | | ||
+ | |||
+ | |||
+ | def coolon(): | ||
+ | print(" | ||
+ | port = whatstheports() | ||
+ | ser = openport(port) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.5) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.5) | ||
+ | ser.write(b' | ||
+ | a = 1 | ||
+ | while a > 0: | ||
+ | | ||
+ | | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | | ||
+ | |||
+ | |||
+ | def flashon(): | ||
+ | port = whatstheports() | ||
+ | ser = openport(port) | ||
+ | ser.write(b' | ||
+ | return ser | ||
+ | |||
+ | def flashoff(ser): | ||
+ | ser.write(b' | ||
+ | ser.close() | ||
+ | |||
+ | def runcycle(dtemp, | ||
+ | print(int(cy)) | ||
+ | ccy = int(cy) | ||
+ | gg = open(" | ||
+ | timestamp(gg) | ||
+ | gg.write(" | ||
+ | | ||
+ | print(" | ||
+ | print(" | ||
+ | port = whatstheports() | ||
+ | ser = openport(port) | ||
+ | time.sleep(1) | ||
+ | # | ||
+ | for i in range(0, | ||
+ | | ||
+ | | ||
+ | | ||
+ | print(" | ||
+ | timestamp(gg) | ||
+ | gg.write(" | ||
+ | gg.close() | ||
+ | |||
+ | |||
+ | def cycle(dtemp, | ||
+ | print(" | ||
+ | print(" | ||
+ | gg.write(" | ||
+ | gg.write(" | ||
+ | ser.write(b' | ||
+ | ser.write(b' | ||
+ | time.sleep(0.2) | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | while temp < (int(dtemp)+1): | ||
+ | | ||
+ | | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | | ||
+ | | ||
+ | if temp == (int(dtemp)): | ||
+ | break | ||
+ | time.sleep(int(dtemptm)) | ||
+ | ser.write(b' | ||
+ | print(" | ||
+ | print(" | ||
+ | gg.write(" | ||
+ | gg.write(" | ||
+ | ser.write(b' | ||
+ | time.sleep(0.3) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.3) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.2) | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | while temp > (int(etemp)+1): | ||
+ | | ||
+ | | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | | ||
+ | | ||
+ | if temp == (int(etemp)): | ||
+ | break | ||
+ | ser.write(b' | ||
+ | time.sleep(0.3) | ||
+ | ser.write(b' | ||
+ | time.sleep(0.3) | ||
+ | print(" | ||
+ | gg.write(" | ||
+ | time.sleep(0.3) | ||
+ | ser.write(b' | ||
+ | while temp > (int(etemp)+1): | ||
+ | | ||
+ | | ||
+ | b = (ser.readlines()) | ||
+ | temp = float(b[0].decode().rstrip()) | ||
+ | | ||
+ | | ||
+ | gg.write(" | ||
+ | print(" | ||
+ | time.sleep(int(etemptm)) | ||
+ | |||
+ | |||
+ | |||
+ | input_var = input(" | ||
+ | if re.match(" | ||
+ | print(" | ||
+ | print(" | ||
+ | print(" | ||
+ | print(" | ||
+ | print(" | ||
+ | print(" | ||
+ | print(" | ||
+ | elif re.match(" | ||
+ | whatstheports() | ||
+ | elif re.match(" | ||
+ | read() | ||
+ | elif re.match(" | ||
+ | coolon() | ||
+ | elif re.match(" | ||
+ | [arg, temp] = re.split(' | ||
+ | incubate(temp) | ||
+ | elif re.match(" | ||
+ | [arg, dtemp, | ||
+ | runcycle(dtemp, | ||
+ | elif re.match(" | ||
+ | ser = flashon() | ||
+ | floff = input(" | ||
+ | flashoff(ser) | ||
+ | else: | ||
+ | print(" | ||
+ | </ | ||
+ | |||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | |||
+ | < | ||
+ | #include < | ||
+ | #include < | ||
+ | //#include < | ||
+ | Servo myservo; | ||
+ | |||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | |||
+ | #define THERMISTOR_PIN A7 | ||
+ | #define TEMPCONTROL 3 | ||
+ | #define FANPIN 5 | ||
+ | #define THERMOELECTRIC_COOL 6 | ||
+ | #define LED 9 | ||
+ | |||
+ | //Define the aggressive and conservative Tuning Parameters | ||
+ | |||
+ | double aggKp=150, aggKi=0.5, aggKd=0; | ||
+ | |||
+ | |||
+ | int thermoelectricval = 255; | ||
+ | int ledval = 255; | ||
+ | int tempcontrol = 3; | ||
+ | String command; | ||
+ | float currpos; | ||
+ | int celsius; | ||
+ | int levelstreamon; | ||
+ | double settemp = 0; | ||
+ | double Setpoint, Input, Output; | ||
+ | int tme = 250; | ||
+ | |||
+ | //Specify the links and initial tuning parameters | ||
+ | //PID myPID(& | ||
+ | PID myPID(& | ||
+ | |||
+ | #define NUMTEMPS 20 | ||
+ | short temptable[NUMTEMPS][2] = { | ||
+ | {1, 841}, | ||
+ | {54, 255}, | ||
+ | {107, 209}, | ||
+ | {160, 184}, | ||
+ | {213, 166}, | ||
+ | {266, 153}, | ||
+ | {319, 142}, | ||
+ | {372, 132}, | ||
+ | {425, 124}, | ||
+ | {478, 116}, | ||
+ | {531, 108}, | ||
+ | {584, 101}, | ||
+ | {637, 93}, | ||
+ | {690, 86}, | ||
+ | {743, 78}, | ||
+ | {796, 70}, | ||
+ | {849, 61}, | ||
+ | {902, 50}, | ||
+ | {955, 34}, | ||
+ | | ||
+ | }; | ||
+ | |||
+ | |||
+ | void setup() { | ||
+ | // put your setup code here, to run once: | ||
+ | Serial.begin(115200); | ||
+ | analogWrite(TEMPCONTROL, | ||
+ | analogWrite(FANPIN, | ||
+ | |||
+ | Setpoint = settemp; | ||
+ | //turn the PID on | ||
+ | myPID.SetMode(AUTOMATIC); | ||
+ | analogWrite(THERMOELECTRIC_COOL, | ||
+ | analogWrite(LED, | ||
+ | } | ||
+ | |||
+ | void loop() { | ||
+ | Input = read_temp(); | ||
+ | |||
+ | | ||
+ | | ||
+ | | ||
+ | | ||
+ | int rawvalue = analogRead(THERMISTOR_PIN); | ||
+ | float celsius = read_temp(); | ||
+ | int fahrenheit = (((celsius * 9) / 5) + 32); | ||
+ | |||
+ | if (levelstreamon == 1) { | ||
+ | | ||
+ | | ||
+ | } | ||
+ | |||
+ | | ||
+ | { | ||
+ | char c = Serial.read(); | ||
+ | if (c== ' | ||
+ | { | ||
+ | parseCommand(command); | ||
+ | command = ""; | ||
+ | } | ||
+ | else | ||
+ | { | ||
+ | command +=c; | ||
+ | } | ||
+ | } | ||
+ | | ||
+ | } | ||
+ | |||
+ | float parseCommand(String com) | ||
+ | { | ||
+ | |||
+ | if(com.equalsIgnoreCase(" | ||
+ | Serial.println(" | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | double aggKp=19.56, | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | aggKp = com.substring(5).toDouble(); | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | aggKi = com.substring(5).toDouble(); | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | aggKd = com.substring(5).toDouble(); | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | ledval = com.substring(11).toInt(); | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | int rawvalue = analogRead(THERMISTOR_PIN); | ||
+ | float celsius = read_temp(); | ||
+ | | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | // | ||
+ | thermoelectricval = com.substring(13).toInt(); | ||
+ | // | ||
+ | } | ||
+ | else if (com.substring(0, | ||
+ | // | ||
+ | settemp = com.substring(8).toInt(); | ||
+ | // | ||
+ | Setpoint = settemp; | ||
+ | } | ||
+ | else if(com.equalsIgnoreCase(" | ||
+ | strobingmic(ledval, | ||
+ | delay(10); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | float read_temp() | ||
+ | { | ||
+ | int rawtemp = analogRead(THERMISTOR_PIN); | ||
+ | float current_celsius = 0; | ||
+ | |||
+ | byte i; | ||
+ | for (i=1; i< | ||
+ | { | ||
+ | if (temptable[i][0] > rawtemp) | ||
+ | { | ||
+ | float realtemp | ||
+ | |||
+ | if (realtemp > 255) | ||
+ | realtemp = 255; | ||
+ | |||
+ | | ||
+ | |||
+ | | ||
+ | } | ||
+ | } | ||
+ | |||
+ | // Overflow: We just clamp to 0 degrees celsius | ||
+ | if (i == NUMTEMPS) | ||
+ | | ||
+ | |||
+ | | ||
+ | } | ||
+ | void strobingmic(int ledval, int tme){ | ||
+ | int strobtimer = 1; | ||
+ | while (strobtimer > 0 && Serial.available() == 0){ | ||
+ | analogWrite(LED, | ||
+ | delayMicroseconds(tme); | ||
+ | analogWrite(LED, | ||
+ | delayMicroseconds(tme); | ||
+ | strobtimer++; | ||
+ | } | ||
+ | | ||
+ | } | ||
+ | |||
+ | </ | ||