Skip to content

TextWatermark

TextWatermark

Main Class of TextWatermark

TextWatermark

A class to insert watermark into plain text

Example:
Python
import os

from textwatermark.defines import WMMode
from textwatermark.main import TextWatermark
from textwatermark.templates import homograph_numbers

wm_mode = WMMode.REAL_NUMBER
template = homograph_numbers
wm_max = '9'*9
confusables_chars_key = ''

# init
wm = TextWatermark(wm_mode)


wm = set_tpl(template.CONFUSABLES_CHARS,
             template.method,
             confusables_chars_key)
wm.set_wm_max(wm_max)
wm.set_text_file(os.path.abspath(
    os.path.dirname(__file__)+'/../tests/text/1.txt'))

# Insert watermark to text
wm_str = '123456789'

wm_text = wm.insert_watermark(wm_str)

# Save the parameters to retrieve the watermark
params = wm.export_params()

# retrieve the watermark
wm_out_str = TextWatermark.retrieve_watermark(wm_text, params)
print(wm_text)

assert wm_out_str == wm_str
Source code in textwatermark/main.py
Python
class TextWatermark:
    """
    A class to insert watermark into plain text

    Example:
    --------
    ```py
    import os

    from textwatermark.defines import WMMode
    from textwatermark.main import TextWatermark
    from textwatermark.templates import homograph_numbers

    wm_mode = WMMode.REAL_NUMBER
    template = homograph_numbers
    wm_max = '9'*9
    confusables_chars_key = ''

    # init
    wm = TextWatermark(wm_mode)


    wm = set_tpl(template.CONFUSABLES_CHARS,
                 template.method,
                 confusables_chars_key)
    wm.set_wm_max(wm_max)
    wm.set_text_file(os.path.abspath(
        os.path.dirname(__file__)+'/../tests/text/1.txt'))

    # Insert watermark to text
    wm_str = '123456789'

    wm_text = wm.insert_watermark(wm_str)

    # Save the parameters to retrieve the watermark
    params = wm.export_params()

    # retrieve the watermark
    wm_out_str = TextWatermark.retrieve_watermark(wm_text, params)
    print(wm_text)

    assert wm_out_str == wm_str
    ```

    """

    text: str
    """Text to be watermarked.

        Set text by `self.set_text(text)` or `self.set_text_file(file_path)`
    """

    wmc: WMConversion
    """Instance of WMConverison"""
    wmt: WMTemplate
    """Instance of WMTemplate"""

    wm_max: str
    """Maximum watermark string."""

    wm_fixed_len: int = 0
    """The longest (or largest) watermark string.
        When wm_flag_bit is true,
        The value of wm_max_len is 1 larger the actual value """

    tpl_type: Optional[WMTemplateType] = None
    """template type in WMTemplateType"""

    wm_flag_bit: bool = True
    """If set a flag bit when insert watermark or not"""

    def __init__(
        self,
        wm_mode: WMMode,
        wm_base: int = 0,
        start_at: int = 0,
        wm_loop: bool = False,
        wm_flag_bit: bool = True,
    ):
        """Class TextWatermark init

        Args:
            wm_mode (WMMode): WMMode to be set
            wm_base (int): Watermark convert base.
            start_at (int, optional): index where the watermark will be inserted. Defaults to 0.
            wm_loop (bool, optional): If watermark will be repeated until it is inserted to end.
                Defaults to False.
            wm_flag_bit (bool, optional): If True, add a flag bit to watermark. Defaults to True.

        Raises:
            ValueError: if `base` is not in the range [2, 36]
        """

        self.wm_mode = wm_mode
        """Watermark encoding mode. Setting different watermark modes will effectively
        reduce the size of the watermark string."""

        self.wm_base = wm_base
        """watermark convert base must be between 2 and 36,
            and less or euqal than len(list({template}.CONFUSABLES_CHARS.values())[0])
        """

        self.start_at = start_at
        """index where the watermark will be inserted. Defaults to 0."""
        self.wm_loop = wm_loop
        """If watermark will be repeated until it is inserted to end.
                Defaults to False."""
        self.wm_flag_bit = wm_flag_bit
        """If True, add a flag bit to watermark. Defaults to True."""

    def set_tpl(
        self,
        confusables_chars: Union[dict, list],
        method: WMMethod,
        confusables_chars_key: str = "",
    ):
        """Set template arguments and get instance of WMTemplate

        Args:
            confusables_chars (Union[dict, list]): Confusables chars
            method (WMMethod): Method of watermark
            confusables_chars_key (str): Confusables chars key

        """

        self.wmt = WMTemplate(
            confusables_chars, self.wm_base, method, confusables_chars_key
        )

        if self.wm_base == 0:
            self.wm_base = self.wmt.wm_base

    def set_tpl_type(self, tpl_type: WMTemplateType, confusables_chars_key: str = ""):
        """Set template type

        Args:
            tpl_type (WMTemplateType): Template type
            confusables_chars_key (str): Confusables chars key

        """
        template = tpl_type.value

        self.wmt = WMTemplate(
            template.CONFUSABLES_CHARS,
            self.wm_base,
            template.method,
            confusables_chars_key,
        )

        if self.wm_base == 0:
            self.wm_base = self.wmt.wm_base

        self.tpl_type = tpl_type

    def set_wm_max(self, wm_max: str):
        """Set the length of the longest (or largest) watermark string.

            Since the input watermark string is variable in length,
            this will bring trouble to the retrieval of the watermark:
            it will be difficult for us to determine the boundary of the watermark,
            or judge the loss of the watermark. Therefore, in this library,
            we will calculate and set the length of the longest (or largest)
            watermark string as the default length of the watermark string.

        Args:
            wm_max (str): The longest (or largest) watermark string.

        """
        if not hasattr(self, "wmt") or not self.wmt:
            raise ValueError(
                "Set up template with set_tpl or set_tpl_type first please."
            )

        self.wmc = WMConversion(self.wm_mode, self.wmt.wm_base)
        self.wm_max = wm_max

        wm_max_len = self.wmc.calc_max_wm_length(wm_max)
        if self.wm_flag_bit:
            self.wm_fixed_len = wm_max_len + 1
        else:
            self.wm_fixed_len = wm_max_len

    def set_text(self, text: str):
        """Set text string to watermark

        Args:
            text (str): text to be watermarked
        """
        self.text = text

    def set_text_file(self, path: str):
        """Set text string from read file.

        Args:
            path (str): Text file path

        Raises:
            ValueError: If file not found
            OSError: If file cannnot by read
        """
        path = os.path.abspath(path)

        if not os.path.exists(path):
            raise ValueError(f"ERROR: file {path} does not exist")

        try:
            with open(path, "r", encoding="utf-8") as file:
                self.set_text(file.read())
        except OSError as err:
            print(f"ERROR: cannot read file {path}, err is {err.strerror}")
            raise

    def insert_watermark(self, wm_str: str):
        """Insert watermark to text

        Args:
            wm_str (str): Watermark string

        Returns:
            (str): Watermarked text

        Raises:
            ValueError: If watermark string is larger than wm_max
        """
        wm_final = self.wmc.wm_convert_to_arbitrary_base(wm_str)
        if self.wm_flag_bit:
            wm_final = wm_final.zfill(self.wm_fixed_len - 1)
        else:
            wm_final = wm_final.zfill(self.wm_fixed_len)

        wm_final_max = self.wmc.wm_convert_to_arbitrary_base(self.wm_max)
        if self.wm_flag_bit:
            wm_final_max = wm_final_max.zfill(self.wm_fixed_len - 1)
        else:
            wm_final_max = wm_final_max.zfill(self.wm_fixed_len)

        if wm_final > wm_final_max:
            raise ValueError(
                f"ERROR: watermark:{wm_str} is larger than wm_max: {self.wm_max}"
            )

        if self.wm_flag_bit:
            wm_final = "1" + wm_final

        #
        wm_text = self.wmt.insert_watermark(
            text=self.text, wm_final=wm_final, start_at=self.start_at, loop=self.wm_loop
        )
        return wm_text

    def save_to_file(self, wm_text: str, path: str):
        """Save watermarked text string to a file

        Args:
            wm_text (str): text to be watermarked
            path (str): save to text file path

        Raises:
            OSError: If file cannnot be created

        """
        path = os.path.abspath(path)

        try:
            with open(path, "w", encoding="utf-8") as file:
                file.write(wm_text)
        except OSError as err:
            print(f"ERROR: cannot write to file {path}, err is {err.strerror}")
            raise

    def export_params(self):
        """Export watermark params to json string

        Returns:
            (str): Exported JSON string

        """
        if (
            self.tpl_type is not None
            and self.wmt.confusables_chars == self.tpl_type.value.CONFUSABLES_CHARS
        ):
            confusables_chars = []
        else:
            confusables_chars = self.wmt.confusables_chars

        params = {
            "tpl_type": self.tpl_type.name if self.tpl_type is not None else "",
            "confusables_chars": confusables_chars,
            "confusables_chars_key": self.wmt.confusables_chars_key,
            "wm_base": self.wmt.wm_base,
            "method": self.wmt.method.name,
            "wm_mode": self.wm_mode.name,
            "wm_len": self.wm_fixed_len,
            "wm_flag_bit": self.wm_flag_bit,
            "wm_loop": self.wm_loop,
            "wm_max": self.wm_max,
            "start_at": self.start_at,
            "version": __version__,
        }
        # print(params)

        return json.dumps(params, ensure_ascii=False)

    @staticmethod
    def check_if_enough_space(params: Any, text: str, dont_check_version: bool = False):
        """check if there if enough space to insert watermark to text

        Args:
            params (Any): praram string export by export_params
            text (str): text to be watermarked
            dont_check_version (bool, optional): if True then skip check version

        Returns:
            (bool): True or False
        """
        wm_init = TextWatermark.init_from_params(
            params=params, text=text, dont_check_version=dont_check_version
        )

        match wm_init.wmt.method.value:
            case WMMethod.FIND_AND_REPLACE:
                new_text = text[wm_init.start_at :]
                times = wm_init.wmt.check_find_and_replace_space(
                    new_text, wm_init.wm_fixed_len
                )
                if times < 1:
                    return False
            case WMMethod.INSERT_INTO_POSITION:
                if wm_init.start_at > len(text):
                    return False
            case WMMethod.APPEND_TO_CHAR | WMMethod.APPEND_AS_BINARY | WMMethod.DECORATE_EACH_CHAR:
                if wm_init.wm_fixed_len > (len(text) - wm_init.start_at):
                    return False

        return True

    @staticmethod
    def init_from_params(params: Any, text: str, dont_check_version: bool = False):
        """Import watermark params from json string

        Args:
            params (str): Exported JSON string
            text (str): Text to be watermarked

        Returns:
            (Watermark): Watermark object

        Raises:
            ValueError: If version mismatch

        """
        if isinstance(params, str):
            params = json.loads(params)

        if not dont_check_version and __version__ != params["version"]:
            raise ValueError(f'Version mismatch: {__version__}!= {params["version"]}')

        if params["wm_mode"] not in WMMode.__members__:
            raise ValueError(f'wm_mode {params["wm_mode"]} is not valid')

        if params["tpl_type"] and params["tpl_type"] not in WMTemplateType.__members__:
            raise ValueError(f'tpl_type {params["tpl_type"]} is not valid')

        if params["method"] not in WMMethod.__members__:
            raise ValueError(f'method {params["method"]} is not valid')

        wm_init = TextWatermark(
            wm_mode=WMMode[params["wm_mode"]],
            wm_base=params["wm_base"],
            start_at=params["start_at"],
            wm_loop=params["wm_loop"],
            wm_flag_bit=params["wm_flag_bit"],
        )
        if params["tpl_type"]:
            wm_init.set_tpl_type(tpl_type=WMTemplateType[params["tpl_type"]])
        else:
            wm_init.set_tpl(
                confusables_chars=params["confusables_chars"],
                method=WMMethod[params["method"]],
                confusables_chars_key=params["confusables_chars_key"],
            )

        wm_init.wm_fixed_len = params["wm_len"]
        wm_init.set_wm_max(params["wm_max"])
        wm_init.set_text(text=text)

        return wm_init

    @staticmethod
    def retrieve_watermark_from_bin(
        wm_bin: str, params: Any, dont_check_version: bool = False
    ):
        """
        Retrieve watermark from binary string.

        Args:
            wm_bin (str): watermark binary string
            params (json): params containing the watermark options

        Returns:
            (str): watermark string.

        """
        if isinstance(params, str):
            params = json.loads(params)

        wm_len = params["wm_len"]
        wmc = WMConversion(WMMode[params["wm_mode"]], params["wm_base"])

        ver = params["version"]
        if not dont_check_version and ver != __version__:
            raise ValueError(
                f"Not the same version, params version is {ver},"
                f" library version is {__version__}."
                "If you confirm that you want to use a different version to"
                " retrieve the watermark, please set dont_check_version to True"
            )

        if len(wm_bin) < wm_len:
            raise ValueError(f"Watermark length is short than {wm_len}")

        if params["wm_flag_bit"] is True:
            wm_temp = wm_bin[1:wm_len]
        else:
            wm_temp = wm_bin[0:wm_len]

        wm_temp = wm_temp.lstrip("0")

        wm_out_str = wmc.wm_restore_from_arbitrary_base(wm_temp)
        return wm_out_str

    @staticmethod
    def retrieve_watermark(wm_text: str, params: Any, dont_check_version: bool = False):
        """Retrieve watermark from watermarked text
        Note: This is a static method
            You can call this method by `TextWatermark.retrieve_watermark`

        Warning: Retrieve watermark method is not a silver bullet
            In many cases, the watermarked text may be transferred several times,
            which may cause the watermark to change or disappear. At this time,
            the watermark information cannot be retrieved through this function.
            In the case of screen capture and screen capture, manual judgment is
            required to find out the watermark information.

        Args:
            wm_text (str): watermarked text
            params (json): params containing the watermark options

        Returns:
            (str): Watermark string.

        """
        if isinstance(params, str):
            params = json.loads(params)
        # print(params)
        wm_len = params["wm_len"]
        ver = params["version"]
        if len(wm_text) < wm_len:
            raise ValueError(
                f"Watermark length is too short: "
                f"len of wm_text is {len(wm_text)}, wm_len is {wm_len}"
            )

        if not dont_check_version and ver != __version__:
            raise ValueError(
                f"Not the same version, params version is {ver},"
                f" library version is {__version__}."
                "If you confirm that you want to use a different version to"
                " retrieve the watermark, please set dont_check_version to True"
            )

        wmc = WMConversion(WMMode[params["wm_mode"]], params["wm_base"])

        if not params["confusables_chars"]:
            tpl_type = params["tpl_type"]
            if tpl_type not in WMTemplateType.__members__:
                raise ValueError(f"Invalid WMTemplateType: {tpl_type}")
            params["confusables_chars"] = WMTemplateType[
                tpl_type
            ].value.CONFUSABLES_CHARS

        wmt = WMTemplate(
            params["confusables_chars"],
            params["wm_base"],
            WMMethod[params["method"]],
            params["confusables_chars_key"],
        )

        wm_out = wmt.retrieve_watermark(
            wm_text=wm_text,
            wm_base=wmt.wm_base,
            wm_len=wm_len,
            start_at=params["start_at"],
        )

        # print((wm_out))
        if params["wm_flag_bit"] is True:
            wm_temp = wm_out[1:wm_len]
        else:
            wm_temp = wm_out[0:wm_len]

        wm_temp = wm_temp.lstrip("0")
        wm_out_str = wmc.wm_restore_from_arbitrary_base(wm_temp)
        return wm_out_str

