import os
import shutil
import time
import logging

import stat
import subprocess

# Set up logging
logging.basicConfig(level=logging.INFO)
logger = logging.getLogger(__name__)

template_root = os.path.join(os.path.abspath(os.path.dirname(os.path.dirname(os.path.dirname(__file__)))), "templates")

TEMPLATES = {
    "root_dir": template_root,
    "templates": [
        {
            "name": "nextjs-nextjs-postresql",
            "install": [
                "npm run install:all"
            ],
            "start": [
                "npm run dev"
            ],
            "description": "A full-stack template using Next.js for the frontend, NestJS for the backend, and PostgreSQL as the database, suitable for building full-stack web applications with a modern, scalable architecture.",

            # common instruction is included in both frontend and backend instructions
            "common_instruction": """This project template uses Next.js for the frontend, NestJS for the backend, and PostgreSQL as the database.

Quick Start

1. Install dependencies:
   ```bash
   npm run install:all
   ```

2. Start services:
   ```bash
   # Start frontend and backend
   npm run dev
   ```

3. Access the applications:
   - Frontend: http://localhost:3000
   - Backend API: http://localhost:3001

Common Issues and Solutions

- Ensure all services are running on their correct ports. If ports are already in use, find and kill processes bound to them using `ss -tulnp | grep :PORT` and `kill -9 [PID]`
- Verify CORS configuration in `backend/src/main.ts` if frontend cannot connect to backend

IMPORTANT: The PostgreSQL database has been initialized automatically. You **must not** modify the database connection settings, or try to use a different database.""",

            # frontend instruction
            "frontend_instruction": """Frontend (Next.js)

Project Structure

```
frontend/src/
├── app/                  # App Router (pages and layouts)
│   ├── page.tsx         # Home page
│   ├── layout.tsx       # Root layout
│   └── ...
├── components/          # Reusable UI components
│   ├── Header.tsx
│   ├── Hero.tsx
│   ├── Features.tsx
│   └── Footer.tsx
├── lib/                 # Utility functions
│   └── api.ts          # API client
└── app/globals.css      # Global styles
```

Key directories and files:
- `frontend/src/app`: Create new pages or modify existing ones here. Follow Next.js App Router conventions (e.g., `page.tsx` for pages, `layout.tsx` for layouts).
- `frontend/src/components`: Build reusable UI components in this directory.
- `frontend/src/lib`: Place utilities for interacting with your backend API here.
- Styling: The project uses Tailwind CSS with the new `@import "tailwindcss"` syntax.
- Environment Variables: Configure frontend-specific environment variables in `frontend/.env.local`.

Styling

This project uses Tailwind CSS with the new `@import "tailwindcss"` syntax in `frontend/src/app/globals.css`. 
Tailwind configuration can be found in `frontend/tailwind.config.ts`.
Modify these files to customize the design system.

How to Set Colors Correctly:

1. Define colors in `frontend/tailwind.config.ts`, for example:
```
extend: {
    colors: {
    seashell: '#FFF5EE',
    crimson: '#DC143C',
    },
},
```
2. Use CSS variables in `frontend/src/app/globals.css` if you want the root-level defaults to match your theme, for example:
```
:root {
    --background: #FFF5EE; /* seashell */
    --foreground: #DC143C; /* crimson */
}
```
3. Apply colors using Tailwind classes in your JSX/TSX components, for example:
```
<div className="bg-seashell text-crimson">Content</div>
```

Coloring:
- The color of the background of ALL the pages must fit the color described in the **user instruction**.
- The color of ALL the components (e.g., buttons, links, header) must fit the color described in the **user instruction**.
- The color MUST have obvious contrast. For example, a light-colored component must have dark-colored text, and a dark-colored component must have light-colored text. The components and the background color must have similar contrast. ALL the texts, components, and backgrounds MUST have obvious contrast! 
- You should modify the tailwindcss classes and configuration in the codebase.

IMPORTANT: 
- When importing tailwind in files such as `globals.css`, ALWAYS use the new Tailwind import syntax `@import "tailwindcss";` at the start of the file.
- Do NOT use the three lines of old Tailwind import syntax `@tailwind base;`, `@tailwind components;`, and `@tailwind utilities;`. They will cause rendering failures and jumbled pages.

Development Workflow

Creating New Pages:
1. Create a new directory in `src/app/` (e.g., `src/app/about/`)
2. Add a `page.tsx` file with your page content
3. Add a `layout.tsx` file if needed for specific layouts

Creating Components:
When building new features, you should generate components that follow these guidelines:

Component Structure:
- Each component should be in its own file in `src/components/`
- Use TypeScript with proper typing
- Export components as default exports
- Use PascalCase for component names

REQUIRED COMPONENTS:
1. Header.tsx - Navigation bar with links
2. Hero.tsx - Main landing section
3. Features/Services/Products sections - Based on the request
4. Footer.tsx - Footer with links and info
5. App.tsx - Main component that imports and arranges all components
6. src/index.css - With @tailwind directives

Component Design Principles:
- Use Tailwind CSS classes for styling
- Maintain consistent spacing and padding
- Implement responsive design
- Add proper hover states and transitions

IMPORTANT:
- Make ALL these components match the content, theme and coloring of the website!
- Every page MUST have a header and footer!
- ALL content about the template or the technologies used MUST be removed from the final website!
- ALWAYS check that the pages referenced in the header and footer have actually been created!
- ALWAYS create detail pages for items listed on index pages!
- ALWAYS make sure that the frontend is correctly connected to the backend! Look carefully at the backend endpoints!
- Every form, search bar, link, and button MUST have real and functional reactions when user iteracts them. ALWAYS check this is true!

You'll primarily work within the `frontend/src`.""",

            # backend instruction
            "backend_instruction": f"""Backend (NestJS)

Project Structure

```
backend/src/
├── app.controller.ts      # Main application controller
├── app.module.ts         # Root application module
├── app.service.ts        # Main application service
└── main.ts              # Application entry point
```

Development Workflow:
1. Creating new module directory (`backend/src/<module-name>`)
2. Generate controller and service (`*.controller.ts`, `*.service.ts`)
3. Create entity (if needed; `*.entity.ts`)

Key directories and files:
- `backend/src/<module-name>`: Create new NestJS modules here for your features, each containing controllers, services, and entities.
- Controllers (`*.controller.ts`): Define your REST API endpoints.
- Services (`*.service.ts`): Implement your business logic.
- Entities (`*.entity.ts`): Define your data models and database schema using TypeORM.
- Database Configuration: Adjust database settings in `backend/src/app.module.ts` or via `backend/.env`.
- DTOs (`dto/*.dto.ts`): Define data transfer objects for validation and type safety.

IMPORTANT: Notice that a global prefix `/api/` has been set in `backend/src/main.ts`, so don't add it again in individual endpoints:
```
app.setGlobalPrefix('api');
```

Route-Order CAUTION:
- Always declare fixed/static routes BEFORE parameterised routes.
  Example:
    @Get('search')   // static, goes first
    search() {{ ... }}
    @Get(':id')      // dynamic, goes after the static route
    findOne() {{ ... }}
- Symptom if you forget: 
  - If you place `:id` first, calling “…/search” throws would try to match the string 'search' to `:id`, which will cause an error.
  - If you encounter a validation error and the type validations all seem correct, this might be the reason.

Database Configuration:

The application uses TypeORM with PostgreSQL. The database is running on port 5432. Configuration is loaded from environment variables in `backend/.env`:

```
DB_HOST=localhost
DB_PORT=5432
DB_USERNAME=myappuser
DB_PASSWORD=myapppassword
DB_NAME=myapp
```

IMPORTANT:
- When testing the implemented background endpoints, NEVER start the service in the shell. NEVER use `curl` to send the testing requests.
- ALWAYS use the specialized backend testing tool, `backend_test`, to make the tests.
- Continue testing the APIs using `backend_test` until ALL the APIs are correctly functioning!
- Every time a new API endpoint has been implemented, install the dependencies using `run_shell_command`, then call `backend_test` to test it.
- ALWAYS add some mock data if the data returned from the testing is empty.
- You should ALWAYS inject some testing data into the database to make sure that the return is not empty!

Note: You can inject test data into the database by running `sudo -u postgres psql -d myapp -c "[SQL_COMMAND]"` with the shell tool `run_shell_command`.
Examples:
```
# create table
sudo -u postgres psql -d myapp -c 'CREATE TABLE test_table (id SERIAL PRIMARY KEY, name VARCHAR(50) NOT NULL);'
# insert data
sudo -u postgres psql -d myapp -c "INSERT INTO test_table (name) VALUES ('Test Row');"
# view table
sudo -u postgres psql -d myapp -c "SELECT * FROM test_table;"
```

You'll primarily work within the `backend/src`."""
        }
    ],
}
