import tensorflow as tf
from tensorflow.keras import layers, models
from tensorflow.keras.preprocessing.image import ImageDataGenerator
# Define U-Net model with dropout
inputs = layers.Input(shape=(128, 128, 3))
# Encoder
c1 = layers.Conv2D(64, (3,3), activation='relu', padding='same')(inputs)
c1 = layers.Dropout(0.3)(c1)
c1 = layers.Conv2D(64, (3,3), activation='relu', padding='same')(c1)
p1 = layers.MaxPooling2D((2,2))(c1)
c2 = layers.Conv2D(128, (3,3), activation='relu', padding='same')(p1)
c2 = layers.Dropout(0.3)(c2)
c2 = layers.Conv2D(128, (3,3), activation='relu', padding='same')(c2)
p2 = layers.MaxPooling2D((2,2))(c2)
# Bottleneck
b = layers.Conv2D(256, (3,3), activation='relu', padding='same')(p2)
b = layers.Dropout(0.4)(b)
b = layers.Conv2D(256, (3,3), activation='relu', padding='same')(b)
# Decoder
u1 = layers.UpSampling2D((2,2))(b)
u1 = layers.concatenate([u1, c2])
c3 = layers.Conv2D(128, (3,3), activation='relu', padding='same')(u1)
c3 = layers.Dropout(0.3)(c3)
c3 = layers.Conv2D(128, (3,3), activation='relu', padding='same')(c3)
u2 = layers.UpSampling2D((2,2))(c3)
u2 = layers.concatenate([u2, c1])
c4 = layers.Conv2D(64, (3,3), activation='relu', padding='same')(u2)
c4 = layers.Dropout(0.3)(c4)
c4 = layers.Conv2D(64, (3,3), activation='relu', padding='same')(c4)
outputs = layers.Conv2D(3, (1,1), activation='sigmoid')(c4)
model = models.Model(inputs, outputs)
model.compile(optimizer=tf.keras.optimizers.Adam(learning_rate=0.0005), loss='mse')
# Data augmentation
train_datagen = ImageDataGenerator(horizontal_flip=True, vertical_flip=True, rotation_range=20)
# Assume X_train, y_train, X_val, y_val are prepared numpy arrays
# Early stopping
early_stop = tf.keras.callbacks.EarlyStopping(monitor='val_loss', patience=5, restore_best_weights=True)
# Train model
model.fit(train_datagen.flow(X_train, y_train, batch_size=32), epochs=50, validation_data=(X_val, y_val), callbacks=[early_stop])