text: str class-attribute

Text to be watermarked.

Set text by self.set_text(text) or self.set_text_file(file_path)

wmc: WMConversion class-attribute

Instance of WMConverison

wmt: WMTemplate class-attribute

Instance of WMTemplate

wm_max: str class-attribute

Maximum watermark string.

wm_fixed_len: int = 0 class-attribute

The longest (or largest) watermark string. When wm_flag_bit is true, The value of wm_max_len is 1 larger the actual value

tpl_type: Optional[WMTemplateType] = None class-attribute

template type in WMTemplateType

wm_mode = wm_mode instance-attribute

Watermark encoding mode. Setting different watermark modes will effectively reduce the size of the watermark string.

wm_base = wm_base instance-attribute

watermark convert base must be between 2 and 36, and less or euqal than len(list({template}.CONFUSABLES_CHARS.values())[0])

start_at = start_at instance-attribute

index where the watermark will be inserted. Defaults to 0.

wm_loop = wm_loop instance-attribute

If watermark will be repeated until it is inserted to end. Defaults to False.

wm_flag_bit = wm_flag_bit instance-attribute

If True, add a flag bit to watermark. Defaults to True.

__init__(wm_mode, wm_base=0, start_at=0, wm_loop=False, wm_flag_bit=True)

Class TextWatermark init

