Learn how to use Viam to make an LED blink with a Raspberry Pi.

First, you'll use the control interface on the Viam app to turn the LED on and off. Then, you'll write code to control the LED using the Viam software development kits.

LED light on

Prerequisites

What You'll Learn

What You'll Need

What You'll Build

Before you build your circuit, you need to set up the operating system on your Raspberry Pi and install viam-server on the Pi:

Follow the Raspberry Pi Setup Guide to set up the operating system.

Add a new machine in the Viam app. Then follow the setup instructions to install viam-server on the computer you're using for your project and connect to the Viam app. Wait until your machine has successfully connected.

Take a Quiz

Where is `viam-server` running in this project?

The Raspberry Pi Viam cloud servers in Viam's data center The computer where you are taking this quiz

Make sure you understand these concepts before moving ahead.

The next step is to build a simple LED circuit consisting of an LED and a resistor. The resistor protects the LED by limiting the flow of electricity through the circuit and is called a current-limiting resistor.

You will connect the LED and resistor to the pins on the Raspberry Pi to complete the circuit. Since the only power for this circuit comes from a Pi pin, controlling the state of the pin will toggle the LED on or off.

Understand GPIO pinout

A general-purpose input/output (GPIO) pin is a digital signal pin on a circuit board, like a Raspberry Pi, which may be used as an input or output, or both, and is controllable by software.

Raspberry Pi GPIO photo

Each pin has a specific role, and you can use it only for that role. Some of them are input/output, power (3.3V or 5V), or ground. As you can see in the diagram below, there are 40 pins on the Pi. 26 of them are GPIO pins.

Some of the GPIO pins can be used not only as GPIO pins but for other more specific functions as well. The website pinout.xyz is a helpful resource with the exact layout and role of each pin.

Raspberry Pi GPIO pinout diagram

One thing that might be confusing is the board pin numbering versus GPIO pin numbering. There are 40 physical pins numbered from 1 to 40. That is board pin numbering, corresponding to the pin's physical location on the board. Then there's numbering them by function or GPIO connection. For example, "GPIO 22". When working with the GPIO pins with Viam, you will use the board pin numbers. For example, board pin 38 is the same pin as GPIO pin 20, but when configuring your machine with Viam, you should refer to it as pin 38.

Wire your circuit

Now it's time to wire your circuit according to the diagram below. It doesn't matter what color jumper wires you use but for reference, the blue wire below connects the LED to ground, and the red wire connects the resistor to pin 8 (GPIO 14).

wiring diagram

The resistor and LED need to be in series as in the diagram above. To find the right resistor use the resistor color code – for a 100 ohm resistor, it needs to be brown-black-brown. You can use a multimeter to double-check the resistor value or check yours using the photo below.

100 ohm resistor

When hooking up the circuit, note the polarity of the LED. You will notice that the LED has long and short leads. The long lead is the positive side, which is known as the anode. The short lead is the negative side, which is known as the cathode. Connect the long anode to the resistor and the short cathode to ground (pin 6) on the Raspberry Pi.

Now that your circuit is wired, reconnect your Pi to power.

Configure your robot

Before proceeding, be sure that you have connected your Pi to the Viam app.

Now it's time to configure your machine's components. Go to the Viam app under the LOCATIONS tab, and navigate to your new machine's CONFIGURE tab.

Add a board component to represent your single-board computer, which in this case is the Raspberry Pi. To create the new component, click the + icon next to your machine part in the left-hand menu and select Component. Select the board type, then select the pi model. Enter a name or use the suggested name for your board and click Create. We used the name "local".

Your board component panel will look like this:

board component in Viam

Click the Save button in the top right corner of the page to save your changes.

Control your robot using the Viam app

When you configure your board component, the Viam app generates a control panel for it. Click the CONTROL tab to view the control panels for all your machine's components (in this case, just the board).

Click the board card to expand it. Here, you can click on Get to get the current status of your pin. The first time you click Get Pin State, it should return "Pin: 8 is low."

get pin state

You can now use the Set menu to set the status of your pin to high. Once you click Get Pin State again, it will look like this:

set pin state

When you set your pin to high the LED should illuminate. You can set the pin back and forth between high and low, and you will see your LED turn on or off depending on whether you have the value set to low or high.

set pin state

Congratulations! You have just successfully used Viam to make an LED blink with a Raspberry Pi! You have learned how the GPIO on a Raspberry Pi works, and how to build circuits for LED bulbs.

To make your LED blink periodically, you need to use an SDK.

Now you're ready to control the LED with code so you don't have to click buttons to turn it on and off.

In this section, you will learn the basics of programming hardware by using either the Viam Python SDK (software development kit) or the Viam Go SDK to make your LED blink.

Install a Viam SDK

Go ahead and install either the Viam Python SDK or the Viam Go SDK on your local computer. Use which ever programming language you are most comfortable with.

Refer to the appropriate SDK documentation for SDK installation instructions:

Connect your robot to the Viam SDK

The easiest way to get started writing an application with Viam is to navigate to the CONNECT tab of your machine's page on the Viam app and select the Code sample page. For this tutorial, we provide Python and Golang code snippets. Select Python or Golang and follow the instructions to connect to your machine.

These code snippets import all the necessary libraries and set up a connection with the Viam app in the cloud.

On your local computer, create a file called blink.py or blink.go. Paste the boilerplate code from the Code sample tab of the Viam app into the file in your code editor, and save the file.

You can now run the code. Doing so will ensure that the Viam SDK is properly installed and that the viam-server instance on your machine is live.

Run your code in Python

python3 <INSERT PATH TO YOUR FILE>/blink.py

If you successfully configured your machine and it is able to connect to the Viam app, you should see something like the following printed to the terminal after running your program. What you see here is a list of the various resources that have been configured on your machine in the Viam app (the board, as well as various built-in services).

