From e335e9f5d921bd542d04e97da4fa3e2cc1c18e99 Mon Sep 17 00:00:00 2001 From: Alex Date: Sun, 9 Mar 2025 18:02:57 +0300 Subject: [PATCH] init --- .idea/.idea.Hard_Demo/.idea/.gitignore | 13 ++ .idea/.idea.Hard_Demo/.idea/encodings.xml | 4 + .idea/.idea.Hard_Demo/.idea/indexLayout.xml | 8 + .idea/.idea.Hard_Demo/.idea/vcs.xml | 6 + Hard_Demo/Models/Client.cs | 25 +++ Hard_Demo/Models/Employee.cs | 19 ++ Hard_Demo/Models/LastEnter.cs | 15 ++ Hard_Demo/Models/Order.cs | 27 +++ Hard_Demo/Models/Role.cs | 11 ++ Hard_Demo/Models/Service.cs | 17 ++ Hard_Demo/Models/User2Context.cs | 185 ++++++++++++++++++++ Hard_Demo/Program.cs | 22 +++ Library/Library.cs | 40 +++++ Library/Library.csproj | 9 + Tests/Tests.csproj | 27 +++ Tests/UnitTest1.cs | 68 +++++++ 16 files changed, 496 insertions(+) create mode 100644 .idea/.idea.Hard_Demo/.idea/.gitignore create mode 100644 .idea/.idea.Hard_Demo/.idea/encodings.xml create mode 100644 .idea/.idea.Hard_Demo/.idea/indexLayout.xml create mode 100644 .idea/.idea.Hard_Demo/.idea/vcs.xml create mode 100644 Hard_Demo/Models/Client.cs create mode 100644 Hard_Demo/Models/Employee.cs create mode 100644 Hard_Demo/Models/LastEnter.cs create mode 100644 Hard_Demo/Models/Order.cs create mode 100644 Hard_Demo/Models/Role.cs create mode 100644 Hard_Demo/Models/Service.cs create mode 100644 Hard_Demo/Models/User2Context.cs create mode 100644 Hard_Demo/Program.cs create mode 100644 Library/Library.cs create mode 100644 Library/Library.csproj create mode 100644 Tests/Tests.csproj create mode 100644 Tests/UnitTest1.cs diff --git a/.idea/.idea.Hard_Demo/.idea/.gitignore b/.idea/.idea.Hard_Demo/.idea/.gitignore new file mode 100644 index 0000000..5b4f415 --- /dev/null +++ b/.idea/.idea.Hard_Demo/.idea/.gitignore @@ -0,0 +1,13 @@ +# Default ignored files +/shelf/ +/workspace.xml +# Rider ignored files +/projectSettingsUpdater.xml +/contentModel.xml +/modules.xml +/.idea.Hard_Demo.iml +# Editor-based HTTP Client requests +/httpRequests/ +# Datasource local storage ignored files +/dataSources/ +/dataSources.local.xml diff --git a/.idea/.idea.Hard_Demo/.idea/encodings.xml b/.idea/.idea.Hard_Demo/.idea/encodings.xml new file mode 100644 index 0000000..df87cf9 --- /dev/null +++ b/.idea/.idea.Hard_Demo/.idea/encodings.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/.idea/.idea.Hard_Demo/.idea/indexLayout.xml b/.idea/.idea.Hard_Demo/.idea/indexLayout.xml new file mode 100644 index 0000000..7b08163 --- /dev/null +++ b/.idea/.idea.Hard_Demo/.idea/indexLayout.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/.idea.Hard_Demo/.idea/vcs.xml b/.idea/.idea.Hard_Demo/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/.idea.Hard_Demo/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Hard_Demo/Models/Client.cs b/Hard_Demo/Models/Client.cs new file mode 100644 index 0000000..de62b09 --- /dev/null +++ b/Hard_Demo/Models/Client.cs @@ -0,0 +1,25 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class Client +{ + public int Id { get; set; } + + public string Fio { get; set; } = null!; + + public int ClientCode { get; set; } + + public string Passport { get; set; } = null!; + + public DateOnly Birthday { get; set; } + + public string Address { get; set; } = null!; + + public string Email { get; set; } = null!; + + public string Password { get; set; } = null!; + + public int RoleId { get; set; } +} diff --git a/Hard_Demo/Models/Employee.cs b/Hard_Demo/Models/Employee.cs new file mode 100644 index 0000000..3906810 --- /dev/null +++ b/Hard_Demo/Models/Employee.cs @@ -0,0 +1,19 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class Employee +{ + public int EmployeId { get; set; } + + public int RoleId { get; set; } + + public string Fio { get; set; } = null!; + + public string EmployeLogin { get; set; } = null!; + + public string EmployePassword { get; set; } = null!; + + public string? EmployePhoto { get; set; } +} diff --git a/Hard_Demo/Models/LastEnter.cs b/Hard_Demo/Models/LastEnter.cs new file mode 100644 index 0000000..b600ef9 --- /dev/null +++ b/Hard_Demo/Models/LastEnter.cs @@ -0,0 +1,15 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class LastEnter +{ + public int EmployeId { get; set; } + + public DateTime EnterDatetime { get; set; } + + public string EnterType { get; set; } = null!; + + public string? Login { get; set; } +} diff --git a/Hard_Demo/Models/Order.cs b/Hard_Demo/Models/Order.cs new file mode 100644 index 0000000..a4e4797 --- /dev/null +++ b/Hard_Demo/Models/Order.cs @@ -0,0 +1,27 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class Order +{ + public int Id { get; set; } + + public string OrderCode { get; set; } = null!; + + public DateOnly OrderDate { get; set; } + + public TimeOnly OrderTime { get; set; } + + public int ClientCode { get; set; } + + public int ServiceId { get; set; } + + public string Status { get; set; } = null!; + + public DateOnly? DateClose { get; set; } + + public int RentalTime { get; set; } + + public virtual ICollection Idservices { get; set; } = new List(); +} diff --git a/Hard_Demo/Models/Role.cs b/Hard_Demo/Models/Role.cs new file mode 100644 index 0000000..1aaa0a7 --- /dev/null +++ b/Hard_Demo/Models/Role.cs @@ -0,0 +1,11 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class Role +{ + public int RoleId { get; set; } + + public string RoleName { get; set; } = null!; +} diff --git a/Hard_Demo/Models/Service.cs b/Hard_Demo/Models/Service.cs new file mode 100644 index 0000000..51b33c8 --- /dev/null +++ b/Hard_Demo/Models/Service.cs @@ -0,0 +1,17 @@ +using System; +using System.Collections.Generic; + +namespace Hard_Demo.Models; + +public partial class Service +{ + public int Id { get; set; } + + public string ServiceName { get; set; } = null!; + + public string ServiceCode { get; set; } = null!; + + public string ServiceCost { get; set; } = null!; + + public virtual ICollection Idorders { get; set; } = new List(); +} diff --git a/Hard_Demo/Models/User2Context.cs b/Hard_Demo/Models/User2Context.cs new file mode 100644 index 0000000..d77fe89 --- /dev/null +++ b/Hard_Demo/Models/User2Context.cs @@ -0,0 +1,185 @@ +using System; +using System.Collections.Generic; +using Microsoft.EntityFrameworkCore; + +namespace Hard_Demo.Models; + +public partial class User2Context : DbContext +{ + public User2Context() + { + } + + public User2Context(DbContextOptions options) + : base(options) + { + } + + public virtual DbSet Clients { get; set; } + + public virtual DbSet Employees { get; set; } + + public virtual DbSet LastEnters { get; set; } + + public virtual DbSet Orders { get; set; } + + public virtual DbSet Roles { get; set; } + + public virtual DbSet Services { get; set; } + + protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) +#warning To protect potentially sensitive information in your connection string, you should move it out of source code. You can avoid scaffolding the connection string by using the Name= syntax to read it from configuration - see https://go.microsoft.com/fwlink/?linkid=2131148. For more guidance on storing connection strings, see https://go.microsoft.com/fwlink/?LinkId=723263. + => optionsBuilder.UseNpgsql("Host=45.67.56.214;Port=5454;USERNAME=user2;DATABASE=user2;Password=hGcLvi0i"); + + protected override void OnModelCreating(ModelBuilder modelBuilder) + { + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.Id).HasName("clients_pk"); + + entity.ToTable("clients"); + + entity.Property(e => e.Id).HasColumnName("id"); + entity.Property(e => e.Address) + .HasColumnType("character varying") + .HasColumnName("address"); + entity.Property(e => e.Birthday).HasColumnName("birthday"); + entity.Property(e => e.ClientCode).HasColumnName("client_code"); + entity.Property(e => e.Email) + .HasColumnType("character varying") + .HasColumnName("email"); + entity.Property(e => e.Fio) + .HasColumnType("character varying") + .HasColumnName("fio"); + entity.Property(e => e.Passport) + .HasColumnType("character varying") + .HasColumnName("passport"); + entity.Property(e => e.Password) + .HasColumnType("character varying") + .HasColumnName("password"); + entity.Property(e => e.RoleId).HasColumnName("role_id"); + }); + + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.EmployeId).HasName("employees_pk"); + + entity.ToTable("employees"); + + entity.Property(e => e.EmployeId) + .ValueGeneratedNever() + .HasColumnName("employe_id"); + entity.Property(e => e.EmployeLogin) + .HasColumnType("character varying") + .HasColumnName("employe_login"); + entity.Property(e => e.EmployePassword) + .HasColumnType("character varying") + .HasColumnName("employe_password"); + entity.Property(e => e.EmployePhoto) + .HasColumnType("character varying") + .HasColumnName("employe_photo"); + entity.Property(e => e.Fio) + .HasColumnType("character varying") + .HasColumnName("fio"); + entity.Property(e => e.RoleId).HasColumnName("role_id"); + }); + + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.EmployeId).HasName("last_enter_pk"); + + entity.ToTable("last_enter"); + + entity.Property(e => e.EmployeId) + .ValueGeneratedNever() + .HasColumnName("employe_id"); + entity.Property(e => e.EnterDatetime) + .HasColumnType("timestamp without time zone") + .HasColumnName("enter_datetime"); + entity.Property(e => e.EnterType) + .HasColumnType("character varying") + .HasColumnName("enter_type"); + entity.Property(e => e.Login) + .HasColumnType("character varying") + .HasColumnName("login"); + }); + + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.Id).HasName("orders_pk"); + + entity.ToTable("orders"); + + entity.Property(e => e.Id).HasColumnName("id"); + entity.Property(e => e.ClientCode).HasColumnName("client_code"); + entity.Property(e => e.DateClose).HasColumnName("date_close"); + entity.Property(e => e.OrderCode) + .HasColumnType("character varying") + .HasColumnName("order_code"); + entity.Property(e => e.OrderDate).HasColumnName("order_date"); + entity.Property(e => e.OrderTime).HasColumnName("order_time"); + entity.Property(e => e.RentalTime).HasColumnName("rental_time"); + entity.Property(e => e.ServiceId).HasColumnName("service_id"); + entity.Property(e => e.Status) + .HasColumnType("character varying") + .HasColumnName("status"); + }); + + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.RoleId).HasName("roles_pk"); + + entity.ToTable("roles"); + + entity.Property(e => e.RoleId).HasColumnName("role_id"); + entity.Property(e => e.RoleName) + .HasColumnType("character varying") + .HasColumnName("role_name"); + }); + + modelBuilder.Entity(entity => + { + entity.HasKey(e => e.Id).HasName("service_pk"); + + entity.ToTable("service"); + + entity.HasIndex(e => e.ServiceCode, "service_unique").IsUnique(); + + entity.Property(e => e.Id) + .ValueGeneratedNever() + .HasColumnName("id"); + entity.Property(e => e.ServiceCode) + .HasColumnType("character varying") + .HasColumnName("service_code"); + entity.Property(e => e.ServiceCost) + .HasColumnType("character varying") + .HasColumnName("service_cost"); + entity.Property(e => e.ServiceName) + .HasColumnType("character varying") + .HasColumnName("service_name"); + + entity.HasMany(d => d.Idorders).WithMany(p => p.Idservices) + .UsingEntity>( + "OrderService", + r => r.HasOne().WithMany() + .HasForeignKey("Idorder") + .OnDelete(DeleteBehavior.ClientSetNull) + .HasConstraintName("order_service_orders_fk"), + l => l.HasOne().WithMany() + .HasForeignKey("Idservice") + .OnDelete(DeleteBehavior.ClientSetNull) + .HasConstraintName("order_service_service_fk"), + j => + { + j.HasKey("Idservice", "Idorder").HasName("order_service_pk"); + j.ToTable("order_service"); + j.IndexerProperty("Idservice").HasColumnName("idservice"); + j.IndexerProperty("Idorder").HasColumnName("idorder"); + }); + }); + + OnModelCreatingPartial(modelBuilder); + } + + partial void OnModelCreatingPartial(ModelBuilder modelBuilder); +} diff --git a/Hard_Demo/Program.cs b/Hard_Demo/Program.cs new file mode 100644 index 0000000..bb0743e --- /dev/null +++ b/Hard_Demo/Program.cs @@ -0,0 +1,22 @@ +using Avalonia; +using System; +using Hard_Demo; + +namespace demo_hard; + +class Program +{ + // Initialization code. Don't use any Avalonia, third-party APIs or any + // SynchronizationContext-reliant code before AppMain is called: things aren't initialized + // yet and stuff might break. + [STAThread] + public static void Main(string[] args) => BuildAvaloniaApp() + .StartWithClassicDesktopLifetime(args); + + // Avalonia configuration, don't remove; also used by visual designer. + public static AppBuilder BuildAvaloniaApp() + => AppBuilder.Configure() + .UsePlatformDetect() + .WithInterFont() + .LogToTrace(); +} \ No newline at end of file diff --git a/Library/Library.cs b/Library/Library.cs new file mode 100644 index 0000000..8b60da8 --- /dev/null +++ b/Library/Library.cs @@ -0,0 +1,40 @@ +using System; +using System.Collections.Generic; + +namespace Library +{ + public class Calculations + { + public string[] AvailablePeriods(TimeSpan[] startTimes, int[] durations, TimeSpan beginWorkingTime, TimeSpan endWorkingTime, int consultationTime) + { + TimeSpan consultationDuration = TimeSpan.FromMinutes(consultationTime); + List freeSlots = new List(); + TimeSpan current = beginWorkingTime; + + for (int i = 0; i < startTimes.Length; i++) + { + TimeSpan start = startTimes[i]; + TimeSpan end = start.Add(TimeSpan.FromMinutes(durations[i])); + + while (current.Add(consultationDuration) <= start) + { + freeSlots.Add($"{current:hh\\:mm}-{current.Add(consultationDuration):hh\\:mm}"); + current = current.Add(consultationDuration); + } + + if (current < end) + { + current = end; + } + } + + while (current.Add(consultationDuration) <= endWorkingTime) + { + freeSlots.Add($"{current:hh\\:mm}-{current.Add(consultationDuration):hh\\:mm}"); + current = current.Add(consultationDuration); + } + + return freeSlots.ToArray(); + } + } +} \ No newline at end of file diff --git a/Library/Library.csproj b/Library/Library.csproj new file mode 100644 index 0000000..3a63532 --- /dev/null +++ b/Library/Library.csproj @@ -0,0 +1,9 @@ + + + + net8.0 + enable + enable + + + diff --git a/Tests/Tests.csproj b/Tests/Tests.csproj new file mode 100644 index 0000000..cfb6060 --- /dev/null +++ b/Tests/Tests.csproj @@ -0,0 +1,27 @@ + + + + net8.0 + enable + enable + + false + true + + + + + + + + + + + + + + + + + + diff --git a/Tests/UnitTest1.cs b/Tests/UnitTest1.cs new file mode 100644 index 0000000..b63c38d --- /dev/null +++ b/Tests/UnitTest1.cs @@ -0,0 +1,68 @@ +using System; +using Xunit; +using Library; + +namespace Library.Tests +{ + public class CalculationsTests + { + [Fact] + public void AvailablePeriods_ReturnsFullDaySlots_WhenNoAppointments() + { + // Arrange + var calculations = new Calculations(); + TimeSpan[] startTimes = Array.Empty(); + int[] durations = Array.Empty(); + TimeSpan beginWorkingTime = new TimeSpan(9, 0, 0); + TimeSpan endWorkingTime = new TimeSpan(17, 0, 0); + int consultationTime = 30; + + // Act + var result = calculations.AvailablePeriods(startTimes, durations, beginWorkingTime, endWorkingTime, consultationTime); + + // Assert + Assert.NotEmpty(result); + Assert.Contains("09:00-09:30", result); + Assert.Contains("16:30-17:00", result); + } + + [Fact] + public void AvailablePeriods_ReturnsCorrectSlots_WhenThereAreAppointments() + { + // Arrange + var calculations = new Calculations(); + TimeSpan[] startTimes = { new TimeSpan(10, 0, 0), new TimeSpan(14, 0, 0) }; + int[] durations = { 60, 60 }; + TimeSpan beginWorkingTime = new TimeSpan(9, 0, 0); + TimeSpan endWorkingTime = new TimeSpan(17, 0, 0); + int consultationTime = 30; + + // Act + var result = calculations.AvailablePeriods(startTimes, durations, beginWorkingTime, endWorkingTime, consultationTime); + + // Assert + Assert.DoesNotContain("10:00-10:30", result); + Assert.DoesNotContain("14:00-14:30", result); + Assert.Contains("09:00-09:30", result); + Assert.Contains("15:30-16:00", result); + } + + [Fact] + public void AvailablePeriods_ReturnsEmpty_WhenNoFreeSlots() + { + // Arrange + var calculations = new Calculations(); + TimeSpan[] startTimes = { new TimeSpan(9, 0, 0) }; + int[] durations = { 480 }; // 8 hours (full workday) + TimeSpan beginWorkingTime = new TimeSpan(9, 0, 0); + TimeSpan endWorkingTime = new TimeSpan(17, 0, 0); + int consultationTime = 30; + + // Act + var result = calculations.AvailablePeriods(startTimes, durations, beginWorkingTime, endWorkingTime, consultationTime); + + // Assert + Assert.Empty(result); + } + } +}