Parameters:

Name Type Description Default
wm_mode WMMode

WMMode to be set

required
wm_base int

Watermark convert base.

0
start_at int

index where the watermark will be inserted. Defaults to 0.

0
wm_loop bool

If watermark will be repeated until it is inserted to end. Defaults to False.

False
wm_flag_bit bool

If True, add a flag bit to watermark. Defaults to True.

True

Raises:

Type Description
ValueError

if base is not in the range [2, 36]

Source code in textwatermark/main.py
Python
def __init__(
    self,
    wm_mode: WMMode,
    wm_base: int = 0,
    start_at: int = 0,
    wm_loop: bool = False,
    wm_flag_bit: bool = True,
):
    """Class TextWatermark init

    Args:
        wm_mode (WMMode): WMMode to be set
        wm_base (int): Watermark convert base.
        start_at (int, optional): index where the watermark will be inserted. Defaults to 0.
        wm_loop (bool, optional): If watermark will be repeated until it is inserted to end.
            Defaults to False.
        wm_flag_bit (bool, optional): If True, add a flag bit to watermark. Defaults to True.

    Raises:
        ValueError: if `base` is not in the range [2, 36]
    """

    self.wm_mode = wm_mode
    """Watermark encoding mode. Setting different watermark modes will effectively
    reduce the size of the watermark string."""

    self.wm_base = wm_base
    """watermark convert base must be between 2 and 36,
        and less or euqal than len(list({template}.CONFUSABLES_CHARS.values())[0])
    """

    self.start_at = start_at
    """index where the watermark will be inserted. Defaults to 0."""
    self.wm_loop = wm_loop
    """If watermark will be repeated until it is inserted to end.
            Defaults to False."""
    self.wm_flag_bit = wm_flag_bit
    """If True, add a flag bit to watermark. Defaults to True."""

