All you need to know about Robotics, AI and what to do with them.
A fair warning: Robotics does require certain knowleges. So when I say "take pliers",
I assume you know which end to hold. In other words,
Please read the disclaimer
Good news, everyone! (C)
I have implemented by-subscription access to some sections on this site. I hope this desperate measure will find your understanding.
Finally, my 2.5d navigation supports landmarks. So far, this is the easiest kind of them, one using aruco markers.
Those are great for indoor navigation, in a controlled environment. As this is a tutorial, I made it work outdoors, though I do not think it is the best possible solution: having 1 meter aruco boards all over your land is plain ugly. Unless it is an industrial environment, like a mining site.
Aruco were chosen because all computer vision work is already done in OpenCV, so I do not have to focus on recognition part. In future, i will add other types of landmarks, like road signs, hills, trees, houses, sun and stars and maybe, other robots.
Before, there was simply a link to a rather messy catalog. Now it is possible to navigate my robotics hardware articles using tree on the left.
Finally, a complete example of (one of many possible approaches to a) localization in 2.5D world. Robot is made aware of its position on a 2.5D surface, both in Gazebo simulator and in a faster home made one. Robot uses additional data from the map. And so on.
In a real life, fusing random subset of awailable sensors is not something you usually do, but as this is a tutorial, the goal is pretty ambitious: any combination of sensors should work for localization.
A long article with lots of images illustrating sensor fusion for our robot localization, as well as possible problems and workarounds.
Unlike most tutorials online, I have created a very unfriendly testing environment with slippage, large sensor errors etc. Resulting localization routine is less accurate but more robust and can survive situations when some sensors are suddenly not available.
As for "2d world" limitation, I am going to switch to 2.5d in the following sections.
This article adds proper simulation of wheel slippage to localization based on Unscented Kalman filter to ROS2 / Gazebo simulation.
This slippage includes both sliding due to slippery road and by-design slippade of some robots, for example, ones using 4 wheeled dif. drive.
Localization works properly both in a Gazebo simulation and in a simulation without Gazebo (which is much faster though not as advanced).
So far it is a simplified (2d world, no sensor fusion) version; I am going to gradually improve it in the following sections.
This article adds localization based on Unscented Kalman filter to ROS2 / Gazebo simulation. So far it is a simplified (2d world, no sensor fusion) version; I am going to gradually improve it in the following sections.
Sample code to illustrate the use of Kalman filter in robot localization. Multiple sensors, dynamic sensors configuration, merging sensor data and so on.
This is an introduction, with code explained. In the following sections I will implement it in ROS2.
In two words: the world has changed, the robot has changed, and we now have a First Person View tool to control any of multiple robots in a simulation.
In one of the previous sections we have created the teleop_gui, which was a rather primitive utility imitating the corresponding ROS2 command line tool, plus some UI. This tool is no longer supported (though the article is still there, so it can be used as a source of info).
Instead, a more advanced and universal dashboard to control robots in a multi-robot environment was created
The teleop_gui utility works with mouse (you can move the control knob imitating the joystick) and with arrows. Space bar is for STOP command. Robot moves and turns at different speed (depending on the position of a knob).
Parameters of the world and robot's materials changed, so friction is realistic and robot can handle rather ragged terrain:
The center of gravity was moved down, camera was moved up for better view, and all sensors were moved forward, just to keep them in the same place.
In the previous section, we learned how to create 2.5D landscapes from a topo map, turn them into a textured DAE file and use them in a Gazebo simulation.
However, topo maps have one huge "incompatibility" with the real world, which is called "roads". A road hhould be smooth, it should follow the landscepe, BUT it should be horizontal in the direction perpendicular to its orientation. In other words, while driving along the road, your car should tilt up or down, but not sideways:
When building roads, construction workers make them horizontal (in an "across" direction) so that a car doesn't slide off the road. The question is, how can we do it using topo map?
The answer is, we probably should not. Topo map is for smooth hills and valleys, while the road should be created in a way similar to what construction workers do: by removing part of the landscale and adding the road instead.
And of course, we should do it on a mesh, not at the topo map.
In this section, I am going to add this functionality to landscape generator (topo_3d.py utility). It will level up ground for the road, add road, stitch together the road and the rest of the landscape in a watertight (no holes) way and even add roads (darg bands) to the texture to make it look nicer.
Unfortunately, Nav2 utilities of ROS2 are mostly dealing with the flat world. In the same time, the real world has hills your robotic tractor has to plow, caves it has to explore, not to mention canyons it should stay away from.
In this post I am going to begin a rather long journey towards 2.5D navigation, and the first thing we need to do it to build 2.5D world. Traditionally, in Gazebo 3d meshes are used, and we even can pull textures on them... except the process of building these objects is far from straightforward. Just search the Internet forums to see lots of confused people. Let's create a tool that will help us building textured 2.5D meshes.
The utility takes a colored topo map as a source:
... and produces a DAE mesh with a texture mapped to it.
Imagine that you have parked you robot at a parking lot, next to a highway. There is a nice paved road connecting parking with a highway, and a ditch that is occupying the rest of space between them.
So far we used A Star Graph navigation algorithm on a road that had graph and if the initial position of a robot was outside the graph, well, we just used straight line navigation to get to the closest node of a graph.
There are few problems with this approach. First of all, if you drive from the parking lot to a road by a straight line, you will most likely end up in a ditch. Second, there is also a "last mile" problem, which is "drive there by a graph and then go to parking at your destination". Current implementation of our a-star navigation algorithm doesn't have this functionality. Third, we might want to drive along a highway, then do some cross-terrain driving, then highway again and so on, in other words, what if we need to randomly switch from graph based navigation to grid based navigation and back?
In the following project, a simple solution is implemented: robot uses a-star grid navigation to get to the road, and a-star graph navigation as soon as it is close to nodes (as soon as it is on a road).
The ROS2 project now supports navigation based on keepout map, using a-star algorithm. Should also work with multiple robots, but not tested yet (will be in the next iteration).
The ROS2 project now supports navigation through array of graph nodes, using a-star algorithm. Should also work with multiple robots, but not tested yet (will be in the next iteration).
A* algorithm is a faster alternative to Bellman-Ford and is widely used to find a shortest path in a graph. In this section I am going to provide two versions of code for two different scenarios of use.
The objective is to find the optimal path from start point to a destination point taking into consideration distance and road conditions (or "price") associated with a particular path (dirt roads vs highway or toll roads vs free ones).
A* algorithm is one of the most used for navigation, as it is reasonably fast, accurate and customizable. In this section, I am going to implement two versions of A-star algorithm, one for an "uncharted" land, and one for an existing road network.
Waypoints editor I have introduced earlier now changed into a full scale ROS2 map, waypoints, graph and keepout zones editor.
In ROS2, Path is a set of linear segments. As coordinates of the end of one segment are the same as coordinates of beginning of the next segment, it is convenient to store path as an array of points.
Path following code will figure out all the necessary turns based on coordinates of segments start and end points. A custom Path Follower class will then be used to navigate through the path.
As usual, supported both single and multi robot cases.
In this project I will add path visualization in RViz, both simple one (by publishing path) and more visual one, with markers:
Same can be done in Gazebo, plus a ground in Gazebo world can be turned into a video screen, with map, ddynamic routes and everything:
I am going to overwrite some Navigation (and not only) algorithms, because I do not like the way they are implemented in Nav2, and because I do not want to deal with black boxes I have little control over.
Let's create a fully functional robotic arm, make sure it can move to a specified position (forward and reverse kinematics) and pick up objects in Gazebo (now, THAT was a tough one!)
Let's create a project for multiple robots, and make sure we can visualize them both in RViz and Gazebo! Let's also make Nav2 algorithms work with them.
Creating map from an image, ROS2 Nav2, Waypoint Follower, Navigating with Keepout Zones, Navigating along given path, Utility to add waypoints to a map, Navigating to a battery charger, docking to a battery charger.
I use Nav2 packages to do all that - for now.
Often, when you close Gazebo, RViz doesn't close, or when you close RViz, Gazebo (either client or VERY often, server) keeps running, or when you press CTRL-C... same thing. Let's fix it.
We are going to add event handlers that, when Gazebo dies, kills RViz, and when RViz dies, kills Gazebo.
From this point on I am switching to ROS2 Galactic. The reason is, a signifficant part of navigation functionality is not implemented in Foxy.
In this section, we are going to finalize the design of our robot, so that it can use different planner and controller algorithms to perform navigation tasks.
Note that the launch file can now take bunch of parameters: file name for the world, file name for the map to load in RViz and so on. One of such parameters is slam, which can be True or False. Depending on that parameters, we either launch slam tools or not. In our particular case, we use this parameter to either create a lidar-based map and save it, or to load a pre-saved map in RViz.
This bot allows us to do pretty much the same as earlier version did (drive around and save the map), however, the entire system of launch files is now altered to make launch more flexible. We are going to start the changes here, and finish them in the navigation_bot_05 section (next one). Overall, the launch structure will now mimic what official ROS2 documentation has for sam_bot, except some problems are going to be fixed.
The robot that we are going to create in this section will be able to drive around (we use keyboard commands), and as it drives, the lidar will create a map. Then we will be able to save this mat to disk for future use.
Here are some examples of commonly used sensors: lidar, radar, RGB camera, depth camera, IMU, and GPS. To make things more standard, ROS2 provides the sensor_msgs package that defines the common sensor interfaces. Accordingly, users can use any sensor by any vendor as long as it follows the standard format in sensor_msgs.
As some examples of such messages, let's mention the sensor_msgs/LaserScan, sensor_msgs/PointCloud2, sensor_msgs/Range, and sensor_msgs/Image.
In addition to sensor_msgs package, there are also the radar_msgs and vision_msgs standard interfaces you should be aware of.
Let's add some sensors to a robot that we have created in a previous section. Particularly, we will add a lidar sensor and a depth camera (note: we are still following the official "sam_bot" tutorial, except, we do it step by step).
Wheel encoders, adding IMU sensor, using localization package.
This is an introductory chapter, it contains some theory. At the end you will get the robot, that is configured properly, so in the following sections we are going to use it as a base for navigation.
The theory is important though, as otherwise some details can easily be missed, so - regardless the fact that this code will be repeated and improved in later sections - I strongly recommend following this chapter and making sure the code works in your workspace, just for the sake of an exercise.
The robot I use in this section is based on a ROS2 official documentation, particularly, on a sam_bot. However, as always, there are some minor (or not so minor) problems I had to fix, plus sam_bot is written in a "one example covers all" style, which makes gradual learning difficult.
Instead, in the next few sections, I am going to create a simple bot, and add complexity in a step-by-step way. Result will be same as in an official tutorial, but learning will be much easier.
To make it clear: ROS2 offers not one but two solutions under the same "differential drive control": the differential drive itself and a "skid drive". Plus, we can also implement "Ackermann steering vehicle" using the ackermann_steering_controller.
The diff. drive controller that ROS2 has can be used to drive devices with two wheels and a cast wheel: just picture robotic vacuum cleaner. This is a traditional application of a differential drive: two wheels can rotate independently, so the robot can turn "on the spot", and to help it to keep balance, a passive "cast wheel" is used.
This is what we are going to do in this section.
An introductory article to a large list of ROS2 tutorials I am about to start. Correction: a practical introductory article.
In this post we are going to perform the image segmentation detecting a single quadcopter (one quadcopter per image) using a simple CNN. This aproach has its advantages: it is way faster than more complex YOLO, SSD or Faster-RCNN approaches. All we need is to show the image to a network and to get quadcopter's coordinates as an output.
During previous step we got a rather large dataset composed of pairs of images. Those images were uploaded to the Internet by people we do not know, and though they are supposed to contain coin's head in one image and tail in the other, we can not rule out a situation when we have two heads and no tail and vice versa. Also at the moment we have no idea which image contains head and which contains tail: this might be important when we feed data to our final classifier.
So let's write a program to distinguish heads from tails. It is a rather simple task, involving a convolutional neural network that is using transfer learning. Same way as before, we are going to use Google Colab environment, taking the advantage of a free video card they grant us an access to. We will store data on a Google Drive, so first thing we need is to allow Colab to access the Drive
During previous step we got a rather large dataset composed of pairs of images. Those images were uploaded to the Internet by people we do not know, and though they are supposed to contain coin's head in one image and tail in the other, we can not rule out a situation when we have two heads and no tail and vice versa. Also at the moment we have no idea which image contains head and which contains tail: this might be important when we feed data to our final classifier.
So let's write a program to distinguish heads from tails. It is a rather simple task, involving a convolutional neural network that is using transfer learning. Same way as before, we are going to use Google Colab environment, taking the advantage of a free video card they grant us an access to. We will store data on a Google Drive, so first thing we need is to allow Colab to access the Drive
In our first tutorial we used the image segmentation tool (YOLO) to find the coin (or coins) in the image and crop them. In our second tutorial, we have created a classifier to figure out if this particular cropped image contains "head" or "tail". Now it is time to learn to classify images by their value.
This is the 4th tutorial dedicated to coins classification. So far we were using the same approach: transfer learning CNN. And here we go again, we use it for classifying coins by year.
So why don't we classify it all in one pass: head-or-tail, value and year? All it takes is having a classifier with combined outputs, right? The reason is, we don't have large (and clean) enough dataset. Even with my 20K+ images, it will suffer for some years as there were only limited minting that year and I only have few images. So if I break them additionally for side/value/year... The net will undertrain.
The second reason is, we are NOT classifying coins yet! What we do is building a dataset for a "real" training! That's right, all we did so far, including current tutorial is sorting out a huge heap of images I got (automatically) from online, and marking them, assigning labels. To train a network for classification by values, we need, say, 100 images per coin. So a human (who else?) have to select about 400 images (1R, 2R, 5R, 10R). But if we wanted to train a net for value AND year, we would have to select, ideally, 100 coins per each year and value, which is 2000 images (Russian coins were chosen for this task because they have relatively short history, from 1997, but still, it is about 20 years).
Full cycle of development: from Keras Neural Network on Python to Android application on Google Play
The program attempts to determine the ripeness and sweetness of a watermelon: you knock on the fruit, it is analyzing the sound of it. I am going to perform a full development cycle, up to uploading the program to Google Play and improving it in an automated mode using user feedback.
Full cycle of development: from Keras Neural Network on Python to Android application on Google Play
In this tutorial i am going to walk through a Dog Breed Identifier: we are going to create and teach a Neural Network, then we will port it to Java for Android and publish on Google Play.
Peltier elements are known as a simple alternative to compression coolers and we can often see opinions that they can be used to produce deep freezers, going as far as -40C. This is... kind of true. However, there are some formidable challenges on this path making Pelier elements nearly useless for practical freezing applications.
This is part of my Cryonics project: i am going to check the claim of some Japanese companies that water freezing point can be affected by EM and electric fields. The Generator will be used to feed the coil, producing EM field of different frequencies. I am going to publish results here when they are available.
There are many ways of sending data between Arduino (or similar controller) and something else, and one of the most convenient is by using NRF24L01 transceiver module. It uses the 2.4 GHz frequency and it can operate with baud rates from 250 kbps up to 2 Mbps. The higher the baud rate, the shorter the distance, but if used in open space and with lower baud rate its range can reach up to 100 meters. Plus, there are also anthennas to increase it..
These are cheapest solutions on market, and I'll try explaining why they should not be used.
Sometimes we only have one analog port available on our controller, and still have to read data from multiple sensors. This simple chip ptovides a nice solution.
ESP8266 is one of the most populat chips for Internet of Things programming. Unfortunately, it only has one analog pin (A0), which is not enough in case you want to read multiple sensors. A common solution ot this problem is using a multiplexer: it has multiple inputs and you call (programmatically, in real time) tell it which one to read, or to put it differently, which one to conect to its single output.
In this tutorial we are going to use 74HC 4051 - a 8 channels multiplexer. We are going to create a server that provides a web page. This web page is similar to previous example: it allows sending commands to 8266 (turn light dione on/off), and receives thermistor-provided themperature data. The difference is, we now have two thermistors (and can increase it up to eight), so we use a multiplexer.
In this project we are going to create a prototype of a remotely accessible sensor (one so far, for multiple sensors, see the next tutorial).
In other words, you will be able to read sensor (thermistor, in this example) data, which means receiving data FROM server. You wil also be able to send commands to server, in our case it is going to be turning on and off the light diode, but same logic applies to controlling motors and so on.
Important note: we are going to create a web server that works with your modem within your LOCAL network. It can not be seen from the outside Internet. So you can use this approach for your IOT projects, but can not post data to the Web (a different approach exists). The "local" approach is generally safer, so when it comes to IOT, it should be considered first. In other words, this project is for sending data from the window in the second floow to a computer on a first floor, nothing else.
In this post we will build an ESP8266 based Web Server and walk through both the code and schematics. This tutorial is a step-by-step guide that shows how to build a standalone ESP8266 Web Server that controls a single output (one LED). This ESP8266 NodeMCU Web Server is mobile responsive and it can be accessed with any device with a browser in your local network.
Please pay attention to the last statement: we are going to create a web server that works with your modem within your LOCAL network. It can not be seen from the outside Internet. So you can use this approach for your IOT projects, but can not post data to the Web (a different approach exists, it is just... different).
In this tutorial we are going to read temperature from a simple 10K thermistor. By simple i mean, there are more advanced serial enabled devices... this is not one of them.
Thermistor is just a resistor that changes its value depending on the temperature. The idea of this project is in using diode bridge: we connect "+" (which is marked 3V) with "-" (the ground) through the resistor of a known value and a thermistor.
A "Blinker" program is a bare minimum to start with ESP8266. It allows you to turn a light diode on and off. Of course, to master this tutorial, you have to lear the board's pinouts, as well as installation and programming. In other words, if this subject is new for you, do not miss this tutorial.
We are going to use Arduino IDE
Arduino is a very popular controller, and its IDE is popular, too. So NodeMCU was
made compatible with it, and you can write Arduino code (just keep in mind that pin
numbers are different), and it will work.
Follow the link above to add NodeMCU support to your Arduino environment.
A very brief introduction to what it is and why is it so popular. By ESP8266 here and everywhere else I mean NodeMCU, as it has all the necessary extras to make it easier to use the controller.
An introduction to building a DIY robotic shoulder, far from complete. Often, people rush into assembling robotic arms, downloading ideas from the internet... Well, if you only want it to be able to lift a match box, that's fine. But for anything strong enough to lift a basket of water, you either need to forget about cheap plastic solutions (and steel is 100 times more expensive), or keep certain pointers in mind. Here we go.
Building a vaccuum cleaner involves more responcibility than one can possibly handle.
Will it work? Will it try to kill you? Will it fall in love with your lawn mover and run away?
In other words, one needs to stop and think hard. You know, like when a child wants a
doggy, and you tell him to get a rock first, to walk it, to feed it; and when, two
weeks later, the rock dies of starvation, you tell your kid: "no, see, you are just not ready".
And possibly, running a simulation or two might help.
Let's build a robotic vacuum cleanersimulator, one that can be used to figure out how
many sensors it really needs (three) and which room exploring algoritm is better. That's it. Simple?
Robotic Floor Washer DIY
In this tutorial I am going to build a fully working prototype of a robotic floor
washer. By "fully working" I mean that it is going to wash floor, instead of moving
dirt around like most robotic "moppers" do. While by "prototype" I mean it is
going to be the first step towards production-ready unit, but not a
production-ready unit yet.
Brainless Platform for a Robotic Vacuum Cleaner
When we hear "robotic vacuum cleaner", most of us would imagine a round thing with
a bumper, wondering around the room, controlled by a simple microprocessor and
acquiring the information from IR and Ultrasonic sensors as well as from the bumper
(which is nothing but a touch sensor).
Let's explore a different approach. Let's have no microprocessor and no sensors at all.
Still, the "platform" we are about to create will navigate around, finding its way from
under the furniture, and, given time, visiting every spot, cleaning it (if we install a
VC on the platform).
There is an important lesson for someone building robots: what can be done with advanced
electronics, can (not always) be also done using mechanics. How?
A very basic guide to building your own Roomba.
Prototyping a Robotic Vacuum Cleaner.
In this tutorial we are going to create a platform that can be extended to become a full
scale robotic vacuum cleaner. This particular platform as is will not clean anything:
it will move, use bump sensors to locate the walls and turn around to avoid them.
However, approaches we use are going to help you in case you need to create an
industrial prototype of the real thing.
A rather large article about using Triplet Loss, where I am trying to show what can be twisted and what the results will be.
Good luck.