my-cloud-devops-consulting 1.1.202507090955__tar.gz → 1.1.202507100004__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. {my-cloud-devops-consulting-1.1.202507090955/my_cloud_devops_consulting.egg-info → my-cloud-devops-consulting-1.1.202507100004}/PKG-INFO +1 -1
  2. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/app.py +46 -0
  3. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004/my_cloud_devops_consulting.egg-info}/PKG-INFO +1 -1
  4. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/my_cloud_devops_consulting.egg-info/SOURCES.txt +3 -0
  5. my-cloud-devops-consulting-1.1.202507100004/static/robots.txt +4 -0
  6. my-cloud-devops-consulting-1.1.202507100004/templates/forgot-password.html +47 -0
  7. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/login.html +20 -11
  8. my-cloud-devops-consulting-1.1.202507100004/templates/reset-password.html +9 -0
  9. my-cloud-devops-consulting-1.1.202507100004/test/test.py +21 -0
  10. my-cloud-devops-consulting-1.1.202507090955/test/test.py +0 -9
  11. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/MANIFEST.in +0 -0
  12. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/README.md +0 -0
  13. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/my_cloud_devops_consulting.egg-info/dependency_links.txt +0 -0
  14. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/my_cloud_devops_consulting.egg-info/requires.txt +0 -0
  15. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/my_cloud_devops_consulting.egg-info/top_level.txt +0 -0
  16. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/setup.cfg +0 -0
  17. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/setup.py +0 -0
  18. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/anifest.json +0 -0
  19. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/about.css +0 -0
  20. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/assessment.css +0 -0
  21. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/contact-form.css +0 -0
  22. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/index.css +0 -0
  23. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/login.css +0 -0
  24. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/main.css +0 -0
  25. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/register.css +0 -0
  26. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/css/services.css +0 -0
  27. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/facebook.png +0 -0
  28. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/favicon.svg +0 -0
  29. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/image.jpg +0 -0
  30. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/instagram.png +0 -0
  31. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/linkedin.png +0 -0
  32. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/images/whatsapp-navbar.png +0 -0
  33. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/counter.js +0 -0
  34. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/index.js +0 -0
  35. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/login.js +0 -0
  36. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/main.js +0 -0
  37. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/register.js +0 -0
  38. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/services.js +0 -0
  39. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/js/typing.js +0 -0
  40. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/static/sitemap.xml +0 -0
  41. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/about.html +0 -0
  42. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/assessment.html +0 -0
  43. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/base.html +0 -0
  44. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/contact-form.html +0 -0
  45. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/index.html +0 -0
  46. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/private-videos.html +0 -0
  47. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/register.html +0 -0
  48. {my-cloud-devops-consulting-1.1.202507090955 → my-cloud-devops-consulting-1.1.202507100004}/templates/services.html +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: my-cloud-devops-consulting
3
- Version: 1.1.202507090955
3
+ Version: 1.1.202507100004
4
4
  Summary: This is my consulting website for Cloud & DevOps services.
5
5
  Home-page: https://github.com/Betrand1999/project-root
6
6
  Author: Betrand Mutagha
@@ -196,6 +196,52 @@ def google_verification():
196
196
  def sitemap():
197
197
  return app.send_static_file('sitemap.xml')
198
198
 
199
+ @app.route('/robots.txt')
200
+ def robots():
201
+ return app.send_static_file('robots.txt')
202
+
203
+
204
+
205
+
206
+ # adding logic for passwords reset
207
+
208
+ from itsdangerous import URLSafeTimedSerializer
209
+
210
+ serializer = URLSafeTimedSerializer(SECRET_KEY)
211
+
212
+ @app.route('/forgot-password', methods=['GET', 'POST'])
213
+ def forgot_password():
214
+ if request.method == 'POST':
215
+ email = request.form['email']
216
+ user = users_collection.find_one({'username': email})
217
+ if user:
218
+ token = serializer.dumps(str(user['_id']), salt='password-reset')
219
+ reset_url = url_for('reset_password', token=token, _external=True)
220
+ send_email("Password Reset Request", email, f"Click here to reset your password: {reset_url}")
221
+ flash("Password reset link sent to your email.", "info")
222
+ else:
223
+ flash("User not found.", "error")
224
+ return render_template('forgot-password.html')
225
+
226
+
227
+ # adding logic for passwords reset
228
+
229
+ @app.route('/reset-password/<token>', methods=['GET', 'POST'])
230
+ def reset_password(token):
231
+ try:
232
+ user_id = serializer.loads(token, salt='password-reset', max_age=3600) # 1 hour
233
+ except:
234
+ flash("The reset link is invalid or has expired.", "danger")
235
+ return redirect(url_for('login'))
236
+
237
+ if request.method == 'POST':
238
+ new_password = request.form['password']
239
+ hashed = generate_password_hash(new_password)
240
+ users_collection.update_one({'_id': ObjectId(user_id)}, {'$set': {'password': hashed}})
241
+ flash("Password updated. Please log in.", "success")
242
+ return redirect(url_for('login'))
243
+
244
+ return render_template('reset-password.html')
199
245
 
200
246
 
201
247
  if __name__ == '__main__':
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: my-cloud-devops-consulting
3
- Version: 1.1.202507090955
3
+ Version: 1.1.202507100004
4
4
  Summary: This is my consulting website for Cloud & DevOps services.
5
5
  Home-page: https://github.com/Betrand1999/project-root
6
6
  Author: Betrand Mutagha
@@ -8,6 +8,7 @@ my_cloud_devops_consulting.egg-info/dependency_links.txt
8
8
  my_cloud_devops_consulting.egg-info/requires.txt
9
9
  my_cloud_devops_consulting.egg-info/top_level.txt
10
10
  static/anifest.json
11
+ static/robots.txt
11
12
  static/sitemap.xml
12
13
  static/css/about.css
13
14
  static/css/assessment.css
@@ -34,9 +35,11 @@ templates/about.html
34
35
  templates/assessment.html
35
36
  templates/base.html
36
37
  templates/contact-form.html
38
+ templates/forgot-password.html
37
39
  templates/index.html
38
40
  templates/login.html
39
41
  templates/private-videos.html
40
42
  templates/register.html
43
+ templates/reset-password.html
41
44
  templates/services.html
42
45
  test/test.py
@@ -0,0 +1,4 @@
1
+ User-agent: *
2
+ Allow: /
3
+
4
+ Sitemap: https://1beto.com/sitemap.xml
@@ -0,0 +1,47 @@
1
+ <!-- templates/forgot-password.html -->
2
+ {% extends 'base.html' %} {% block content %}
3
+ <div class="container">
4
+ <div class="row justify-content-center my-5">
5
+ <div class="col-md-6 mx-auto">
6
+ <h2 class="text-center">Forgot Password</h2>
7
+
8
+ {% with messages = get_flashed_messages(with_categories=true) %} {% if
9
+ messages %}
10
+ <div class="alert alert-dismissible fade show" role="alert">
11
+ {% for category, message in messages %}
12
+ <div
13
+ class="alert alert-{{ category }} alert-dismissible fade show"
14
+ role="alert"
15
+ >
16
+ {{ message }}
17
+ <button
18
+ type="button"
19
+ class="btn-close"
20
+ data-bs-dismiss="alert"
21
+ aria-label="Close"
22
+ ></button>
23
+ </div>
24
+ {% endfor %}
25
+ </div>
26
+ {% endif %} {% endwith %}
27
+
28
+ <form method="POST" class="p-4 border rounded bg-light">
29
+ <div class="mb-3">
30
+ <label for="email" class="form-label">Email address</label>
31
+ <input
32
+ type="email"
33
+ class="form-control"
34
+ id="email"
35
+ name="email"
36
+ placeholder="Enter your email"
37
+ required
38
+ />
39
+ </div>
40
+ <button type="submit" class="btn btn-primary w-100">
41
+ Send Reset Link
42
+ </button>
43
+ </form>
44
+ </div>
45
+ </div>
46
+ </div>
47
+ {% endblock %}
@@ -1,25 +1,31 @@
1
- {% extends "base.html" %}
2
-
3
- {% block main %}
1
+ {% extends "base.html" %} {% block main %}
4
2
 