set_tpl(confusables_chars, method, confusables_chars_key='')

Set template arguments and get instance of WMTemplate

Parameters:

Name Type Description Default
confusables_chars Union[dict, list]

Confusables chars

required
method WMMethod

Method of watermark

required
confusables_chars_key str

Confusables chars key

''
Source code in textwatermark/main.py
Python
def set_tpl(
    self,
    confusables_chars: Union[dict, list],
    method: WMMethod,
    confusables_chars_key: str = "",
):
    """Set template arguments and get instance of WMTemplate

    Args:
        confusables_chars (Union[dict, list]): Confusables chars
        method (WMMethod): Method of watermark
        confusables_chars_key (str): Confusables chars key

    """

    self.wmt = WMTemplate(
        confusables_chars, self.wm_base, method, confusables_chars_key
    )

    if self.wm_base == 0:
        self.wm_base = self.wmt.wm_base

set_tpl_type(tpl_type, confusables_chars_key='')

Set template type

Parameters:

Name Type Description Default
tpl_type WMTemplateType

Template type

required
confusables_chars_key str

Confusables chars key

''
Source code in textwatermark/main.py
Python
def set_tpl_type(self, tpl_type: WMTemplateType, confusables_chars_key: str = ""):
    """Set template type

    Args:
        tpl_type (WMTemplateType): Template type
        confusables_chars_key (str): Confusables chars key

    """
    template = tpl_type.value

    self.wmt = WMTemplate(
        template.CONFUSABLES_CHARS,
        self.wm_base,
        template.method,
        confusables_chars_key,
    )

    if self.wm_base == 0:
        self.wm_base = self.wmt.wm_base

    self.tpl_type = tpl_type

