aboutsummaryrefslogtreecommitdiff
path: root/src/get_cmd_path.c
blob: c35b3bc5bb55918c6379ec651f1e1225d62834ed (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/* ************************************************************************** */
/*                                                                            */
/*                                                        :::      ::::::::   */
/*   get_cmd_path.c                                     :+:      :+:    :+:   */
/*                                                    +:+ +:+         +:+     */
/*   By: chuhlig <chuhlig@student.42.fr>            +#+  +:+       +#+        */
/*                                                +#+#+#+#+#+   +#+           */
/*   Created: 2024/12/17 19:19:59 by chuhlig           #+#    #+#             */
/*   Updated: 2025/01/19 18:56:22 by chuhlig          ###   ########.fr       */
/*                                                                            */
/* ************************************************************************** */

#include "libft.h"
#include "minishell.h"
#include <errno.h>
#include <sys/errno.h>
#include <sys/stat.h>
#include <sys/unistd.h>
#include <unistd.h>

static char	*get_simple_cmd_path(char *cmd, int *return_code);
static char	*get_absolute_cmd_path(char *cmd, t_env *env, int *return_code);
static char	*find_in_path(char *cmd, t_env *env, int *return_code);
char		**get_split_path(t_env *env);
static int	is_directory(char *path);

char	*get_cmd_path(char *cmd, t_env *env, int *return_code)
{
	if (cmd[0] == '/')
		return (get_simple_cmd_path(cmd, return_code));
	else if (ft_strchr(cmd, '/'))
		return (get_absolute_cmd_path(cmd, env, return_code));
	else
		return (find_in_path(cmd, env, return_code));
}

static char	*get_absolute_cmd_path(char *cmd, t_env *env, int *return_code)
{
	char	*cur_dir;
	char	*result;

	cur_dir = ft_strjoin(env_get(env, "PWD"), "/");
	if (!cur_dir)
		return (NULL);
	result = ft_strjoin(cur_dir, cmd);
	free(cur_dir);
	if (!result)
		return (error(ENOENT, cmd, 127, return_code));
	if (access(result, F_OK) == -1)
	{
		free(result);
		return (error(ENOENT, cmd, 127, return_code));
	}
	if (access(result, X_OK) == -1)
	{
		free(result);
		return (error(EACCES, cmd, 126, return_code));
	}
	if (is_directory(cmd))
		return (error(EISDIR, cmd, 126, return_code));
	return (result);
}

static char	*find_in_path(char *cmd, t_env *env, int *return_code)
{
	char	*cur_path;
	char	*cmd_path;
	char	**path;

	path = get_split_path(env);
	cmd_path = NULL;
	while (*path)
	{
		if (cmd_path)
			free(cmd_path);
		cur_path = ft_strjoin(*path, "/");
		if (!cur_path)
			return (NULL);
		cmd_path = ft_strjoin(cur_path, cmd);
		free(cur_path);
		if (!cmd_path)
			return (NULL);
		if (access(cmd_path, X_OK) != -1)
			return (cmd_path);
		path++;
	}
	*return_code = 127;
	printf("%s:", cmd);
	ft_putstr_fd(" command not found", 2);//output is shit even if its fine for tester
	return (NULL);
}

static char	*get_simple_cmd_path(char *cmd, int *return_code)
{
	char	*result;

	result = ft_strdup(cmd);
	if (access(result, F_OK) == -1)
	{
		free(result);
		return (error(ENOENT, cmd, 127, return_code));
	}
	else if (access(result, X_OK) == -1)
	{
		free(result);
		return (error(EACCES, cmd, 126, return_code));
	}
	if (is_directory(cmd))
		return (error(EISDIR, cmd, 126, return_code));
	return (result);
}

static int	is_directory(char *path)
{
	struct stat	path_stat;

	stat(path, &path_stat);
	if ((path_stat.st_mode & S_IFMT) == S_IFDIR)
		return (1);
	else
		return (0);
}