Skip to contents

About

SIOPE (Sistema de Informações sobre Orçamentos Públicos em Educação) provides education spending data from states and municipalities. The system is maintained by FNDE (Fundo Nacional de Desenvolvimento da Educação) and the API uses an OData-style interface hosted at https://www.fnde.gov.br/olinda-ide/servico/DADOS_ABERTOS_SIOPE/.

Data covers revenues, expenses, indicators, official contacts, and staff compensation related to public education spending.

Available functions

Portuguese English Description
get_siope_dados_gerais() get_siope_general_data() General data (demographics, GDP, totals)
get_siope_responsaveis() get_siope_officials() Officials responsible for declarations
get_siope_despesas() get_siope_expenses() Education expenses
get_siope_despesas_funcao() get_siope_expenses_by_function() Expenses by government function
get_siope_indicadores() get_siope_indicators() Education spending indicators
get_siope_info_complementares() get_siope_supplementary() Supplementary information
get_siope_receitas() get_siope_revenues() Education revenues
get_siope_remuneracao() get_siope_compensation() Staff compensation

Required parameters

All functions share three required parameters:

Portuguese English Description Example
ano year Year of the data 2023
periodo period Bimester (1-6) 6 (Nov-Dec)
uf state State abbreviation "PE"

Exception: get_siope_remuneracao() / get_siope_compensation() has a 4th required parameter: mes / month (1-12).

Period mapping

Period Months
1 January – February
2 March – April
3 May – June
4 July – August
5 September – October
6 November – December

Performance tip: use filter to reduce downloads

The SIOPE API returns all municipalities for a given state. A single call for São Paulo (state = "SP") returns 645+ rows. Use the filter parameter to apply server-side filtering and download only the data you need. This is much faster than downloading everything and filtering locally with dplyr::filter().

Filtering by municipality

# SLOW: downloads all 185 municipalities, then filters locally
dados_pe <- get_siope_general_data(year = 2023, period = 6, state = "PE")
recife <- dplyr::filter(dados_pe, nom_muni == "Recife")

# FAST: server returns only Recife's data
recife <- get_siope_general_data(
  year = 2023, period = 6, state = "PE",
  filter = "NOM_MUNI eq 'Recife'"
)

# Filter by IBGE code
recife <- get_siope_general_data(
  year = 2023, period = 6, state = "PE",
  filter = "COD_MUNI eq 261160"
)

OData filter syntax

The filter parameter uses OData v4 syntax. Column names must use the original API names (uppercase), not the cleaned snake_case names. Use verbose = TRUE on a max_rows = 1 call to inspect column names.

Operator Meaning Example
eq Equals "NOM_MUNI eq 'Recife'"
ne Not equals "TIPO ne 'Estado'"
gt, ge Greater (or equal) "NUM_POPU gt 100000"
lt, le Less (or equal) "NUM_POPU le 50000"
and Logical AND "NOM_MUNI eq 'Recife' and TIPO eq 'Município'"
or Logical OR "NOM_MUNI eq 'Recife' or NOM_MUNI eq 'Olinda'"
contains() Substring match "contains(NOM_MUNI, 'Porto')"

Selecting specific columns

Use select to request only the columns you need (reduces payload):

# Only municipality name and declared value
resumo <- get_siope_expenses(
  year = 2023, period = 6, state = "PE",
  select = c("NOM_MUNI", "NOM_ITEM", "VAL_DECL"),
  filter = "NOM_MUNI eq 'Recife'"
)

Column names in filter/select/orderby must use the original API names (UPPERCASE), not the cleaned snake_case names. To discover valid column names, run a quick test:

sample <- get_siope_expenses(
  year = 2023, period = 6, state = "PE", max_rows = 1
)
# snake_case names → convert back to UPPER for filter/select
toupper(names(sample))

Sorting results

# Sort by population descending
dados <- get_siope_general_data(
  year = 2023, period = 6, state = "PE",
  orderby = "NUM_POPU desc", max_rows = 10
)

Compensation data with filters

# Staff compensation for Agrestina, only "Efetivo" professionals
rem <- get_siope_compensation(
  year = 2024, period = 1, month = 1, state = "PE",
  filter = "NOM_MUNI eq 'Agrestina' and DS_SITUACAO_PROFISSIONAL eq 'Efetivo'"
)

Examples

library(tesouror)
library(dplyr)

# General data for all municipalities in Pernambuco, last bimester 2023
dados_pe <- get_siope_general_data(year = 2023, period = 6, state = "PE")

# Education revenues for São Paulo
receitas_sp <- get_siope_revenues(year = 2023, period = 6, state = "SP")

# Education indicators for Minas Gerais
indicadores_mg <- get_siope_indicators(year = 2023, period = 6, state = "MG")

# Expenses by function for Rio de Janeiro
desp_func_rj <- get_siope_expenses_by_function(
  year = 2023, period = 6, state = "RJ"
)

# Staff compensation for December 2023 in Bahia
remuneracao_ba <- get_siope_compensation(
  year = 2023, period = 6, month = 12, state = "BA"
)

Quick test with max_rows

# Grab just 10 rows to inspect the structure
sample <- get_siope_general_data(
  year = 2023, period = 6, state = "PE", max_rows = 10
)
glimpse(sample)

Combining states

# Fetch multiple states and combine
nordeste <- c("AL", "BA", "CE", "MA", "PB", "PE", "PI", "RN", "SE")

indicadores_ne <- purrr::map_dfr(nordeste, function(uf) {
  get_siope_indicators(year = 2023, period = 6, state = uf)
})

How the OData URL works

The SIOPE API uses an OData-style URL pattern different from the other APIs in this package. Internally, the package builds URLs like:

https://www.fnde.gov.br/olinda-ide/servico/DADOS_ABERTOS_SIOPE/versao/v1/odata/
  Dados_Gerais_Siope(Ano_Consulta=@Ano_Consulta,Num_Peri=@Num_Peri,Sig_UF=@Sig_UF)
  ?@Ano_Consulta=2023&@Num_Peri=6&@Sig_UF='PE'&$format=json

You don’t need to worry about this — just pass the parameters and the package handles the rest. Use verbose = TRUE to see the full URL:

get_siope_general_data(year = 2023, period = 6, state = "PE", verbose = TRUE)
#> ℹ API call: https://www.fnde.gov.br/olinda-ide/servico/...