set_wm_max(wm_max)

Set the length of the longest (or largest) watermark string.

Text Only
Since the input watermark string is variable in length,
this will bring trouble to the retrieval of the watermark:
it will be difficult for us to determine the boundary of the watermark,
or judge the loss of the watermark. Therefore, in this library,
we will calculate and set the length of the longest (or largest)
watermark string as the default length of the watermark string.

Parameters:

Name Type Description Default
wm_max str

The longest (or largest) watermark string.

required
Source code in textwatermark/main.py
Python
def set_wm_max(self, wm_max: str):
    """Set the length of the longest (or largest) watermark string.

        Since the input watermark string is variable in length,
        this will bring trouble to the retrieval of the watermark:
        it will be difficult for us to determine the boundary of the watermark,
        or judge the loss of the watermark. Therefore, in this library,
        we will calculate and set the length of the longest (or largest)
        watermark string as the default length of the watermark string.

    Args:
        wm_max (str): The longest (or largest) watermark string.

    """
    if not hasattr(self, "wmt") or not self.wmt:
        raise ValueError(
            "Set up template with set_tpl or set_tpl_type first please."
        )

    self.wmc = WMConversion(self.wm_mode, self.wmt.wm_base)
    self.wm_max = wm_max

    wm_max_len = self.wmc.calc_max_wm_length(wm_max)
    if self.wm_flag_bit:
        self.wm_fixed_len = wm_max_len + 1
    else:
        self.wm_fixed_len = wm_max_len

set_text(text)

Set text string to watermark

Parameters:

Name Type Description Default
text str

text to be watermarked

required
Source code in textwatermark/main.py
Python
def set_text(self, text: str):
    """Set text string to watermark

    Args:
        text (str): text to be watermarked
    """
    self.text = text

set_text_file(path)

Set text string from read file.

Parameters:

Name Type Description Default
path str

Text file path

required

Raises:

Type Description
ValueError

If file not found

OSError

If file cannnot by read

Source code in textwatermark/main.py
Python
def set_text_file(self, path: str):
    """Set text string from read file.

    Args:
        path (str): Text file path

    Raises:
        ValueError: If file not found
        OSError: If file cannnot by read
    """
    path = os.path.abspath(path)

    if not os.path.exists(path):
        raise ValueError(f"ERROR: file {path} does not exist")

    try:
        with open(path, "r", encoding="utf-8") as file:
            self.set_text(file.read())
    except OSError as err:
        print(f"ERROR: cannot read file {path}, err is {err.strerror}")
        raise

insert_watermark(wm_str)

Insert watermark to text

Parameters:

Name Type Description Default
wm_str str

Watermark string

required

Returns:

Type Description
str

Watermarked text

Raises:

Type Description
ValueError

If watermark string is larger than wm_max

Source code in textwatermark/main.py
Python
def insert_watermark(self, wm_str: str):
    """Insert watermark to text

    Args:
        wm_str (str): Watermark string

    Returns:
        (str): Watermarked text

    Raises:
        ValueError: If watermark string is larger than wm_max
    """
    wm_final = self.wmc.wm_convert_to_arbitrary_base(wm_str)
    if self.wm_flag_bit:
        wm_final = wm_final.zfill(self.wm_fixed_len - 1)
    else:
        wm_final = wm_final.zfill(self.wm_fixed_len)

    wm_final_max = self.wmc.wm_convert_to_arbitrary_base(self.wm_max)
    if self.wm_flag_bit:
        wm_final_max = wm_final_max.zfill(self.wm_fixed_len - 1)
    else:
        wm_final_max = wm_final_max.zfill(self.wm_fixed_len)

    if wm_final > wm_final_max:
        raise ValueError(
            f"ERROR: watermark:{wm_str} is larger than wm_max: {self.wm_max}"
        )

    if self.wm_flag_bit:
        wm_final = "1" + wm_final

    #
    wm_text = self.wmt.insert_watermark(
        text=self.text, wm_final=wm_final, start_at=self.start_at, loop=self.wm_loop
    )
    return wm_text

save_to_file(wm_text, path)

Save watermarked text string to a file

Parameters:

Name Type Description Default
wm_text str

text to be watermarked

required
path str

save to text file path

required

Raises:

Type Description
OSError

If file cannnot be created

Source code in textwatermark/main.py
Python
def save_to_file(self, wm_text: str, path: str):
    """Save watermarked text string to a file

    Args:
        wm_text (str): text to be watermarked
        path (str): save to text file path

    Raises:
        OSError: If file cannnot be created

    """
    path = os.path.abspath(path)

    try:
        with open(path, "w", encoding="utf-8") as file:
            file.write(wm_text)
    except OSError as err:
        print(f"ERROR: cannot write to file {path}, err is {err.strerror}")
        raise

export_params()

Export watermark params to json string

Returns:

Type Description
str

Exported JSON string

Source code in textwatermark/main.py
Python
def export_params(self):
    """Export watermark params to json string

    Returns:
        (str): Exported JSON string

    """
    if (
        self.tpl_type is not None
        and self.wmt.confusables_chars == self.tpl_type.value.CONFUSABLES_CHARS
    ):
        confusables_chars = []
    else:
        confusables_chars = self.wmt.confusables_chars

    params = {
        "tpl_type": self.tpl_type.name if self.tpl_type is not None else "",
        "confusables_chars": confusables_chars,
        "confusables_chars_key": self.wmt.confusables_chars_key,
        "wm_base": self.wmt.wm_base,
        "method": self.wmt.method.name,
        "wm_mode": self.wm_mode.name,
        "wm_len": self.wm_fixed_len,
        "wm_flag_bit": self.wm_flag_bit,
        "wm_loop": self.wm_loop,
        "wm_max": self.wm_max,
        "start_at": self.start_at,
        "version": __version__,
    }
    # print(params)

    return json.dumps(params, ensure_ascii=False)

check_if_enough_space(params, text, dont_check_version=False) staticmethod

check if there if enough space to insert watermark to text

Parameters:

Name Type Description Default
params Any

praram string export by export_params

required
text str

text to be watermarked

required
dont_check_version bool

if True then skip check version

False

Returns:

Type Description
bool

True or False

Source code in textwatermark/main.py
Python
@staticmethod
def check_if_enough_space(params: Any, text: str, dont_check_version: bool = False):
    """check if there if enough space to insert watermark to text

    Args:
        params (Any): praram string export by export_params
        text (str): text to be watermarked
        dont_check_version (bool, optional): if True then skip check version

    Returns:
        (bool): True or False
    """
    wm_init = TextWatermark.init_from_params(
        params=params, text=text, dont_check_version=dont_check_version
    )

    match wm_init.wmt.method.value:
        case WMMethod.FIND_AND_REPLACE:
            new_text = text[wm_init.start_at :]
            times = wm_init.wmt.check_find_and_replace_space(
                new_text, wm_init.wm_fixed_len
            )
            if times < 1:
                return False
        case WMMethod.INSERT_INTO_POSITION:
            if wm_init.start_at > len(text):
                return False
        case WMMethod.APPEND_TO_CHAR | WMMethod.APPEND_AS_BINARY | WMMethod.DECORATE_EACH_CHAR:
            if wm_init.wm_fixed_len > (len(text) - wm_init.start_at):
                return False

    return True

init_from_params(params, text, dont_check_version=False) staticmethod

Import watermark params from json string

Parameters:

Name Type Description Default
params str

Exported JSON string

required
text str

Text to be watermarked

required

Returns:

Type Description
Watermark

Watermark object

Raises:

Type Description
ValueError

If version mismatch

Source code in textwatermark/main.py
Python
@staticmethod
def init_from_params(params: Any, text: str, dont_check_version: bool = False):
    """Import watermark params from json string

    Args:
        params (str): Exported JSON string
        text (str): Text to be watermarked

    Returns:
        (Watermark): Watermark object

    Raises:
        ValueError: If version mismatch

    """
    if isinstance(params, str):
        params = json.loads(params)

    if not dont_check_version and __version__ != params["version"]:
        raise ValueError(f'Version mismatch: {__version__}!= {params["version"]}')

    if params["wm_mode"] not in WMMode.__members__:
        raise ValueError(f'wm_mode {params["wm_mode"]} is not valid')

    if params["tpl_type"] and params["tpl_type"] not in WMTemplateType.__members__:
        raise ValueError(f'tpl_type {params["tpl_type"]} is not valid')

    if params["method"] not in WMMethod.__members__:
        raise ValueError(f'method {params["method"]} is not valid')

    wm_init = TextWatermark(
        wm_mode=WMMode[params["wm_mode"]],
        wm_base=params["wm_base"],
        start_at=params["start_at"],
        wm_loop=params["wm_loop"],
        wm_flag_bit=params["wm_flag_bit"],
    )
    if params["tpl_type"]:
        wm_init.set_tpl_type(tpl_type=WMTemplateType[params["tpl_type"]])
    else:
        wm_init.set_tpl(
            confusables_chars=params["confusables_chars"],
            method=WMMethod[params["method"]],
            confusables_chars_key=params["confusables_chars_key"],
        )

    wm_init.wm_fixed_len = params["wm_len"]
    wm_init.set_wm_max(params["wm_max"])
    wm_init.set_text(text=text)

    return wm_init

retrieve_watermark_from_bin(wm_bin, params, dont_check_version=False) staticmethod

Retrieve watermark from binary string.

Parameters:

Name Type Description Default
wm_bin str

watermark binary string

required
params json

params containing the watermark options

required

Returns:

Type Description
str

watermark string.

Source code in textwatermark/main.py
Python
@staticmethod
def retrieve_watermark_from_bin(
    wm_bin: str, params: Any, dont_check_version: bool = False
):
    """
    Retrieve watermark from binary string.

    Args:
        wm_bin (str): watermark binary string
        params (json): params containing the watermark options

    Returns:
        (str): watermark string.

    """
    if isinstance(params, str):
        params = json.loads(params)

    wm_len = params["wm_len"]
    wmc = WMConversion(WMMode[params["wm_mode"]], params["wm_base"])

    ver = params["version"]
    if not dont_check_version and ver != __version__:
        raise ValueError(
            f"Not the same version, params version is {ver},"
            f" library version is {__version__}."
            "If you confirm that you want to use a different version to"
            " retrieve the watermark, please set dont_check_version to True"
        )

    if len(wm_bin) < wm_len:
        raise ValueError(f"Watermark length is short than {wm_len}")

    if params["wm_flag_bit"] is True:
        wm_temp = wm_bin[1:wm_len]
    else:
        wm_temp = wm_bin[0:wm_len]

    wm_temp = wm_temp.lstrip("0")

    wm_out_str = wmc.wm_restore_from_arbitrary_base(wm_temp)
    return wm_out_str

retrieve_watermark(wm_text, params, dont_check_version=False) staticmethod

Retrieve watermark from watermarked text

This is a static method

You can call this method by TextWatermark.retrieve_watermark

Retrieve watermark method is not a silver bullet

In many cases, the watermarked text may be transferred several times, which may cause the watermark to change or disappear. At this time, the watermark information cannot be retrieved through this function. In the case of screen capture and screen capture, manual judgment is required to find out the watermark information.

Parameters:

Name Type Description Default
wm_text str

watermarked text

required
params json

params containing the watermark options

required

Returns:

Type Description
str

Watermark string.

Source code in textwatermark/main.py
Python
@staticmethod
def retrieve_watermark(wm_text: str, params: Any, dont_check_version: bool = False):
    """Retrieve watermark from watermarked text
    Note: This is a static method
        You can call this method by `TextWatermark.retrieve_watermark`

    Warning: Retrieve watermark method is not a silver bullet
        In many cases, the watermarked text may be transferred several times,
        which may cause the watermark to change or disappear. At this time,
        the watermark information cannot be retrieved through this function.
        In the case of screen capture and screen capture, manual judgment is
        required to find out the watermark information.

    Args:
        wm_text (str): watermarked text
        params (json): params containing the watermark options

    Returns:
        (str): Watermark string.

    """
    if isinstance(params, str):
        params = json.loads(params)
    # print(params)
    wm_len = params["wm_len"]
    ver = params["version"]
    if len(wm_text) < wm_len:
        raise ValueError(
            f"Watermark length is too short: "
            f"len of wm_text is {len(wm_text)}, wm_len is {wm_len}"
        )

    if not dont_check_version and ver != __version__:
        raise ValueError(
            f"Not the same version, params version is {ver},"
            f" library version is {__version__}."
            "If you confirm that you want to use a different version to"
            " retrieve the watermark, please set dont_check_version to True"
        )

    wmc = WMConversion(WMMode[params["wm_mode"]], params["wm_base"])

    if not params["confusables_chars"]:
        tpl_type = params["tpl_type"]
        if tpl_type not in WMTemplateType.__members__:
            raise ValueError(f"Invalid WMTemplateType: {tpl_type}")
        params["confusables_chars"] = WMTemplateType[
            tpl_type
        ].value.CONFUSABLES_CHARS

    wmt = WMTemplate(
        params["confusables_chars"],
        params["wm_base"],
        WMMethod[params["method"]],
        params["confusables_chars_key"],
    )

    wm_out = wmt.retrieve_watermark(
        wm_text=wm_text,
        wm_base=wmt.wm_base,
        wm_len=wm_len,
        start_at=params["start_at"],
    )

    # print((wm_out))
    if params["wm_flag_bit"] is True:
        wm_temp = wm_out[1:wm_len]
    else:
        wm_temp = wm_out[0:wm_len]

    wm_temp = wm_temp.lstrip("0")
    wm_out_str = wmc.wm_restore_from_arbitrary_base(wm_temp)
    return wm_out_str