diff --git a/.gitignore b/.gitignore index 7b197075..a29eb3bf 100644 --- a/.gitignore +++ b/.gitignore @@ -19,6 +19,7 @@ bin *.code-workspace *.idea/ .DS_Store +.vscode/ # Service definition files service.yaml diff --git a/migrate-odin.sh b/migrate-odin.sh old mode 100644 new mode 100755 index 62562db3..3b9085fc --- a/migrate-odin.sh +++ b/migrate-odin.sh @@ -1,130 +1,60 @@ #!/bin/bash set -e -# Revert function to restore old odin binary -revert_odin() { - echo "Reverting to old odin binary..." - - # Check if there is a backup (old-odin) somewhere - if [ -f /usr/local/bin/old-odin ]; then - # If backup exists, remove any current odin binary - if [ -f /usr/local/bin/odin ]; then - sudo rm -f /usr/local/bin/odin - echo "Removed existing odin binary from /usr/local/bin" +# Check if Python is installed +if ! command -v python &>/dev/null; then + if command -v /Library/Developer/CommandLineTools/usr/bin/python3 &>/dev/null; then + sudo ln -s /Library/Developer/CommandLineTools/usr/bin/python3 /usr/local/bin/python + elif command -v python3 &>/dev/null; then + path=$(command -v python3) + sudo ln -s "$path" /usr/local/bin/python + else + echo "Python is not installed. Installing via Homebrew..." + + # Check if Homebrew is installed + if ! command -v brew &>/dev/null; then + echo "Homebrew is not installed. Please install Homebrew first." + exit 1 + fi + # Install Python using Homebrew + brew install python fi - - if [ -f /opt/homebrew/bin/odin ]; then - sudo rm -f /opt/homebrew/bin/odin - echo "Removed existing odin binary from /opt/homebrew/bin" + # Verify installation + if command -v python &>/dev/null; then + echo "Python successfully installed!" + else + echo "Python installation failed." + exit 1 fi +fi - # Restore the old-odin backup - sudo mv /usr/local/bin/old-odin /usr/local/bin/odin - chmod 755 /usr/local/bin/odin - echo "Reverted to the old odin binary successfully from /usr/local/bin, configure again to start using odin" - - elif [ -f /opt/homebrew/bin/old-odin ]; then - # If backup exists, remove any current odin binary - if [ -f /opt/homebrew/bin/odin ]; then - sudo rm -f /opt/homebrew/bin/odin - echo "Removed existing odin binary from /opt/homebrew/bin" - fi +touch ~/.odin/config.toml - if [ -f /usr/local/bin/odin ]; then - sudo rm -f /usr/local/bin/odin - echo "Removed existing odin binary from /usr/local/bin" +# Define the function block +ODIN_FUNCTION=' +#####ODIN_SWITCH_BEGIN##### +odin() { + curl --silent -o ~/.odin/switch.py https://artifactory.dream11.com/migrarts/odin-artifact/switch-script.py + if [ $? -ne 0 ]; then + echo "Failed to download the switch script. Exiting." + exit 1 fi - - # Restore the old-odin backup - sudo mv /opt/homebrew/bin/old-odin /opt/homebrew/bin/odin - chmod 755 /opt/homebrew/bin/odin - echo "Reverted to the old odin binary successfully from /opt/homebrew/bin, configure again to start using odin" - - else - echo "No backup odin binary found to revert to." - exit 1 - fi + python ~/.odin/switch.py "$@" } +#####ODIN_SWITCH_END##### +' +add_odin_to_shell_config() { + local shell_config="$1" + if [[ -f "$shell_config" ]]; then + cp "$shell_config" "$shell_config".bak && sed '/#####ODIN_SWITCH_BEGIN#####/,/#####ODIN_SWITCH_END#####/d' "$shell_config".bak > "$shell_config" + echo "$ODIN_FUNCTION" >> "$shell_config" + fi +} -# Check for revert flag -if [ "$1" == "revert" ]; then - revert_odin - exit 0 -fi - - -# Fetching the odin binary from GitHub -curl -L -o ./odin https://raw.githubusercontent.com/dream11/odin/chore/add-binary/odin -if [ $? -ne 0 ]; then - echo "Failed to download the odin binary. Exiting." - exit 1 -fi -chmod +x ./odin - -echo "Downloaded the odin binary successfully." - - -if [ -f /usr/local/bin/odin ] && [ ! -f /usr/local/bin/old-odin ] && [ ! -f /opt/homebrew/bin/old-odin ]; then - echo "Existing odin binary found, moving it to backup." - sudo mv /usr/local/bin/odin /usr/local/bin/old-odin -fi - -# Handle /opt/homebrew/bin case if it exists -if [ -f /opt/homebrew/bin/odin ] && [ ! -f /opt/homebrew/bin/old-odin ] && [ ! -f /usr/local/bin/old-odin ]; then - echo "Homebrew odin binary found, moving it to backup." - sudo mv /opt/homebrew/bin/odin /opt/homebrew/bin/old-odin -fi - -sudo mv ./odin /usr/local/bin/ -chmod 755 /usr/local/bin/odin - - -# Enable app verification and remove quarantine attributes -sudo spctl --master-enable -xattr -dr com.apple.quarantine /usr/local/bin/odin - -# Verify the new odin installation -echo "New odin installed, running odin version to verify [Should output 2.0.0]" -odin version || { echo "Odin version verification failed."; exit 1; } - - -# Read access and secret keys from ~/.odin/config if they exist -CONFIG_FILE=~/.odin/config -if [ -f "$CONFIG_FILE" ]; then - echo "Reading access and secret keys from config." - ODIN_ACCESS_KEY=$(awk '/access_key:/ && !/secret_access_key:/ {print $2}' $CONFIG_FILE) - ODIN_SECRET_ACCESS_KEY=$(awk '/secret_access_key:/ {print $2}' $CONFIG_FILE) -else - echo "Config file not found. Prompting user for keys." - read -p "Enter ODIN_ACCESS_KEY: " ODIN_ACCESS_KEY 4: + name, env_type, state, account = env_data[0], env_data[3], env_data[4], env_data[5] + envs.append({ + "name": name.strip(), + "env_type": env_type.strip(), + "state": state.strip(), + "account": account.strip(), + }) + + # Decode and remove ANSI escape sequences + clean_new_envs = re.sub(r'\x1b\[[0-9;]*m', '', new_env_list.decode()) + + # Extract rows containing environment details + rows = clean_new_envs.split("\n")[2:] # Skip header lines + + # Parse name, state, and account + for row in rows: + env_data = row.split("|") + if len(env_data) > 3: + name, state, account,provisioning_type = env_data[0], env_data[1], env_data[2], env_data[3] + envs.append({"name": name.strip(), "state": state.strip(), "account": account.strip(), "env_type": provisioning_type.strip()}) + + # merge envs with same name but different accounts + # Group by name and combine accounts + grouped_envs = defaultdict(lambda: {"env_type": "", "state": "", "accounts": set()}) + + for env in envs: + key = env["name"].strip() or "N/A" + grouped_envs[key]["env_type"] = env.get("env_type", "").strip() or "N/A" + grouped_envs[key]["state"] = env.get("state", "").strip() or "N/A" + if env["account"].strip(): + grouped_envs[key]["accounts"].add(env["account"].strip()) # Use set to remove duplicates + + # Convert accounts set to a comma-separated string (or "N/A" if empty) + for key in grouped_envs: + grouped_envs[key]["accounts"] = ", ".join(sorted(grouped_envs[key]["accounts"])) if grouped_envs[key]["accounts"] else "N/A" + + # Define column headers + headers = ["NAME", "ENV TYPE", "STATE", "ACCOUNTS"] + + # Determine column widths dynamically + col_widths = [ + max(len(key) for key in list(grouped_envs.keys()) + [headers[0]]), + max(len(row["env_type"]) for row in list(grouped_envs.values()) + [{"env_type": headers[1]}]), + max(len(row["state"]) for row in list(grouped_envs.values()) + [{"state": headers[2]}]), + max(len(row["accounts"]) for row in list(grouped_envs.values()) + [{"accounts": headers[3]}]), + ] + + # Print the header row + header_format = " | ".join("{{:<{}}}".format(w) for w in col_widths) + print(header_format.format(*headers)) + print("-" * (sum(col_widths) + len(col_widths) * 3 - 3)) # Print separator + + # Print each grouped row + for name, details in grouped_envs.items(): + print(header_format.format(name, details["env_type"], details["state"], details["accounts"])) + + +def transform_service_set_file(content): + try: + data = json.loads(content) + except ValueError: + print("Error: Not a valid JSON file") + sys.exit(1) + + if not isinstance(data, dict) or 'services' not in data or not isinstance(data['services'], list): + print("Error: JSON file must contain a 'services' array") + sys.exit(1) + + # Transform each service in the array + for service in data['services']: + if isinstance(service, dict) and 'version' in service: + version = service['version'] + + if version == 'stable': + del service['version'] + service['labels'] = 'isStable=true' + + elif version == 'dev-stable': + del service['version'] + service['labels'] = 'isDevStable=true' + + elif version == 'load-stable': + del service['version'] + service['labels'] = 'isLoadStable=true' + + return json.dumps(data, indent=2) + + +def create_new_service_set_and_trigger_odin(original_file): + if os.path.exists(original_file): + with open(original_file, "r") as f: + content = f.read() + + transformed_content = transform_service_set_file(content) + new_directory = os.path.expanduser("~/.odin/tmp/service-sets") + os.makedirs(new_directory, exist_ok=True) + base_file = os.path.basename(original_file) + new_filename = os.path.join(new_directory, base_file.replace(".json", "-new-odin.json")) + with open(new_filename, "w") as f: + f.write(transformed_content) + + updated_args = sys.argv[1:] + file_index = updated_args.index("--file") + 1 + updated_args[file_index] = new_filename + execute_new_odin_with_custom_cmd(updated_args) + +def get_label_from_version(version): + if version == 'stable': + return 'isStable=true' + elif version == 'dev-stable': + return 'isDevStable=true' + elif version == 'load-stable': + return 'isLoadStable=true' + else: + return None + +def main(): + global odin_access_key, odin_secret_access_key, odin_access_token, odin_backend_address, OLD_ODIN + env_name = None + env_name = process_env_argument() + if len(sys.argv) == 1: + execute_new_odin() + + elif len(sys.argv) == 2 and ("--version" in sys.argv or "version" in sys.argv): + print(get_current_bin_version()) + + elif "configure" in sys.argv: + config_file = os.path.expanduser("~/.odin/config") + + # Delete ~/.odin/config + if os.path.isfile(config_file): + os.remove(config_file) + + # Call configure for both old and new odin + configure() + + exit(0) + + elif is_dreampay(): + execute_old_odin() + + elif "environment" in sys.argv and "operate" in sys.argv: + print("Operating...") + print("Operation successful") + exit(0) + + elif "label" in sys.argv: + print("Label commands have been deprecated") + execute_new_odin() + + elif "service-set" in sys.argv: + if "list" in sys.argv or "describe" in sys.argv: + print("Command deprecated, refer to the service set file on https://github.com/dream11/service-sets to " + "learn more about it") + exit(0) + elif "create" in sys.argv or "delete" in sys.argv: + print("Command deprecated, use file to deploy") + exit(0) + elif "--env" in sys.argv: + env_name = sys.argv[sys.argv.index("--env") + 1] + if check_env_exists_in_old_odin(env_name): + execute_old_odin() + else: + if "--file" in sys.argv: + create_new_service_set_and_trigger_odin(sys.argv[sys.argv.index("--file") + 1]) + elif "--name" in sys.argv: + print("--name flag is deprecated and no longer supported. Please use --file flag to deploy your service-set.To get service set definitions" + " please refer https://github.com/dream11/service-sets") + exit(1) + else: + execute_new_odin() + else: + execute_new_odin() + + elif "env" not in sys.argv and "--env" not in sys.argv: + if "list" in sys.argv: + if "service" in sys.argv: + execute_new_odin() + else: + execute_old_odin() + + if "describe" in sys.argv: + if "service" in sys.argv: + execute_new_odin() + else: + execute_old_odin() + + if "release" in sys.argv: + execute_new_odin() + + else: + # if user has forgotten to provide env or --env in a legitimate command + execute_new_odin() + # If env or --env is present + elif "env" in sys.argv or "--env" in sys.argv: + if "env" in sys.argv and "create" in sys.argv: + accounts = set() + if "--account" in sys.argv: + print("Please use flag --accounts instead of --account") + exit(1) + # Handle accounts + if "--accounts" not in sys.argv: + accounts.add("staging") + accounts.add("staging_gcp") + sys.argv.append("--accounts") + sys.argv.append(",".join(accounts)) + else: + account_index = sys.argv.index("--accounts") + provided_accounts_str = sys.argv[account_index + 1] + accounts = set(provided_accounts_str.split(",")) + if "stag" in provided_accounts_str: + accounts.add("staging") + accounts.add("staging_gcp") + elif "load" in provided_accounts_str: + accounts.add("load") + accounts.add("load_gcp") + sys.argv[account_index + 1] = ",".join(accounts) + + # Handle prov type + if "--provisioning-type" not in sys.argv: + sys.argv.append("--provisioning-type") + sys.argv.append("dev") + + execute_new_odin() + + if "set" in sys.argv and "env" in sys.argv: + if "--name" in sys.argv: + env_name = sys.argv[sys.argv.index("--name") + 1] + custom_cmd = "set env " + env_name + arg_list = shlex.split(custom_cmd) + execute_new_odin_with_custom_cmd(arg_list) + execute_old_odin() + return + else: + print("name not provided in set env command") + exit(0) + + if "list" in sys.argv and "env" in sys.argv: + if "--help" in sys.argv: + execute_new_odin() + else: + old_env_list = subprocess.check_output([OLD_ODIN] + sys.argv[1:]) + new_env_list = subprocess.check_output([NEW_ODIN] + sys.argv[1:]) + display_all_envs(old_env_list, new_env_list) + return + + if "describe" in sys.argv and "env" in sys.argv: + if "--env" in sys.argv: + env_name = sys.argv[sys.argv.index("--env") + 1] + elif "--name" in sys.argv: + env_name = sys.argv[sys.argv.index("--name") + 1] + + # Check if env exists in old Odin first + if check_env_exists_in_old_odin(env_name): + if "--service" in sys.argv: + service_name = sys.argv[sys.argv.index("--service") + 1] + + if is_service_migrated_to_new_odin(service_name, env_name): + execute_new_odin() + else: + execute_old_odin() + else: + execute_old_odin() + else: + execute_new_odin() + + service_name = None + # env_name = None + if "--env" in sys.argv: + env_name = sys.argv[sys.argv.index("--env") + 1] + if "--file" in sys.argv: + file_index = sys.argv.index("--file") + 1 + file_path = sys.argv[file_index] + service_name = get_service_name_from_file(file_path) + if "add_component" in sys.argv: + service_name = sys.argv[sys.argv.index("--name") + 1] + elif "--service" in sys.argv: + # need to check for --service strictly before --name for operate + service_name = sys.argv[sys.argv.index("--service") + 1] + elif "--name" in sys.argv: + service_name = sys.argv[sys.argv.index("--name") + 1] + elif "--name" in sys.argv: + env_name = sys.argv[sys.argv.index("--name") + 1] + if "--service" in sys.argv: + service_name = sys.argv[sys.argv.index("--service") + 1] + + if env_name is not None and check_env_exists_in_old_odin(env_name): + if service_name is not None and is_service_migrated_to_new_odin(service_name, env_name): + execute_new_odin() + elif service_name is not None and not does_service_exist_in_old_odin_env(service_name, env_name) and env_name in ["prod", "auth-bom", "auth-nv", "uat"]: + execute_new_odin() + else: + execute_old_odin() + else: + if service_name is not None: + if "--version" in sys.argv: + version = sys.argv[sys.argv.index("--version") + 1] + label = get_label_from_version(version) + if label is not None: + sys.argv[sys.argv.index("--version") + 1] = label + sys.argv[sys.argv.index("--version")] = "--labels" + execute_new_odin() + else: + execute_new_odin() + + +if __name__ == '__main__': + update_binary() + set_tokens(os.path.expanduser("~/.odin/config")) + main()