Primary — Trading API de Matba ROFEX
API para operar en el Mercado Argentino de Futuros y Opciones (Matba ROFEX) a través de Primary Trading Platform (PTP). Soporta futuros (dólar, soja, trigo, maíz, índices), opciones sobre futuros, acciones, bonos y CEDEARs.
Base URL: https://api.remarkets.primary.com.ar
Docs: github.com/matbarofex — repositorio oficial con ejemplos open source.
Autenticación
Obtener Credenciales
| Entorno | URL | Descripción |
|---|
| REMARKET (demo) | remarkets.primary.ventures | Crear cuenta gratis para paper trading |
| LIVE (producción) | Contactar a mpi@primary.com.ar | Solicitar acceso al equipo MPI |
Una vez creada la cuenta en REMARKET, tendrás:
- Usuario: el que registraste
- Password: el que configuraste
- Account: tu número de cuenta (suele ser + últimos dígitos del usuario)
Obtener Token
python
import requests
r = requests.post("https://api.remarkets.primary.com.ar/auth/getToken",
headers={"X-Username": tu_usuario, "X-Password": tu_password})
token = r.headers["X-Auth-Token"]
El token se envía en adelante como header
en todos los requests.
⚠️ NUNCA hardcodear credenciales. Usar variables de entorno o parámetros CLI.
python
import os
TOKEN = os.getenv("PRIMARY_TOKEN") # Opcional: cachear token
USER = os.getenv("PRIMARY_USER")
PASS = os.getenv("PRIMARY_PASSWORD")
ACCOUNT = os.getenv("PRIMARY_ACCOUNT") # Ej: 12345 o REM12345
Renew Token
Si recibís un 401, el token expiró. Renovalo con un nuevo POST a
.
Segmentos
El mercado se organiza en segmentos (ruedas de negociación). Cada instrumento pertenece a un segmento.
| Segmento | Descripción |
|---|
| Derivados Financieros (futuros de dólar, índices) |
| Derivados Agropecuarios (soja, trigo, maíz) |
| Instrumentos listados en ambas divisiones |
| Mercados externos a Matba ROFEX (BYMA) |
| Mercado Abierto Electrónico |
| Ambiente de pruebas |
| , , , , , | Sub-segmentos de usuarios |
| , | Otros segmentos |
Listar Segmentos
http
GET https://api.remarkets.primary.com.ar/rest/segment/all
python
r = requests.get("https://api.remarkets.primary.com.ar/rest/segment/all",
headers={"X-Auth-Token": token})
print(r.json()["segments"])
Respuesta:
json
{"status":"OK","segments":[
{"marketSegmentId":"DDF","marketId":"ROFX"},
{"marketSegmentId":"DDA","marketId":"ROFX"},
...
]}
Instrumentos (Securities)
Los instrumentos se identifican por su símbolo y marketId. Ejemplos:
| Símbolo | Descripción | CFI Code |
|---|
| Futuro de dólar Junio 2026 | |
| Futuro de soja Rosario Mayo 2026 | |
| Opción Call sobre futuro dólar | |
| Opción Put sobre futuro dólar | |
| Acción Grupo Galicia | |
Todos los Instrumentos
http
GET https://api.remarkets.primary.com.ar/rest/instruments/all
Instrumentos con Detalle
http
GET https://api.remarkets.primary.com.ar/rest/instruments/details
Devuelve:
,
,
,
,
,
,
,
,
,
,
,
,
,
,
.
Detalle de un Instrumento
http
GET https://api.remarkets.primary.com.ar/rest/instruments/detail?symbol=DLR/JUN26&marketId=ROFX
Por Código CFI
http
GET https://api.remarkets.primary.com.ar/rest/instruments/byCFICode?CFICode=FXXXSX
| CFI Code | Tipo |
|---|
| Futuro |
| Futuro (genérico) |
| Opción Call sobre Futuro |
| Opción Put sobre Futuro |
| Opción Call europea sobre Futuro |
| Opción Put europea sobre Futuro |
| Acción |
| Bono |
| CEDEAR |
| Opción Call sobre Acción |
| Opción Put sobre Acción |
| Obligación Negociable |
Por Segmento
http
GET https://api.remarkets.primary.com.ar/rest/instruments/bySegment?MarketSegmentID=DDF&MarketID=ROFX
Market Data
En Tiempo Real (REST Snapshot)
http
GET https://api.remarkets.primary.com.ar/rest/marketdata/get
?marketId=ROFX
&symbol=DLR/JUN26
&entries=BI,OF,LA,OP,CL,SE,OI
&depth=3
| Entry | Significado |
|---|
| Bids (ofertas de compra en el book) |
| Offers (ofertas de venta en el book) |
| Last (último precio operado) |
| Opening Price (precio de apertura) |
| Closing Price (cierre rueda anterior) |
| Settlement Price (precio de ajuste, solo futuros) |
| High Price (máximo de la rueda) |
| Low Price (mínimo de la rueda) |
| Trade Volume (volumen operado en contratos) |
| Open Interest (interés abierto, solo futuros) |
| Index Value (solo índices) |
| Effective Volume (solo BYMA) |
| Nominal Volume (solo BYMA) |
| Auction Price (cierre del día corriente) |
python
r = requests.get("https://api.remarkets.primary.com.ar/rest/marketdata/get",
headers={"X-Auth-Token": token},
params={"marketId": "ROFX", "symbol": "DLR/JUN26",
"entries": "BI,OF,LA,OP,CL,SE,OI", "depth": 3})
data = r.json()["marketData"]
print(f"Bid: {data['BI']} | Offer: {data['OF']}")
print(f"Last: {data['LA']} | Settle: {data['SE']}")
Histórica (Trades)
http
GET https://api.remarkets.primary.com.ar/rest/data/getTrades
?marketId=ROFX
&symbol=DLR/JUN26
&dateFrom=2026-06-01
&dateTo=2026-06-08
Parámetros:
,
,
(una fecha),
/
(rango),
(para mercados externos),
(
).
python
r = requests.get("https://api.remarkets.primary.com.ar/rest/data/getTrades",
headers={"X-Auth-Token": token},
params={"marketId": "ROFX", "symbol": "DLR/JUN26",
"date": "2026-06-05"})
trades = r.json()["trades"]
for t in trades[:3]:
print(f"{t['datetime']} {t['price']} {t['size']}")
Órdenes
Tipos de Órdenes
| Tipo | Descripción |
|---|
| Orden con precio límite |
| Orden a mercado |
| Orden stop que se activa como limit |
| Market que se convierte en limit |
Nota: STOP_LIMIT y MARKET_TO_LIMIT no están disponibles para todos los instrumentos. Verificar en el detalle del instrumento.
Time in Force (TIF)
| TIF | Descripción |
|---|
| Solo válida por el día. Se expira al cierre de rueda |
| Immediate or Cancel |
| Fill or Kill |
| Good Till Date (requiere ) |
Ingresar Orden (REST)
http
GET https://api.remarkets.primary.com.ar/rest/order/newSingleOrder
?marketId=ROFX
&symbol=DLR/JUN26
&side=BUY
&orderQty=10
&ordType=LIMIT
&price=1450.0
&timeInForce=DAY
&account=TU_CUENTA
&cancelPrevious=False
&iceberg=False
Parámetros:
| Parámetro | Tipo | Obligatorio | Descripción |
|---|
| String | ✅ | |
| String | ✅ | Símbolo del instrumento |
| String | ✅ | o |
| Integer | ✅ | Cantidad de contratos |
| String | ✅ | o |
| Float | Condicional | Requerido para LIMIT |
| String | No | (default), , , |
| Integer/String | ✅ | Número de cuenta |
| Boolean | No | Cancela órdenes previas del mismo contrato/lado |
| Boolean | No | Orden Iceberg (default: false) |
| Integer | Condicional | Cantidad a divulgar (para iceberg) |
| Date | Condicional | Requerido para GTD (formato: YYYYMMDD) |
Respuesta:
json
{"status":"OK","order":{"clientId":"21581341758","proprietary":"PBCP"}}
El
es el
clOrdId (Client Order ID) que se usa para consultar/cancelar la orden.
Ingresar Orden (WebSocket)
json
{"type":"no","product":{"marketId":"ROFX","symbol":"DLR/JUN26"},
"price":185,"quantity":23,"side":"BUY","account":"20","iceberg":false}
Para identificar la orden vía WebSocket, incluir
:
json
{"type":"no","product":{"marketId":"ROFX","symbol":"DLR/JUN26"},
"price":185,"quantity":23,"side":"BUY","account":"20",
"iceberg":false,"wsClOrdId":"mioid-unico-123"}
Respuesta WebSocket (Execution Report):
json
{"type":"or","orderReport":{"orderId":"1128056","clOrdId":"user14545...",
"status":"PENDING_NEW","text":"Enviada","wsClOrdId":"mioid-unico-123"}}
Importante: El
solo aparece en el primer execution report. Luego se debe usar el
devuelto para seguimiento.
Reemplazar Orden
http
GET https://api.remarkets.primary.com.ar/rest/order/replaceById
?clOrdId=user144733478280357
&proprietary=api
&price=17
&orderQty=10
Cancelar Orden
http
GET https://api.remarkets.primary.com.ar/rest/order/cancelById
?clOrdId=ajduj3l13ieci2jr4ck
&proprietary=PBCP
Cancelar por WebSocket
json
{"type":"co","clientId":"user114121092035207","proprietary":"PBCP"}
Consultar Estado de la Orden (REST)
| Endpoint | Descripción |
|---|
GET /rest/order/id?clOrdId=...&proprietary=api
| Último estado del request |
GET /rest/order/allById?clOrdId=...&proprietary=api
| Todos los estados del request |
GET /rest/order/byOrderId?orderId=...
| Estado por Order ID |
GET /rest/order/actives?accountId=10
| Órdenes activas (NEW o PARTIALLY_FILLED) |
GET /rest/order/filleds?accountId=10
| Órdenes total o parcialmente operadas |
GET /rest/order/all?accountId=10
| Todos los estados de la cuenta |
GET /rest/order/byExecId?execId=T1234567
| Estado por Execution ID |
Execution Reports (WebSocket)
Suscribirse a una cuenta:
json
{"type":"os","account":{"id":"40"}}
Varias cuentas:
json
{"type":"os","accounts":[{"id":"40"},{"id":"4000"}]}
Todas las cuentas:
Solo órdenes activas:
json
{"type":"os","snapshotOnlyActive":true}
Risk API
La Risk API usa
HTTP Basic Auth con el mismo user/password, no token. Requiere el header
Authorization: Basic <base64>
adicionalmente al
.
python
import base64
auth = base64.b64encode(f"{user}:{password}".encode()).decode()
headers = {"X-Auth-Token": token, "Authorization": f"Basic {auth}"}
Posiciones de una Cuenta
http
GET https://api.remarkets.primary.com.ar/rest/risk/position/getPositions/{accountName}
python
r = requests.get(f"https://api.remarkets.primary.com.ar/rest/risk/position/getPositions/TU_CUENTA",
headers=headers)
positions = r.json()["positions"]
for p in positions:
print(f"{p['symbol']} Buy:{p['buySize']} Sell:{p['sellSize']} Diff:{p['totalDiff']}")
Posiciones Detalladas
http
GET https://api.remarkets.primary.com.ar/rest/risk/detailedPosition/{accountName}
Devuelve desglose por instrumento con:
,
,
,
,
,
,
,
.
Reporte de Cuenta
http
GET https://api.remarkets.primary.com.ar/rest/risk/accountReport/{accountName}
python
import base64
auth = base64.b64encode(f"{user}:{password}".encode()).decode()
headers = {"X-Auth-Token": token, "Authorization": f"Basic {auth}"}
r = requests.get(f"https://api.remarkets.primary.com.ar/rest/risk/accountReport/TU_CUENTA",
headers=headers)
data = r.json()["accountData"]
print(f"Colateral: {data['collateral']}")
print(f"Margen: {data['margin']}")
print(f"Disponible: {data['availableToCollateral']}")
# Saldos por moneda
for moneda, saldo in data['detailedAccountReports']['0']['currencyBalance']['detailedCurrencyBalance'].items():
print(f" {moneda}: consumido={saldo['consumed']} disponible={saldo['available']}")
WebSocket
URL: wss://api.remarkets.primary.com.ar/
La API WebSocket recibe mensajes asíncronos. El token se envía como header en la conexión:
python
import websocket
ws = websocket.WebSocketApp(
"wss://api.remarkets.primary.com.ar/",
header=[f"X-Auth-Token: {token}"],
on_open=on_open,
on_message=on_message,
...
)
ws.run_forever()
Los mensajes tienen el formato:
| type | Significado |
|---|
| New Order (enviar orden) |
| Cancel Order (cancelar orden) |
| Order Report (execution report recibido) |
| Order Subscription (suscribirse a reports) |
| Subscribe Market Data |
| Market Data (recibido) |
Market Data por WebSocket
json
{"type":"smd","level":1,"entries":["OF","BI","LA"],
"products":[{"symbol":"DLR/JUN26","marketId":"ROFX"}],"depth":2}
Respuesta:
json
{"type":"Md","instrumentId":{"marketId":"ROFX","symbol":"DLR/JUN26"},
"marketData":{"OF":[{"price":189,"size":21},{"price":188,"size":13}]}}
Estados de una Orden
| Estado | Significado |
|---|
| Enviada al mercado, aún no procesada |
| Aceptada, activa en el book |
| Parcialmente operada |
| Totalmente operada |
| Cancelada |
| Rechazada (ver para motivo) |
| Cancelación en proceso |
| Reemplazo en proceso |
| Reemplazada |
| Pendiente de aprobación |
Errores Comunes
| Error | Causa | Solución |
|---|
| 401 Unauthorized | Token inválido o expirado | Renovar con |
"No tiene acceso a la cuenta"
| Account ID incorrecto | Verificar |
| Symbol incorrecto | Verificar símbolo con |
| Sin permisos para el endpoint | Verificar segmento/método |
| Endpoint no existe | Revisar URL |
| Bid/Offer vacíos | Mercado cerrado o sin liquidez | Consultar en horario de rueda |
Scripts de Ejemplo
Ver ./scripts/:
bash
# Autenticación y token (opcional, los scripts hacen login automático)
export PRIMARY_USER="tu_usuario"
export PRIMARY_PASSWORD="tu_password"
export PRIMARY_ACCOUNT="TU_CUENTA"
# Listar segmentos e instrumentos
python scripts/instruments.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD
# Market data de un futuro
python scripts/market_data.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--symbol DLR/JUN26 --entries BI,OF,LA
# Ver reporte de cuenta
python scripts/check_account.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--account TU_CUENTA
# Ver posiciones
python scripts/check_positions.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--account TU_CUENTA
# Enviar orden (¡cuidado! orden real en live)
python scripts/place_order.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--symbol DLR/JUN26 --side BUY --qty 1 --type LIMIT --price 1450 --account TU_CUENTA
# WebSocket: Market Data en tiempo real
python scripts/websocket_md.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--symbols DLR/JUN26 --entries BI,OF,LA --depth 3
# WebSocket: Execution Reports
python scripts/websocket_orders.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--account TU_CUENTA
# WebSocket: Enviar orden (requiere suscripción a execution reports aparte)
python scripts/websocket_send_order.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--symbol DLR/JUN26 --side BUY --qty 1 --type LIMIT --price 1450 --account TU_CUENTA
# WebSocket: Cancelar orden
python scripts/websocket_send_order.py --user $PRIMARY_USER --password $PRIMARY_PASSWORD \
--cancel --clordid user12345... --proprietary PBCP
Glosario de Campos
| Campo | Descripción |
|---|
| Client Order ID — ID del request al mercado |
| Order ID — ID de la orden en el mercado |
| Execution ID — ID de una ejecución particular |
| Usuario FIX que envió la orden ( o ) |
| ID de orden enviada por WebSocket (solo en 1er report) |
| Precio promedio operado |
| Cantidad acumulada operada |
| Cantidad remanente |
| Último precio operado |
| Última cantidad operada |
| Fecha y hora de la transacción |
| Incremento mínimo de cantidad |
| Incremento mínimo de precio (tick price) |
| Multiplicador del contrato |
| Fecha de vencimiento |
| Factor para precio unitario |
| Límite mínimo de precio |
| Límite máximo de precio |