python3 my-project-folder/blink.py
2023-07-14 13:05:50,958        INFO    viam.rpc.dial (dial.py:212)
    Connecting to socket: xxxxxxxxxxxx
Resources:
[namespace: "rdk"
type: "service"
subtype: "sensors"
name: "builtin"
, namespace: "rdk"
type: "service"
subtype: "data_manager"
name: "builtin"
, namespace: "rdk"
type: "service"
subtype: "motion"
name: "builtin"
, namespace: "rdk"
type: "component"
subtype: "board"
name: "local"
]

Run your code in Go

Navigate to the directory where your blink.go file is saved. Create the necessary go.mod and go.sum files (more information on the Go Blog) by running these commands:

go mod init blink
go mod tidy

Then, run your code:

go run blink.go

If you successfully configured your machine and it is able to connect to the Viam app, you should see something like the following printed to the terminal after running your program. What you see here is a list of the various resources that have been configured on your machine in the Viam app (the board, as well as various built-in services).

go run blink.go
2023-07-14T13:05:50.958-0700        INFO    client my-project-folder/blink.go:41
    Resources:
2023-07-14T13:05:50.958-0700        INFO    client my-project-folder/blink.go:41
    [rdk:component:board/local rdk:service:sensors/builtin
    rdk:service:motion/builtin rdk:service:data_manager/builtin]

In order to interact with the GPIO pins on our Raspberry Pi, you need to import the board component from the Viam SDK.

The Code sample page automatically adds the board import for you, but it doesn't hurt to double-check.

Write code in Python

At the top of your blink.py file, make sure the the following import is included:

from viam.components.board import Board

Next, you need to initialize the Raspberry Pi board and tell Viam which GPIO pin your LED is on. At the bottom of the main function, paste the following:

# Initialize the board and the LED on pin 8
local = Board.from_robot(robot, 'local')
led = await local.gpio_pin_by_name('8')

Now that the board and LED are initialized, let's create an infinite loop that will blink the LED on and off. Within the main function, add the following code to create an infinite loop. You can remove the line to close out the connection to your machine, since the infinite loop will never hit that line.

async def main():
    robot = await connect()

    print('Resources:')
    print(robot.resource_names)

    # Initialize the board and the LED on pin 8
    local = Board.from_robot(robot, 'local')
    led = await local.gpio_pin_by_name('8')

    # Create an infinite loop that will blink the LED on and off
    while (True):
        # True sets the LED pin to high/on.
        await led.set(True)
        print('LED is on')
        await asyncio.sleep(1)

        # False sets the pin to low/off.
        await led.set(False)
        print('LED is off')
        await asyncio.sleep(1)

Run your finished code:

python3 blink.py

Write code in Go

At the top of your blink.go, make sure to include the following import statements:

import (
  "fmt"
  "time"
  "go.viam.com/rdk/components/board"
)

The fmt package enables you to print and the time package enables you to add time delays.

Next, you need to initialize the Raspberry Pi board and tell Viam which GPIO pin your LED is on. At the bottom of the main function, paste the following:

// Initialize the board
myBoard, err := board.FromRobot(robot, "myBoard")
if err != nil {
    logger.Fatalf("could not get board: %v", err)
}

// Initialize the board's GPIO pin (pin 8) and name it "led"
led, err := myBoard.GPIOPinByName("8")
if err != nil {
    logger.Fatalf("could not get led: %v", err)
}

Now that the board and LED are initialized, let's create an infinite loop that will blink the LED on and off. Within the main function, add the following code to create an infinite loop. You can remove the line to close out the connection to your machine, since the infinite loop will never hit that line.

func main() {
  logger := logging.NewLogger("client")
  robot, err := client.New(
      context.Background(),
      "ADDRESS FROM THE VIAM APP",
      logger,
      client.WithDialOptions(rpc.WithEntityCredentials(
      // Replace "<API-KEY-ID>" (including brackets) with your machine's API key ID
      "<API-KEY-ID>",
      rpc.Credentials{
          Type:    rpc.CredentialsTypeAPIKey,
          // Replace "<API-KEY>" (including brackets) with your machine's API key
          Payload: "<API-KEY>",
      })),
  )
  if err != nil {
      logger.Fatal(err)
  }
  defer robot.Close(context.Background())
  logger.Info("Resources:")
  logger.Info(robot.ResourceNames())

  // Initialize the board
  myBoard, err := board.FromRobot(robot, "myBoard")
  if err != nil {
    logger.Fatalf("Could not get board: %v", err)
  }
  // Initialize the board's GPIO pin and name it "led"
  led, err := myBoard.GPIOPinByName("8")
  if err != nil {
    logger.Fatalf("Could not get led: %v", err)
  }

  //   Infinite loop that will blink the LED on and off.
  for {
    // True sets the LED pin to high/on.
    err = led.Set(context.Background(), true, nil)
    if err != nil {
      logger.Fatalf("Could not set led to on: %v", err)
    }
    fmt.Println("LED is on")
    time.Sleep(1 * time.Second)

    // False sets the pin to low/off.
    err = led.Set(context.Background(), false, nil)
    if err != nil {
      logger.Fatalf("Could not set led to off: %v", err)
    }
    fmt.Println("LED is off")
    time.Sleep(1 * time.Second)
  }
}

Run your finished code:

go run blink.go

Celebrate

If all goes well, you should see your LED blinking on and off again every second!

blink3

You can exit this program by pressing Ctrl + C in your terminal window.

If you get an error, you can check your code against the complete code here:

Completed code: https://github.com/viam-labs/LED-Blink

What You Learned

Related Resources

Now that you have completed this robotics project, check out some of our other tutorials.