Compare commits
13 commits
9a1d896979
...
233fda5661
Author | SHA1 | Date | |
---|---|---|---|
233fda5661 | |||
c37ebb9854 | |||
5cab3913d8 | |||
2a1ef713cc | |||
e5b51d37e1 | |||
6dc5746da1 | |||
c98a402b38 | |||
33d24561a9 | |||
2b5035e16e | |||
83c9049cf5 | |||
b8ae14fbe3 | |||
7ce3171a0e | |||
be20614dfe |
12 changed files with 633 additions and 4 deletions
|
@ -127,6 +127,13 @@ install() {
|
||||||
then
|
then
|
||||||
python3_install
|
python3_install
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Verificar si pip está instalado, instalarlo si es necesario
|
||||||
|
command_installed pip3
|
||||||
|
if [ $? -ne 0 ]
|
||||||
|
then
|
||||||
|
pip_install
|
||||||
|
fi
|
||||||
|
|
||||||
# Install mozilla sops from OS Packages
|
# Install mozilla sops from OS Packages
|
||||||
command_installed $SOPS_PACKAGE
|
command_installed $SOPS_PACKAGE
|
||||||
|
|
1
bin/config/bash_la.rate
Normal file
1
bin/config/bash_la.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
20.00
|
1
bin/config/bash_ww.rate
Normal file
1
bin/config/bash_ww.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
30.00
|
1
bin/config/fullstack_la.rate
Normal file
1
bin/config/fullstack_la.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
100.00
|
1
bin/config/fullstack_ww.rate
Normal file
1
bin/config/fullstack_ww.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
200.00
|
1
bin/config/kdevs.rate
Normal file
1
bin/config/kdevs.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
3
|
1
bin/config/python_la.rate
Normal file
1
bin/config/python_la.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
30.00
|
1
bin/config/python_ww.rate
Normal file
1
bin/config/python_ww.rate
Normal file
|
@ -0,0 +1 @@
|
||||||
|
100.00
|
|
@ -45,6 +45,38 @@ DOCKER_ENTRY=entrypoint.sh
|
||||||
|
|
||||||
VERSION="$(cat < ${BIN_HOME}/${BIN_CONF}/version)"
|
VERSION="$(cat < ${BIN_HOME}/${BIN_CONF}/version)"
|
||||||
|
|
||||||
|
# Verificar si el script se está ejecutando como usuario root (superusuario)
|
||||||
|
function is_root() {
|
||||||
|
if [ "$(id -u)" -eq 0 ]; then
|
||||||
|
return 0 # Es root
|
||||||
|
else
|
||||||
|
return 1 # No es root
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Escalar privilegios a superusuario si es necesario
|
||||||
|
# Esta función no ejecuta el script con sudo, solo verifica si tenemos privilegios
|
||||||
|
function check_root_privileges() {
|
||||||
|
if ! is_root; then
|
||||||
|
echo -e "\n${head_info}: Se requieren privilegios de administrador para esta operación."
|
||||||
|
|
||||||
|
# Verificar si sudo está instalado
|
||||||
|
command -v sudo >/dev/null 2>&1
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
echo -e "${head_error}: El comando 'sudo' no está instalado. No se pueden escalar privilegios."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Informar que se necesita ejecutar con sudo
|
||||||
|
echo -e "${head_error}: Este script debe ejecutarse con sudo."
|
||||||
|
echo -e "${head_info}: Por favor, ejecute: sudo $0"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Ya es root, no necesita hacer nada
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
# Test library
|
# Test library
|
||||||
function baselib_test() {
|
function baselib_test() {
|
||||||
echo "Base Library loaded!"
|
echo "Base Library loaded!"
|
||||||
|
|
|
@ -100,20 +100,20 @@ function python3_install() {
|
||||||
if [ "$(uname)" == "Darwin" ]; then
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
# En macOS, instalamos o actualizamos Python a través de Homebrew
|
# En macOS, instalamos o actualizamos Python a través de Homebrew
|
||||||
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
||||||
brew install python python-pip
|
brew install python
|
||||||
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
# En sistemas Debian y derivados, instalamos o actualizamos Python a través de apt
|
# En sistemas Debian y derivados, instalamos o actualizamos Python a través de apt
|
||||||
apt update
|
apt update
|
||||||
apt install -y python3 python3-pip
|
apt install -y python3
|
||||||
elif [ -f /etc/redhat-release ]; then
|
elif [ -f /etc/redhat-release ]; then
|
||||||
# En sistemas Red Hat, instalamos o actualizamos Python a través de yum
|
# En sistemas Red Hat, instalamos o actualizamos Python a través de yum
|
||||||
dnf install -y python3 python3-pip
|
dnf install -y python3
|
||||||
elif [ -f /etc/arch-release ]; then
|
elif [ -f /etc/arch-release ]; then
|
||||||
# En Arch Linux, instalamos o actualizamos Python a través de pacman
|
# En Arch Linux, instalamos o actualizamos Python a través de pacman
|
||||||
pacman -Sy --noconfirm python
|
pacman -Sy --noconfirm python
|
||||||
elif [ -f /etc/rc.conf ]; then
|
elif [ -f /etc/rc.conf ]; then
|
||||||
# En BSD, instalamos o actualizamos Python a través de pkg
|
# En BSD, instalamos o actualizamos Python a través de pkg
|
||||||
pkg install -y python3 python3-pip
|
pkg install -y python3
|
||||||
else
|
else
|
||||||
echo "${os_nofound}"
|
echo "${os_nofound}"
|
||||||
exit 1
|
exit 1
|
||||||
|
@ -122,6 +122,120 @@ function python3_install() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Update python3 package
|
||||||
|
function python3_update() {
|
||||||
|
|
||||||
|
echo "Actualizando Python..."
|
||||||
|
|
||||||
|
# Verificar si Python está instalado
|
||||||
|
command_installed python3
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
# Si Python no está instalado, llamamos a la función de instalación
|
||||||
|
python3_install
|
||||||
|
return $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Actualizar Python según el sistema operativo
|
||||||
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
|
# En macOS, actualizamos Python a través de Homebrew
|
||||||
|
brew upgrade python
|
||||||
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
|
# En sistemas Debian y derivados
|
||||||
|
apt update
|
||||||
|
apt install --only-upgrade -y python3
|
||||||
|
elif [ -f /etc/redhat-release ]; then
|
||||||
|
# En sistemas Red Hat
|
||||||
|
dnf upgrade -y python3
|
||||||
|
elif [ -f /etc/arch-release ]; then
|
||||||
|
# En Arch Linux
|
||||||
|
pacman -Syu --noconfirm python
|
||||||
|
elif [ -f /etc/rc.conf ]; then
|
||||||
|
# En BSD
|
||||||
|
pkg upgrade -y python3
|
||||||
|
else
|
||||||
|
echo "${os_nofound}"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Python actualizado correctamente."
|
||||||
|
return 0
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
# Install pip package
|
||||||
|
function pip_install() {
|
||||||
|
|
||||||
|
echo "Instalando pip..."
|
||||||
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
|
# En macOS, instalamos o actualizamos pip a través de Homebrew
|
||||||
|
brew install python-pip
|
||||||
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
|
# En sistemas Debian y derivados, instalamos o actualizamos pip a través de apt
|
||||||
|
apt update
|
||||||
|
apt install -y python3-pip
|
||||||
|
elif [ -f /etc/redhat-release ]; then
|
||||||
|
# En sistemas Red Hat, instalamos o actualizamos pip a través de yum
|
||||||
|
dnf install -y python3-pip
|
||||||
|
elif [ -f /etc/arch-release ]; then
|
||||||
|
# En Arch Linux, instalamos o actualizamos pip a través de pacman
|
||||||
|
pacman -Sy --noconfirm python-pip
|
||||||
|
elif [ -f /etc/rc.conf ]; then
|
||||||
|
# En BSD, instalamos o actualizamos pip a través de pkg
|
||||||
|
pkg install -y python3-pip
|
||||||
|
else
|
||||||
|
echo "${os_nofound}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Pip instalado correctamente."
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
# Update pip package
|
||||||
|
function pip_update() {
|
||||||
|
|
||||||
|
echo "Actualizando pip..."
|
||||||
|
|
||||||
|
# Verificar si pip está instalado
|
||||||
|
command_installed pip3
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
# Si pip no está instalado, llamamos a la función de instalación
|
||||||
|
pip_install
|
||||||
|
return $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Actualizar pip usando el método adecuado según el sistema operativo
|
||||||
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
|
# En macOS, actualizamos pip a través de Homebrew
|
||||||
|
brew upgrade python-pip
|
||||||
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
|
# En sistemas Debian y derivados
|
||||||
|
apt update
|
||||||
|
apt install --only-upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
elif [ -f /etc/redhat-release ]; then
|
||||||
|
# En sistemas Red Hat
|
||||||
|
dnf upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
elif [ -f /etc/arch-release ]; then
|
||||||
|
# En Arch Linux
|
||||||
|
pacman -Syu --noconfirm python-pip
|
||||||
|
elif [ -f /etc/rc.conf ]; then
|
||||||
|
# En BSD
|
||||||
|
pkg upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
else
|
||||||
|
echo "${os_nofound}"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Pip actualizado correctamente."
|
||||||
|
return 0
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
# Install mozilla sops package on os system supported
|
# Install mozilla sops package on os system supported
|
||||||
function sops_install() {
|
function sops_install() {
|
||||||
|
|
||||||
|
|
343
bin/rate_update.py
Executable file
343
bin/rate_update.py
Executable file
|
@ -0,0 +1,343 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
"""
|
||||||
|
Rate Update Script
|
||||||
|
|
||||||
|
Este script actualiza las tarifas por hora de diferentes tipos de programadores
|
||||||
|
por región utilizando la API de Perplexity para obtener datos actualizados.
|
||||||
|
|
||||||
|
Licencia: AGPL-3.0
|
||||||
|
Modified: 2025-03-12
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import glob
|
||||||
|
import re
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import requests
|
||||||
|
import sys
|
||||||
|
import subprocess
|
||||||
|
import importlib.util
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
# Configuración de logging
|
||||||
|
logging.basicConfig(
|
||||||
|
level=logging.INFO,
|
||||||
|
format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
|
||||||
|
handlers=[logging.StreamHandler()]
|
||||||
|
)
|
||||||
|
logger = logging.getLogger('rate_update')
|
||||||
|
|
||||||
|
# Verificar si pycountry está instalado, si no, instalarlo
|
||||||
|
def check_install_pycountry():
|
||||||
|
"""Verifica si pycountry está instalado y lo instala si es necesario."""
|
||||||
|
if importlib.util.find_spec("pycountry") is None:
|
||||||
|
logger.info("La biblioteca pycountry no está instalada.")
|
||||||
|
|
||||||
|
# Verificar si pip está disponible
|
||||||
|
try:
|
||||||
|
# Primero comprobamos si pip está instalado
|
||||||
|
pip_check = subprocess.run([sys.executable, "-m", "pip", "--version"],
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.PIPE)
|
||||||
|
|
||||||
|
if pip_check.returncode == 0:
|
||||||
|
logger.warning("La biblioteca pycountry no está instalada.")
|
||||||
|
logger.warning("Se continuará sin validación de códigos de país.")
|
||||||
|
logger.warning("Para habilitar la validación, instale pycountry manualmente:")
|
||||||
|
logger.warning(" sudo pip3 install pycountry")
|
||||||
|
logger.warning("O:")
|
||||||
|
logger.warning(" sudo apt-get install python3-pycountry")
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
logger.warning("pip no está instalado. No se puede instalar pycountry automáticamente.")
|
||||||
|
logger.warning("Se continuará sin validación de códigos de país.")
|
||||||
|
logger.warning("Para habilitar la validación, instale pycountry manualmente:")
|
||||||
|
logger.warning(" sudo apt-get install python3-pip")
|
||||||
|
logger.warning(" sudo pip3 install pycountry")
|
||||||
|
logger.warning("O:")
|
||||||
|
logger.warning(" sudo apt-get install python3-pycountry")
|
||||||
|
return False
|
||||||
|
|
||||||
|
except FileNotFoundError:
|
||||||
|
logger.warning("No se pudo ejecutar pip. El sistema no puede encontrar el ejecutable de Python.")
|
||||||
|
logger.warning("Se continuará sin validación de códigos de país.")
|
||||||
|
logger.warning("Para habilitar la validación, verifique su instalación de Python y pycountry.")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# Intentamos importar pycountry
|
||||||
|
try:
|
||||||
|
global pycountry
|
||||||
|
import pycountry
|
||||||
|
return True
|
||||||
|
except ImportError:
|
||||||
|
logger.warning("No se puede importar pycountry. Continuando sin validación de códigos de país.")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# Directorio base del proyecto
|
||||||
|
BASE_DIR = Path(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
|
||||||
|
CONFIG_DIR = BASE_DIR / 'bin' / 'config'
|
||||||
|
|
||||||
|
# Archivo de configuración para el modelo de IA
|
||||||
|
MODEL_CONFIG_FILE = CONFIG_DIR / 'rate_model.ai'
|
||||||
|
|
||||||
|
def get_ai_model():
|
||||||
|
"""Obtener el modelo de IA configurado."""
|
||||||
|
try:
|
||||||
|
with open(MODEL_CONFIG_FILE, 'r') as f:
|
||||||
|
return f.read().strip()
|
||||||
|
except FileNotFoundError:
|
||||||
|
logger.warning(f"Archivo de configuración {MODEL_CONFIG_FILE} no encontrado. Usando modelo predeterminado 'o1'.")
|
||||||
|
return "o1"
|
||||||
|
|
||||||
|
def get_perplexity_api_key():
|
||||||
|
"""Obtener la clave API de Perplexity desde una variable de entorno."""
|
||||||
|
api_key = os.environ.get('PERPLEXITY_API_KEY')
|
||||||
|
if not api_key:
|
||||||
|
logger.error("No se encontró la clave API de Perplexity. Establezca la variable de entorno PERPLEXITY_API_KEY.")
|
||||||
|
sys.exit(1)
|
||||||
|
return api_key
|
||||||
|
|
||||||
|
def query_perplexity(prompt, model="o1"):
|
||||||
|
"""Realizar una consulta a la API de Perplexity."""
|
||||||
|
api_key = get_perplexity_api_key()
|
||||||
|
|
||||||
|
url = "https://api.perplexity.ai/chat/completions"
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"Bearer {api_key}",
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
}
|
||||||
|
|
||||||
|
data = {
|
||||||
|
"model": model,
|
||||||
|
"messages": [
|
||||||
|
{
|
||||||
|
"role": "system",
|
||||||
|
"content": "You are a helpful assistant that provides accurate salary information for developers worldwide. Respond only with numeric values in USD with 2 decimal places."
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "user",
|
||||||
|
"content": prompt
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"max_tokens": 100
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = requests.post(url, headers=headers, json=data)
|
||||||
|
response.raise_for_status()
|
||||||
|
result = response.json()
|
||||||
|
|
||||||
|
# Extraer solo el valor numérico de la respuesta
|
||||||
|
content = result.get('choices', [{}])[0].get('message', {}).get('content', '')
|
||||||
|
# Intentar encontrar un número con 2 decimales en la respuesta
|
||||||
|
match = re.search(r'\$?(\d+\.\d{2})', content)
|
||||||
|
if match:
|
||||||
|
return float(match.group(1))
|
||||||
|
|
||||||
|
# Si no encuentra un formato exacto, intentar convertir toda la respuesta a float
|
||||||
|
try:
|
||||||
|
# Eliminar cualquier símbolo de moneda y espacios
|
||||||
|
cleaned_content = re.sub(r'[^\d.]', '', content)
|
||||||
|
return round(float(cleaned_content), 2)
|
||||||
|
except ValueError:
|
||||||
|
logger.error(f"No se pudo extraer un valor numérico de la respuesta: {content}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
logger.error(f"Error al conectar con la API de Perplexity: {e}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
# Variable global para indicar si pycountry está disponible
|
||||||
|
pycountry_available = False
|
||||||
|
|
||||||
|
def is_valid_country_code(code):
|
||||||
|
"""
|
||||||
|
Verifica si un código de país de 2 letras es válido.
|
||||||
|
Usa pycountry si está disponible, o una lista básica de códigos comunes.
|
||||||
|
También acepta códigos especiales 'ww' y 'la'.
|
||||||
|
"""
|
||||||
|
# Códigos especiales para propósitos internos
|
||||||
|
if code in ['ww', 'la']: # worldwide y Latin America
|
||||||
|
return True
|
||||||
|
|
||||||
|
# Lista básica de códigos de país comunes (ISO 3166-1 alpha-2)
|
||||||
|
common_country_codes = [
|
||||||
|
'af', 'al', 'dz', 'as', 'ad', 'ao', 'ai', 'aq', 'ag', 'ar',
|
||||||
|
'am', 'aw', 'au', 'at', 'az', 'bs', 'bh', 'bd', 'bb', 'by',
|
||||||
|
'be', 'bz', 'bj', 'bm', 'bt', 'bo', 'ba', 'bw', 'br', 'bn',
|
||||||
|
'bg', 'bf', 'bi', 'cv', 'kh', 'cm', 'ca', 'ky', 'cf', 'td',
|
||||||
|
'cl', 'cn', 'co', 'km', 'cg', 'cd', 'ck', 'cr', 'ci', 'hr',
|
||||||
|
'cu', 'cy', 'cz', 'dk', 'dj', 'dm', 'do', 'ec', 'eg', 'sv',
|
||||||
|
'gq', 'er', 'ee', 'et', 'fk', 'fo', 'fj', 'fi', 'fr', 'gf',
|
||||||
|
'pf', 'ga', 'gm', 'ge', 'de', 'gh', 'gi', 'gr', 'gl', 'gd',
|
||||||
|
'gp', 'gu', 'gt', 'gn', 'gw', 'gy', 'ht', 'va', 'hn', 'hk',
|
||||||
|
'hu', 'is', 'in', 'id', 'ir', 'iq', 'ie', 'il', 'it', 'jm',
|
||||||
|
'jp', 'jo', 'kz', 'ke', 'ki', 'kp', 'kr', 'kw', 'kg', 'la',
|
||||||
|
'lv', 'lb', 'ls', 'lr', 'ly', 'li', 'lt', 'lu', 'mo', 'mk',
|
||||||
|
'mg', 'mw', 'my', 'mv', 'ml', 'mt', 'mh', 'mq', 'mr', 'mu',
|
||||||
|
'yt', 'mx', 'fm', 'md', 'mc', 'mn', 'me', 'ms', 'ma', 'mz',
|
||||||
|
'mm', 'na', 'nr', 'np', 'nl', 'nc', 'nz', 'ni', 'ne', 'ng',
|
||||||
|
'nu', 'nf', 'mp', 'no', 'om', 'pk', 'pw', 'ps', 'pa', 'pg',
|
||||||
|
'py', 'pe', 'ph', 'pn', 'pl', 'pt', 'pr', 'qa', 're', 'ro',
|
||||||
|
'ru', 'rw', 'bl', 'sh', 'kn', 'lc', 'mf', 'pm', 'vc', 'ws',
|
||||||
|
'sm', 'st', 'sa', 'sn', 'rs', 'sc', 'sl', 'sg', 'sx', 'sk',
|
||||||
|
'si', 'sb', 'so', 'za', 'gs', 'ss', 'es', 'lk', 'sd', 'sr',
|
||||||
|
'sj', 'sz', 'se', 'ch', 'sy', 'tw', 'tj', 'tz', 'th', 'tl',
|
||||||
|
'tg', 'tk', 'to', 'tt', 'tn', 'tr', 'tm', 'tc', 'tv', 'ug',
|
||||||
|
'ua', 'ae', 'gb', 'us', 'uy', 'uz', 'vu', 've', 'vn', 'vg',
|
||||||
|
'vi', 'wf', 'eh', 'ye', 'zm', 'zw'
|
||||||
|
]
|
||||||
|
|
||||||
|
# Si pycountry está disponible, lo usamos para validación
|
||||||
|
global pycountry_available
|
||||||
|
if pycountry_available:
|
||||||
|
try:
|
||||||
|
# Verifica si el código existe en pycountry
|
||||||
|
return pycountry.countries.get(alpha_2=code.upper()) is not None
|
||||||
|
except (AttributeError, Exception) as e:
|
||||||
|
logger.warning(f"Error al validar el código de país '{code}' con pycountry: {e}")
|
||||||
|
# Fallback a la lista básica
|
||||||
|
|
||||||
|
# Si pycountry no está disponible o falló, usamos la lista básica
|
||||||
|
return code.lower() in common_country_codes
|
||||||
|
|
||||||
|
def parse_rate_filename(filename):
|
||||||
|
"""
|
||||||
|
Analiza el nombre del archivo de tarifa para extraer el tipo de programador y la región.
|
||||||
|
Formato: [tipo_de_programador]_[código_país].rate
|
||||||
|
Valida el código de país usando pycountry.
|
||||||
|
"""
|
||||||
|
base_name = os.path.basename(filename)
|
||||||
|
match = re.match(r'([a-z]+)_([a-z]{2})\.rate$', base_name)
|
||||||
|
|
||||||
|
if not match:
|
||||||
|
# Para archivos como kdevs.rate que no siguen el patrón estándar
|
||||||
|
if base_name.endswith('.rate'):
|
||||||
|
programmer_type = base_name.replace('.rate', '')
|
||||||
|
return programmer_type, None
|
||||||
|
return None, None
|
||||||
|
|
||||||
|
programmer_type, region_code = match.groups()
|
||||||
|
|
||||||
|
# Validar el código de país
|
||||||
|
if not is_valid_country_code(region_code):
|
||||||
|
logger.warning(f"Código de país no válido '{region_code}' en el archivo {base_name}")
|
||||||
|
# Aun así continuamos con el proceso
|
||||||
|
|
||||||
|
return programmer_type, region_code
|
||||||
|
|
||||||
|
def generate_prompt(programmer_type, region_code):
|
||||||
|
"""Genera un prompt para consultar a Perplexity sobre la tarifa horaria."""
|
||||||
|
region_map = {
|
||||||
|
'ww': 'mundial',
|
||||||
|
'la': 'Latinoamérica',
|
||||||
|
'pa': 'Panamá',
|
||||||
|
'co': 'Colombia',
|
||||||
|
'mx': 'México',
|
||||||
|
'ar': 'Argentina',
|
||||||
|
'br': 'Brasil',
|
||||||
|
'cl': 'Chile',
|
||||||
|
'pe': 'Perú',
|
||||||
|
'ec': 'Ecuador',
|
||||||
|
'us': 'Estados Unidos',
|
||||||
|
'ca': 'Canadá',
|
||||||
|
'uk': 'Reino Unido',
|
||||||
|
'de': 'Alemania',
|
||||||
|
'fr': 'Francia',
|
||||||
|
'es': 'España',
|
||||||
|
'it': 'Italia'
|
||||||
|
}
|
||||||
|
|
||||||
|
# Archivos especiales como kdevs.rate se manejan de forma diferente
|
||||||
|
if programmer_type == 'kdevs':
|
||||||
|
# Simplemente mantener el valor actual para kdevs.rate
|
||||||
|
return None
|
||||||
|
|
||||||
|
programmer_map = {
|
||||||
|
'bash': 'programador de scripts Bash/Shell',
|
||||||
|
'python': 'desarrollador Python',
|
||||||
|
'fullstack': 'desarrollador Full Stack',
|
||||||
|
'frontend': 'desarrollador Frontend',
|
||||||
|
'backend': 'desarrollador Backend',
|
||||||
|
'devops': 'ingeniero DevOps',
|
||||||
|
'mobile': 'desarrollador de aplicaciones móviles',
|
||||||
|
'java': 'desarrollador Java',
|
||||||
|
'php': 'desarrollador PHP',
|
||||||
|
'ruby': 'desarrollador Ruby',
|
||||||
|
'dotnet': 'desarrollador .NET',
|
||||||
|
'data': 'científico de datos',
|
||||||
|
'ml': 'ingeniero de Machine Learning',
|
||||||
|
'cloud': 'arquitecto Cloud',
|
||||||
|
'odoo': 'desarrollador Odoo'
|
||||||
|
}
|
||||||
|
|
||||||
|
region_name = region_map.get(region_code, f'región con código {region_code}')
|
||||||
|
programmer_description = programmer_map.get(programmer_type, f'desarrollador {programmer_type}')
|
||||||
|
|
||||||
|
prompt = f"¿Cuál es la tarifa por hora promedio en dólares estadounidenses (USD) para un {programmer_description} en {region_name} en 2025? Responde solo con el valor numérico con dos decimales."
|
||||||
|
|
||||||
|
return prompt
|
||||||
|
|
||||||
|
def update_rate_files():
|
||||||
|
"""Actualiza los archivos de tarifas con datos de Perplexity."""
|
||||||
|
# Obtener modelo configurado
|
||||||
|
model = get_ai_model()
|
||||||
|
logger.info(f"Usando modelo de IA: {model}")
|
||||||
|
|
||||||
|
# Buscar todos los archivos .rate
|
||||||
|
rate_files = glob.glob(str(CONFIG_DIR / '*.rate'))
|
||||||
|
logger.info(f"Encontrados {len(rate_files)} archivos de tarifas para actualizar.")
|
||||||
|
|
||||||
|
for rate_file in rate_files:
|
||||||
|
programmer_type, region_code = parse_rate_filename(rate_file)
|
||||||
|
|
||||||
|
if programmer_type is None:
|
||||||
|
logger.warning(f"No se pudo analizar el nombre del archivo: {rate_file}, saltando.")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Si es un archivo especial como kdevs.rate
|
||||||
|
if region_code is None:
|
||||||
|
prompt = generate_prompt(programmer_type, None)
|
||||||
|
# Saltamos los archivos especiales que no necesitan actualización
|
||||||
|
if prompt is None:
|
||||||
|
logger.info(f"Saltando archivo especial: {os.path.basename(rate_file)}")
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
prompt = generate_prompt(programmer_type, region_code)
|
||||||
|
|
||||||
|
logger.info(f"Consultando tarifa para {programmer_type}" +
|
||||||
|
(f" en región {region_code}" if region_code else ""))
|
||||||
|
|
||||||
|
# Consultar a Perplexity
|
||||||
|
rate = query_perplexity(prompt, model)
|
||||||
|
|
||||||
|
if rate is not None:
|
||||||
|
# Guardar el resultado en el archivo
|
||||||
|
with open(rate_file, 'w') as f:
|
||||||
|
f.write(f"{rate:.2f}\n")
|
||||||
|
logger.info(f"Actualizado {os.path.basename(rate_file)} con valor: {rate:.2f}")
|
||||||
|
else:
|
||||||
|
logger.error(f"No se pudo obtener la tarifa para {programmer_type}" +
|
||||||
|
(f" en región {region_code}" if region_code else ""))
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
|
||||||
|
logger.info("Iniciando actualización de tarifas...")
|
||||||
|
|
||||||
|
# Verificar e instalar pycountry si es necesario
|
||||||
|
pycountry_available = check_install_pycountry()
|
||||||
|
|
||||||
|
# Actualizar la variable global
|
||||||
|
globals()['pycountry_available'] = pycountry_available
|
||||||
|
|
||||||
|
# Mostrar estado de validación de países
|
||||||
|
if pycountry_available:
|
||||||
|
logger.info("Validación de códigos de país habilitada con pycountry.")
|
||||||
|
else:
|
||||||
|
logger.info("Usando lista interna para validación básica de códigos de país.")
|
||||||
|
|
||||||
|
# Actualizar los archivos de tarifas
|
||||||
|
update_rate_files()
|
||||||
|
|
||||||
|
logger.info("Proceso de actualización de tarifas completado.")
|
126
bin/update.sh
126
bin/update.sh
|
@ -58,3 +58,129 @@ else
|
||||||
git pull
|
git pull
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Load bootstrap library for update functions
|
||||||
|
source $BIN_HOME/$BIN_LIBS/bootstrap.lib
|
||||||
|
|
||||||
|
# Update sistema operativo y repositorios primero (no requiere privilegios de root)
|
||||||
|
echo -e "\n${head_info}: Actualizando repositorios locales primero..."
|
||||||
|
|
||||||
|
# Función para actualizar Python y pip (requiere privilegios de superusuario)
|
||||||
|
update_python_and_pip() {
|
||||||
|
local BIN_PATH=$1
|
||||||
|
local LIBRARY=$2
|
||||||
|
local MESSAGES=$3
|
||||||
|
local UPDATE_LANG=$4
|
||||||
|
|
||||||
|
# No necesitamos cargar bootstrap.lib, ya que tenemos las funciones definidas internamente
|
||||||
|
|
||||||
|
# Definiciones internas para no depender de fuentes externas
|
||||||
|
function python3_update() {
|
||||||
|
echo "Actualizando Python..."
|
||||||
|
|
||||||
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
|
# En macOS, actualizamos Python a través de Homebrew
|
||||||
|
brew upgrade python
|
||||||
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
|
# En sistemas Debian y derivados
|
||||||
|
apt update
|
||||||
|
apt install --only-upgrade -y python3
|
||||||
|
elif [ -f /etc/redhat-release ]; then
|
||||||
|
# En sistemas Red Hat
|
||||||
|
dnf upgrade -y python3
|
||||||
|
elif [ -f /etc/arch-release ]; then
|
||||||
|
# En Arch Linux
|
||||||
|
pacman -Syu --noconfirm python
|
||||||
|
elif [ -f /etc/rc.conf ]; then
|
||||||
|
# En BSD
|
||||||
|
pkg upgrade -y python3
|
||||||
|
else
|
||||||
|
echo "Sistema operativo no soportado"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Python actualizado correctamente."
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
function pip_update() {
|
||||||
|
echo "Actualizando pip..."
|
||||||
|
|
||||||
|
# Actualizar pip usando el método adecuado según el sistema operativo
|
||||||
|
if [ "$(uname)" == "Darwin" ]; then
|
||||||
|
# En macOS, actualizamos pip a través de Homebrew
|
||||||
|
brew upgrade python-pip
|
||||||
|
elif [ -f /etc/debian_version ] || [ -f /etc/os-release ]; then
|
||||||
|
# En sistemas Debian y derivados
|
||||||
|
apt update
|
||||||
|
apt install --only-upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
elif [ -f /etc/redhat-release ]; then
|
||||||
|
# En sistemas Red Hat
|
||||||
|
dnf upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
elif [ -f /etc/arch-release ]; then
|
||||||
|
# En Arch Linux
|
||||||
|
pacman -Syu --noconfirm python-pip
|
||||||
|
elif [ -f /etc/rc.conf ]; then
|
||||||
|
# En BSD
|
||||||
|
pkg upgrade -y python3-pip
|
||||||
|
# También actualizar usando el propio pip
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
else
|
||||||
|
echo "Sistema operativo no soportado"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Pip actualizado correctamente."
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
echo -e "\n${head_info}: Actualizando Python y pip..."
|
||||||
|
|
||||||
|
# Update Python
|
||||||
|
command_installed python3
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
python3_update
|
||||||
|
else
|
||||||
|
echo -e "\n${head_error}: Python3 no está instalado. Ejecute bin/bootstrap.sh primero."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Update pip separately
|
||||||
|
command_installed pip3
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
pip_update
|
||||||
|
else
|
||||||
|
echo -e "\n${head_error}: pip3 no está instalado. Ejecute bin/bootstrap.sh primero."
|
||||||
|
echo -e "Esta dependencia es necesaria para varias herramientas del sistema."
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo -e "\n${head_info}: Actualización de Python y pip completada."
|
||||||
|
}
|
||||||
|
|
||||||
|
# Verificar si Python o pip están instalados y si se necesitan actualizar
|
||||||
|
command_installed python3
|
||||||
|
PYTHON_INSTALLED=$?
|
||||||
|
|
||||||
|
command_installed pip3
|
||||||
|
PIP_INSTALLED=$?
|
||||||
|
|
||||||
|
# Si cualquiera de ellos está instalado, necesitamos actualizar con privilegios de superusuario
|
||||||
|
if [ $PYTHON_INSTALLED -eq 0 ] || [ $PIP_INSTALLED -eq 0 ]; then
|
||||||
|
# Ejecutar la actualización de Python y pip con sudo
|
||||||
|
echo -e "\n${head_info}: Se necesitan privilegios de administrador para actualizar Python y pip..."
|
||||||
|
|
||||||
|
# Exportar todas las funciones necesarias para que estén disponibles en el subproceso sudo
|
||||||
|
# Esto incluye command_installed y otras funciones de base.lib que se necesitan
|
||||||
|
sudo bash -c "$(declare -f command_installed; declare -f update_python_and_pip); update_python_and_pip $BIN_HOME $BIN_LIBS $BIN_MESG $BIN_LANG"
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
echo -e "\n${head_error}: No se pudo actualizar Python y pip. Verifique sus privilegios."
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo -e "\n${head_info}: No se detectaron Python ni pip instalados. No hay actualizaciones que realizar."
|
||||||
|
echo -e "Para instalar estas dependencias, ejecute bin/bootstrap.sh primero."
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo -e "\n${head_info}: Proceso de actualización completado."
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue