Version Control With Git

Module 1 of 11 0%

Version Control With Git

Introduction

In this article, we’ll explore:

  1. What Git Is
  2. Popular Git Hosting Services
  3. Other Version Control Systems
  4. Git Basics: Repositories, Branches, Commits, Pull Requests, & Conflicts
  5. Practical Steps: Setting Up Our Unsplash App Repository

1. What Is Git?

Git is a version control system (VCS)—a tool that keeps track of every change made to a set of files (usually code). Here’s an easy analogy:

Key Benefits of Git:


GitHub

GitLab

Bitbucket

(All three platforms do essentially the same thing: they host your Git repository and provide tools for collaboration. The choice often depends on personal preference or organizational requirements.)


3. Other Version Control Systems

Before Git became the dominant standard, other systems were widely used:

Even though these older systems exist, Git stands out today because it’s fast, flexible, and widely supported by modern development tools.


4. Detailed Git Concepts

4.1. Repositories

A Git repository (repo) is basically a folder (on your computer) that Git tracks. It contains:

You can have a local repository on your machine and a remote repository on GitHub or GitLab. You sync changes between them by using commands like push (send commits to the server) and pull (fetch and integrate commits from the server).

Example Visual (Local vs. Remote)

 Local Repository        Remote Repository
       +---+                   +-------+
       | .git (history)       | .git   |
       | project files        | master |
       +---+                   +-------+
         |      push/pull        ^
         |------------------------|

4.2. Working Directory, Staging Area, and Commits

Example Workflow

  1. Make changes to LoginViewController.swift.
  2. Stage them:

     git add LoginViewController.swift
    
    
  3. Commit them with a message:

     git commit -m "Implement login functionality"
    
    
  4. Push them to the remote repository (assuming your branch is main):

     git push origin main
    
    

ASCII Diagram (Working Directory → Staging Area → Repository)

+------------------+
| Working Directory|
|  (edited files)  |
+---------+--------+
          | git add
          v
+------------------+
|  Staging Area    |
|  (files ready    |
|   to commit)     |
+---------+--------+
          | git commit
          v
+------------------+
|   Local Repo     |
| (committed files)|
+------------------+
          | git push
          v
+------------------+
| Remote Repo (GitHub)
|  (shared repository) |
+------------------+

4.3. Branching

Branches in Git let you create an isolated environment for new features or fixes:

By branching, you don’t disturb the main codebase. Once your feature is tested, you can merge it into main. If the feature doesn’t work out, you can discard the branch without affecting other parts of the project.

ASCII Diagram (Basic Branching)

         (main)
           |
           o--- Commit 1
           |
           o--- Commit 2
           |\\
           | \\
           |  o--- Commit 3 (feature branch)
           |  |
           |  o--- Commit 4
           | /
           |/
           o--- Merge feature back into main
           |
           o--- Continue...

4.4. Pull Requests (PRs) / Merge Requests

A Pull Request (PR) is a request to merge your branch’s changes into another branch (often main). It’s your way of saying, “Hey team, I’ve finished this work. Please review and merge it.”

Why PRs Are Crucial:

ASCII Diagram (Pull Request Flow)

 Developer     GitHub (Remote)         Teammates
     |               |                      |
     |--- push --->  |                      |
     |               |--- PR creation ----->|
     |               |                      |
     |               |<---- Review, comment |
     |               |                      |
     |<--- feedback- |                      |
     |--- fix&push-> |                      |
     |               |--- Merge PR          |

4.5. Handling Merge Conflicts

A merge conflict happens when two people edit the same part of the same file. Git can’t automatically decide whose change to keep. For instance:

Conflict Resolution Steps:

  1. Open the file in conflict; you’ll see markers like <<<<<<< HEAD and >>>>>>> feature/login-screen.
  2. Decide which version of the code to keep (or combine parts of each).
  3. Delete the conflict markers and save the file.
  4. Stage the resolved file:

     git add LoginViewController.swift
    
    
  5. Commit the resolution with a message like:

     git commit -m "Resolve merge conflict in LoginViewController"
    
    
  6. Proceed with the merge.

4.6. The .gitignore File

A .gitignore file is a simple text file in your repository that tells Git which files or folders to ignore. This prevents unneeded or sensitive files (like secrets, credentials, build artifacts, or large cached data) from being tracked and pushed into the remote repository.

For iOS projects, a typical .gitignore might include:

# Xcode
build/
DerivedData/
*.xcuserdata/

# macOS
.DS_Store

# Swift Package Manager
/.build/
*.xcodeproj/

# Cocoapods
Pods/

# Carthage
Carthage/

# SwiftPM Packages
*.swiftpm/

# Other
*.log

How to Use It:

  1. Create a file named .gitignore in the root of your project.
  2. Paste in the patterns corresponding to files and folders you do not want to commit.
  3. Save and commit .gitignore early (ideally right after creating a repository) so extraneous files aren’t tracked at all.
  4. If you add entries later, you might need to manually untrack files that are already in the repo using git rm --cached <filename> if they were committed before.

Including a .gitignore file is considered best practice in almost every Git-based project, especially to keep your repository clean from build artifacts and large or sensitive files that don’t belong in version control.

4.7. Branching Strategies in Projects

Beyond creating simple feature branches, many development teams follow branching strategies to handle releases, bug fixes, and ongoing development more systematically. Here are a few common approaches:

4.7.1. Git Flow

Workflow in Git Flow:

  1. New Feature? Create a feature branch from develop.
  2. Feature Done? Merge feature branch into develop.
  3. Ready to Release? Create a release branch from develop.
  4. Release Branch Stabilized? Merge it back into both develop and main (the main now represents the new production release).
  5. Found a Critical Bug in Production? Create a hotfix branch from main, fix the bug, then merge it back to both main and develop.

This strategy is highly structured, good for large teams or projects with scheduled releases.

4.7.2. GitHub Flow

Workflow in GitHub Flow:

  1. Create a Branch for a feature or bug fix from main.
  2. Work & Commit changes in that branch.
  3. Open a Pull Request against main.
  4. Review & Merge: Once approved, merge the branch back into main.
  5. Deploy Immediately if your CI/CD pipeline is set up to do so.

This strategy is simpler and works well for teams practicing continuous delivery.

4.7.3. Trunk-Based Development

Trunk-Based Development focuses on rapid integration and continuous merges, minimizing long-lived branches.

4.7.4. Handling Bug Fixes & Releases

In Real Projects:


5. Homework (After Git Presentation)

  1. Clone the Repo & Set Up Remote Access
    • Each intern must clone or fork the UnsplashApp repo to their local machine.
    • Configure Git to work with HTTPS or SSH keys.
  2. Pick a Preferred Git App or Terminal
    • If you’re not comfortable with command-line Git, try using GitHub Desktop, Sourcetree, or Xcode’s built-in Git support.
  3. Commit and Make a Simple PR
    • Create a new branch (e.g., feature/add-contributor-name).
    • Edit a file (like README.md) to add your name under “Contributors.”
    • Stage and commit the change:

        git add README.md
        git commit -m "Add John Doe to Contributors"
      
      
    • Push your branch:

        git push origin feature/add-contributor-name
      
      
    • Open a PR on GitHub to merge feature/add-contributor-name into main.
    • Request a review from your team member.

Conclusion

With Git fundamentals—branching, commits, pull requests, conflict resolution —under your belt, you can collaborate confidently and keep your code organized.

Next, we’ll tackle Application Architecture, where we’ll explore clean code practices, SOLID principles, and how to use Swift Package Manager for modular iOS development. Get ready to structure your app the right way!