5
3
  <div class="container">
6
- <div class="row justify-content-center my-5" style="width: 100%;">
7
- <div class="col-md-6 mx-auto"> <!-- Centering the form -->
4
+ <div class="row justify-content-center my-5" style="width: 100%">
5
+ <div class="col-md-6 mx-auto">
6
+ <!-- Centering the form -->
8
7
  <h2 class="text-center">Login</h2>
9
8
 
10
9
  <!-- Flash messages for feedback -->
11
- {% with messages = get_flashed_messages(with_categories=true) %}
12
- {% if messages %}
10
+ {% with messages = get_flashed_messages(with_categories=true) %} {% if
11
+ messages %}
13
12
  <div class="alert alert-dismissible fade show" role="alert">
14
13
  {% for category, message in messages %}
15
- <div class="alert alert-{{ category }} alert-dismissible fade show" role="alert">
14
+ <div
15
+ class="alert alert-{{ category }} alert-dismissible fade show"
16
+ role="alert"
17
+ >
16
18
  {{ message }}
17
- <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
19
+ <button
20
+ type="button"
21
+ class="btn-close"
22
+ data-bs-dismiss="alert"
23
+ aria-label="Close"
24
+ ></button>
18
25
  </div>
19
26
  {% endfor %}
20
27
  </div>
21
- {% endif %}
22
- {% endwith %}
28
+ {% endif %} {% endwith %}
23
29
 
24
30
  <form id="loginForm" method="post" class="p-4 border rounded bg-light">
25
31
  <div class="mb-3">
@@ -49,6 +55,9 @@
49
55
  <button type="submit" class="btn btn-primary w-100">Login</button>
50
56
  </form>
51
57
  <p class="mt-3 text-center">
58
+
59
+ <a href="{{ url_for('forgot_password') }}">Forgot Password?</a>
60
+
52
61
  Don't have an account?
53
62
  <a href="{{ url_for('register') }}">Register here</a>
54
63
  </p>
@@ -0,0 +1,9 @@
1
+ <!-- templates/reset-password.html -->
2
+ {% extends 'base.html' %} {% block content %}
3
+ <h2>Reset Password</h2>
4
+ <form method="POST">
5
+ <label>New Password:</label>
6
+ <input type="password" name="password" required />
7
+ <button type="submit">Set New Password</button>
8
+ </form>
9
+ {% endblock %}
@@ -0,0 +1,21 @@
1
+ class LibraryBook:
2
+ def __init__(self,title,author,year_published,genre,is_checked_out):
3
+ self.title=title
4
+ self.author=author
5
+ self.year_published=year_published
6
+ self.genre=genre
7
+ self.is_checked_out=is_checked_out
8
+
9
+ book_1=LibraryBook("The Alchemist","Paulo Coelho"," 2025","Fantasy,","False")
10
+ print(book_1.title)
11
+ print(book_1.author)
12
+ print(book_1.year_published)
13
+ print(book_1.genre)
14
+ print(book_1.is_checked_out)
15
+
16
+ book_2=LibraryBook("Betrand","Love book","2026"," Fiction","False")
17
+ print(book_2.title)
18
+ print(book_2.author)
19
+ print(book_2.year_published)
20
+ print(book_2.genre)
21
+ print(book_2.is_checked_out)
@@ -1,9 +0,0 @@
1
- def create_hero(name, power, city):
2
- return f"Captain {name} has the {power} of fire and protects New {city}"
3
- message = create_hero("Flames", "power","York")
4
- print(message)
5
-
6
- def hero_intro(hero_description):
7
- return message
8
- new_message = hero_intro(message)
9
- print(new_message)