Определите, к каким ресурсам SMB у меня есть доступ на чтение и/или запись

Определите, к каким ресурсам SMB у меня есть доступ на чтение и/или запись

Для ряда целевых объектов (IP-адресов) я хотел бы определить, к каким общим ресурсам SMB у моей учетной записи нет доступа, к каким у нее есть доступ на чтение, а к каким — доступ на чтение/запись.

В настоящее время я использую smbclient. Команда, которую я запускаю первой, это

smbclient -L [targetIP] -U [user] -p 445

Это дает мне список акций. Например;

        Sharename       Type      Comment
        ---------       ----      -------
        ADMIN$          Disk      Remote Admin
        C$              Disk      Default share
        IPC$            IPC       Remote IPC
        print$          Disk      Printer Drivers
        MySecrets       Disk

Затем я могу подключиться к файловому ресурсу с помощью этой команды

smbclient //[target]/[name_of_share_from_list] -U [user] -p 445

Что приводит к подсказке SMB. Из подсказки, которую я печатаю, lsи если я вижу файлы, я знаю, что у меня есть доступ на чтение. Я предполагаю, что мне нужно отправить файл, чтобы проверить, есть ли у меня доступ на запись.

Это утомительно. Как мне автоматизировать это так, чтобы для заданного списка целей я получал список всех акций и уровень доступа к ним моей учетной записи?

решение1

У вас уже была проделана большая часть работы. Чтение страницы руководства smbclientдало бы вам -c <command>параметр, который можно использовать для предоставления одной или нескольких команд напрямую, а не интерактивно.

#!/bin/bash
username="DOMAIN\\USER"    # Double backslash
password="PASSWORD"        # For demonstration purposes only
hostname="TARGET_HOST"     # SMB hostname of target

cd "${TMPDIR:-/tmp}"
touch tmp_$$.tmp           # Required locally to copy to target

smbclient -L "$hostname" -g -A <( echo "username=$username"; echo "password=$password" ) 2>/dev/null |
    awk -F'|' '$1 == "Disk" {print $2}' |
    while IFS= read -r share
    do
        echo "Checking root of share '$share'"

        if smbclient "//$hostname/$share/" "$password" -U "$username" -c "dir" >/dev/null 2>&1
        then
            status=READ

            # Try uprating to read/write
            if smbclient "//$hostname/$share/" "$password" -U "$username" -c "put tmp_$$.tmp ; rm tmp_$$.tmp" >/dev/null 2>&1
            then
                status=WRITE
            fi
        else
            status=NONE
        fi

        case "$status" in
            READ) echo "Well, $username has read access" ;;
            WRITE) echo "Yes, $username has write access" ;;
            *) echo "No, $username has no access" ;;
        esac
    done

rm -f tmp_$$.tmp

решение2

Может бытьSMBкартаможет помочь вам с этой задачей. Его автор разработал его именно для этой цели.

$  python smbmap.py -H [targetIP] -u [user] -P 445
[+] Finding open SMB ports....
[+] User SMB session establishd...
[+] IP: 192.168.0.4:445 Name: 192.168.0.4
        Disk                                                    Permissions
        ----                                                    -----------
        ADMIN$                                                  READ, WRITE
        C$                                                      READ, WRITE
        IPC$                                                    NO ACCESS
        TMPSHARE                                                READ, WRITE     

Вы можете использовать --host-fileдля передачи списка целей.

Внутри он пытается создать каталог со случайным именем, чтобы проверить, есть ли у нас права на запись.

решение3

Тот же подход, что и в принятом ответе, но с использованиемansible. Затем его можно использовать для монтирования в режиме только для чтения или чтения-записи. (Что затем делает другие инструменты, такие какmergerfsработают как и ожидалось при создании объединения файловых систем)

- name: get available samba shares
  ansible.builtin.shell:
    cmd: "smbclient -L //{{ samba_ip }} -U % -g | grep Disk"
  register: samba_shares_raw

- name: extract samba shares
  ansible.builtin.set_fact:
    samba_shares: "{{ (samba_shares | default([])) + [ ( item | split('|') )[1] ] }}"
  loop: "{{ samba_shares_raw.stdout_lines }}"

- name: send a file to test ro / rw access
  ansible.builtin.file:
    path: "{{ ansible_env.HOME }}/samba_client"
    state: touch

- name: test writeable samba
  ansible.builtin.shell:
    cmd: "smbclient -U % //{{ samba_ip }}/{{ item }} -c 'put samba_client'"
    chdir: "{{ ansible_env.HOME }}"
  register: writeable_raw
  ignore_errors: true
  loop: "{{ samba_shares }}"

- name: extract writeable results
  ansible.builtin.set_fact:
    writeable: "{{ writeable | default({}) | combine({ item.item : not item.failed }) }}"
  loop: "{{ writeable_raw.results }}"
  loop_control:
    label: "{{ item.item }}"

- name: samba mount directories
  ansible.builtin.file:
    state: directory
    recurse: true
    path: "{{ ansible_env.HOME }}/samba/{{ item }}"
  loop: "{{ samba_shares }}"

- name: mount samba shares
  ansible.posix.mount:
    src: "//{{ samba_ip }}/{{ item }}"
    path: "{{ ansible_env.HOME }}/samba/{{ item }}"
    fstype: cifs
    opts: "{{ 'rw' if writeable[item] else 'ro' }},guest,uid=1000,gid=1000"
    state: mounted
  loop: "{{ samba_shares }}"
  become: true

Связанный контент