Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Zigzag persistence part1 #917

Merged
merged 69 commits into from
Oct 24, 2024
Merged
Show file tree
Hide file tree
Changes from 51 commits
Commits
Show all changes
69 commits
Select commit Hold shift + click to select a range
5407637
restore after merge
hschreiber Jul 17, 2023
f03a8bf
Merge remote-tracking branch 'origin/persistence_matrix' into zigzag_…
hschreiber Jul 18, 2023
3f8409f
doc and concepts
hschreiber Jul 21, 2023
2e5a20b
deleting files not wanted for the PR
hschreiber Jul 21, 2023
bfc5642
doc main page
hschreiber Jul 21, 2023
688b381
restore cmake files
hschreiber Jul 21, 2023
f7087fc
restore cmake files
hschreiber Jul 21, 2023
15fbc53
restore cmake files
hschreiber Jul 21, 2023
87ba652
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Aug 4, 2023
d8d07c5
simplex tree options correction
hschreiber Aug 21, 2023
0644ca6
sync with upstream Gudhi-devel
hschreiber Aug 21, 2023
394e33e
doc
hschreiber Aug 21, 2023
7178d45
first update merge
hschreiber Aug 21, 2023
5c4f9d7
second update merge
hschreiber Aug 21, 2023
2c79bc2
merge error fix
hschreiber Aug 21, 2023
f85ffb9
removal of useless simplex tree options
hschreiber Aug 21, 2023
902db3b
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Aug 22, 2023
da925a0
update from upstream master
hschreiber Apr 18, 2024
5208e5d
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Apr 18, 2024
4436039
update persistence matrix
hschreiber Apr 19, 2024
e384378
update persistence matrix
hschreiber Apr 19, 2024
0c7d3e6
generalized to faces instead of simplices
hschreiber May 31, 2024
ece1294
updates concepts
hschreiber May 31, 2024
20ae249
updates concepts
hschreiber May 31, 2024
df39f9c
zigzag doc update
hschreiber May 31, 2024
c192cd6
merge upstream update
hschreiber May 31, 2024
cadfb5c
merge Persistence_matrix update
hschreiber May 31, 2024
03e7b7d
small fixes
hschreiber May 31, 2024
26161a7
persistence diagram: list -> vector
hschreiber May 31, 2024
9c44409
removal of useless abs
hschreiber May 31, 2024
3fb6227
merge upstream
hschreiber Jun 18, 2024
2b46f13
split of zigzag computation and filtration values management
hschreiber Jun 20, 2024
dcc8bad
Merge remote-tracking branch 'origin/persistence_matrix' into zigzag_…
hschreiber Jun 20, 2024
dfdaaac
change of default values
hschreiber Jun 21, 2024
f7a431b
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Jun 27, 2024
8698036
Merge branch 'zigzag_persistence' of github.com:hschreiber/gudhi-deve…
hschreiber Jun 27, 2024
91dca3b
Merge remote-tracking branch 'origin/matrix_intervals' into zigzag_pe…
hschreiber Jun 28, 2024
e51a1f3
change of barcode + minor corrections
hschreiber Jun 28, 2024
585432c
update example cmake
hschreiber Jun 28, 2024
55d510a
rename files to gudhi convention
hschreiber Jul 8, 2024
0c332e3
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Jul 8, 2024
bf19ed5
fix version problem with boost::unordered_flat_map
hschreiber Jul 9, 2024
46a7d8a
Merge remote-tracking branch 'origin/master' into zigzag_persistence
hschreiber Jul 10, 2024
c42a263
corrections from merge with upstream
hschreiber Jul 10, 2024
efa2546
remove doxygen warnings
hschreiber Jul 11, 2024
cfdc161
dox fix
hschreiber Jul 11, 2024
e025cec
doc fix
hschreiber Jul 11, 2024
99f99ae
small changes
hschreiber Jul 18, 2024
6dc4219
Update src/Zigzag_persistence/include/gudhi/zigzag_persistence.h
hschreiber Jul 18, 2024
9446bad
Update src/Zigzag_persistence/include/gudhi/zigzag_persistence.h
hschreiber Jul 18, 2024
32e3053
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Jul 18, 2024
bbd3d90
doc
hschreiber Jul 19, 2024
c7edb91
replacing map_index_to_filtration_value->pair with get_filtration_val…
hschreiber Jul 19, 2024
3472160
doc
hschreiber Jul 19, 2024
744ac47
dim bug fix
hschreiber Jul 22, 2024
7d84dbf
doc
hschreiber Jul 22, 2024
564f751
doc
hschreiber Jul 22, 2024
bd6d9b5
doc
hschreiber Jul 23, 2024
4de6a26
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Sep 17, 2024
4f40d8b
upstream merge
hschreiber Sep 17, 2024
6a0fad2
renaming after Gudhi convention
hschreiber Sep 17, 2024
2b2a569
doc
hschreiber Sep 17, 2024
1e31a4b
doc
hschreiber Sep 18, 2024
54c00a6
Merge branch 'GUDHI:master' into zigzag_persistence
hschreiber Oct 21, 2024
5db9b3a
name change from 'face' to 'cell'
hschreiber Oct 21, 2024
915b5ea
removal of 'erase_birth_history'
hschreiber Oct 21, 2024
b3158e3
replacing set with vector for boundary translation
hschreiber Oct 22, 2024
5213928
Update src/common/doc/main_page.md
hschreiber Oct 24, 2024
f4c5345
Update src/Zigzag_persistence/doc/COPYRIGHT
hschreiber Oct 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ add_gudhi_module(Toplex_map)
add_gudhi_module(Witness_complex)
add_gudhi_module(Nerve_GIC)
add_gudhi_module(Persistence_matrix)
add_gudhi_module(Zigzag_persistence)

# Include module CMake subdirectories
# GUDHI_SUB_DIRECTORIES is managed in CMAKE_MODULE_PATH/GUDHI_modules.cmake
Expand Down
1 change: 1 addition & 0 deletions src/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ add_gudhi_module(Toplex_map)
add_gudhi_module(Witness_complex)
add_gudhi_module(Nerve_GIC)
add_gudhi_module(Persistence_matrix)
add_gudhi_module(Zigzag_persistence)

set(GUDHI_BIBLIO_DIR ${CMAKE_SOURCE_DIR})
# For "make doxygen" - Requires GUDHI_USER_VERSION_DIR to be set
Expand Down
79 changes: 79 additions & 0 deletions src/Zigzag_persistence/concept/ZigzagOptions.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT.
* See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details.
* Author(s): Hannah Schreiber
*
* Copyright (C) 2024 Inria
*
* Modification(s):
* - YYYY/MM Author: Description of the modification
*/

#ifndef CONCEPT_ZZ_OPTIONS_TYPE_H_
#define CONCEPT_ZZ_OPTIONS_TYPE_H_

/** @file ZigzagOptions.h
* @brief Contains @ref Gudhi::zigzag_persistence::ZigzagOptions and
* @ref Gudhi::zigzag_persistence::FilteredZigzagOptions concept.
*/

namespace Gudhi {
namespace zigzag_persistence {

/**
* @ingroup zigzag_persistence
*
* @brief List of options used for the filtered zigzag persistence computation.
*/
struct FilteredZigzagOptions {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would it make sense to derive from ZigzagOptions, as a way to say that this concept refines the other one?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, but I wanted to wait to see what I do with erase_birth_history and if there will not be other not-shared options.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like no non-shared option has popped up?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes. I was just suddenly not sure how clear it is to make one concept inherit the other in the documentation. It risk to be overlooked, no? But you are right. I will change that in the other branch with the oscillating rips.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can also add in the text that it refines the other concept. Yes, we'll see in the other branch what it looks like.

/**
* @brief Type for the face IDs used internally and other indexations. It must be signed.
*/
using internal_key = unspecified;

/**
* @brief Type for the face IDs used at insertion and in the boundaries given as argument.
hschreiber marked this conversation as resolved.
Show resolved Hide resolved
*/
using face_key = unspecified;

/**
* @brief Type for filtration values.
*/
using filtration_value = unspecified;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@VincentRouvreau would have to weigh in, but I think the type Filtration_value is capitalized in all the rest of Gudhi, so it looks a bit strange here.

Copy link
Collaborator Author

@hschreiber hschreiber Jul 19, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wanted to use lower case for all types which are or have high probability to be native types and upper case for classes/structs. But true, it makes the Filtration_value case weird. I don't have a very strong opinion about it, so I will let you both decide.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did you get inspiration from somewhere for this style? Since we haven't used that in the rest of Gudhi, I am a bit reluctant to introduce this inconsistency whose benefits are not really obvious to me.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't remember where I got it from, but I think it was from a comment of someone, as I didn't paid much attention to this detail before. So either it was from someone working on Gudhi, or from one of my coauthors.

Like I said, I can change that for Gudhi if you prefer. Is it only for Filtration_value or is there some unwritten rule about type nomenclature?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know specifically about Gudhi (although we seem to have mostly done it this way), but it is fairly common to capitalize all types, so you can tell immediately if something is a type or a variable (in a wide sense that also includes functions). You can wait for Vincent to come back, but I think I would prefer to stick to this convention. It also makes it unnecessary (although not forbidden) to end a type name in _t or _type.


/**
* @brief Type for the dimension values.
*/
using dimension_type = unspecified;

/**
* @brief Column type used by the internal matrix.
*/
static const Gudhi::persistence_matrix::Column_types column_type;
};

/**
* @ingroup zigzag_persistence
*
* @brief List of options used for the zigzag persistence computation.
*/
struct ZigzagOptions {
/**
* @brief Type for the face IDs used internally and other indexations. It must be signed.
*/
using internal_key = unspecified;

/**
* @brief Type for the dimension values.
*/
using dimension_type = unspecified;

/**
* @brief Column type used by the internal matrix.
*/
static const Gudhi::persistence_matrix::Column_types column_type;
};

} // namespace zigzag_persistence
} // namespace Gudhi

#endif // CONCEPT_ZZ_OPTIONS_TYPE_H_
12 changes: 12 additions & 0 deletions src/Zigzag_persistence/doc/COPYRIGHT
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
The files of this directory are part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@VincentRouvreau Can you remind me why we have this file in every doc/ subdirectory? Is it because there are images, where we cannot include the text? And having some global statement somewhere would have been complicated because of exceptions?

See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details.

Author(s): Vincent Rouvreau

Copyright (C) 2015 Inria
hschreiber marked this conversation as resolved.
Show resolved Hide resolved

This gives everyone the freedoms to use openFrameworks in any context:
commercial or non-commercial, public or private, open or closed source.

You should have received a copy of the MIT License along with this program.
If not, see https://opensource.org/licenses/MIT.
70 changes: 70 additions & 0 deletions src/Zigzag_persistence/doc/Intro_zigzag_persistence.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT.
* See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details.
* Author(s): Hannah Schreiber
*
* Copyright (C) 2023 Inria
*
* Modification(s):
* - YYYY/MM Author: Description of the modification
*/

#ifndef DOC_ZIGZAG_PERSISTENCE_INTRO_ZIGZAG_PERSISTENCE_H_
#define DOC_ZIGZAG_PERSISTENCE_INTRO_ZIGZAG_PERSISTENCE_H_

// needs namespace for Doxygen to link on classes
namespace Gudhi {
namespace zigzag_persistence {

/** \defgroup zigzag_persistence Zigzag Persistence
* @{
* \author Clément Maria, Hannah Schreiber
*
* \section zigzagintro Zigzag Persistence
*
* We refer to the introduction page \ref persistent_cohomology for persistent (co)homology for an introduction
* to the topic.
* Zigzag persistence is a generalization of the latter. While standard persistence only allows to grow the filtered
* complex by adding faces, zigzag persistence also allows removals. Hence the name "zigzag", as the module
* diagram will have arrows alternating between forward and backward.
*
* The module consists of the @ref Zigzag_persistence class and two wrappers @ref Filtered_zigzag_persistence and
* @ref Filtered_zigzag_persistence_with_storage "":
* - @ref Zigzag_persistence computes the persistence of a sequence of insertions and removals. A face can be inserted
* or removed one at a time and the returned persistence pairs / bars are indexed on the operation numbers.
* For example, if a cycle is born at operation number 6 and dies at operation number 7, it will output a bar starting
* at 6 and ending at 7.
* - @ref Filtered_zigzag_persistence and @ref Filtered_zigzag_persistence_with_storage are adding the notion of
* "filtration value" to @ref Zigzag_persistence. At each call, an operation can be associated to a filtration value,
* which will be used to index the returned bars instead (bars with new length 0 are then ignored). The two classes
* also have more flexible inputs (the boundaries do not have to be ordered, nor identified continuously
* from 0). The difference between both classes is on the way they manage the memory: @ref Filtered_zigzag_persistence
* removes systematically all unnecessary information and outputs a pair as soon it is closed, while
* @ref Filtered_zigzag_persistence_with_storage will store all informations about filtration values and bars until the
* end and output the pairs only when asked. Depending on the use and the length of the filtration, one will be more
* efficient than the other and vice versa.
*
* The implementation is based on the algorithm introduced in \cite zigzag.
*
hschreiber marked this conversation as resolved.
Show resolved Hide resolved
* \subsection zigzaginterface Stream-like interface
mglisse marked this conversation as resolved.
Show resolved Hide resolved
*
* As removals are possible in zigzag filtration, the maximal size of the complex does not depend on the length of the
* filtration anymore. This makes it possible to build very long fine tuned filtrations with relatively small complexes
* which can be processed without overreaching memory space. For this purpose, it is possible to feed the module with
* information about the filtration "on the fly" to avoid loading the whole filtration at once. Information about the
* current barcode can be retrieved between any steps via callback methods.
*
hschreiber marked this conversation as resolved.
Show resolved Hide resolved
* \subsection zigzagexamples Examples
*
* Here is a list of zigzag persistence examples :
* \li \gudhi_example_link{Zigzag_persistence,example_simple_zigzag_filtration.cpp} - A simple example to showcase how
* to use the @ref Filtered_zigzag_persistence_with_storage class.
*
* \li \gudhi_example_link{Zigzag_persistence,example_zzfiltration_from_file.cpp} - An example of a "stream-like" usage
* with @ref Filtered_zigzag_persistence by reading off the filtration from a file.
*
* @}
*/
} // namespace zigzag_persistence
} // namespace Gudhi

#endif // DOC_ZIGZAG_PERSISTENCE_INTRO_ZIGZAG_PERSISTENCE_H_
Binary file added src/Zigzag_persistence/doc/zigzag_ex.png
hschreiber marked this conversation as resolved.
Show resolved Hide resolved
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
11 changes: 11 additions & 0 deletions src/Zigzag_persistence/example/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
project(Zigzag_persistence_examples)

add_executable_with_targets(Zigzag_persistence_example_simple_zigzag_filtration example_simple_zigzag_filtration.cpp TBB::tbb)
add_test(NAME Zigzag_persistence_example_simple_zigzag_filtration COMMAND $<TARGET_FILE:Zigzag_persistence_example_simple_zigzag_filtration>)

add_executable_with_targets(Zigzag_persistence_example_zzfiltration_from_file example_zzfiltration_from_file.cpp TBB::tbb)
file(COPY "zigzag_filtration_example.txt" DESTINATION ${CMAKE_CURRENT_BINARY_DIR}/)
add_test(NAME Zigzag_persistence_example_zzfiltration_from_file COMMAND $<TARGET_FILE:Zigzag_persistence_example_zzfiltration_from_file> "${CMAKE_CURRENT_BINARY_DIR}/zigzag_filtration_example.txt")



140 changes: 140 additions & 0 deletions src/Zigzag_persistence/example/example_simple_zigzag_filtration.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT.
* See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details.
* Author(s): Hannah Schreiber
*
* Copyright (C) 2023 Inria
*
* Modification(s):
* - YYYY/MM Author: Description of the modification
*/

#include <iostream>
#include <vector>

#include <gudhi/filtered_zigzag_persistence.h>

using ZP = Gudhi::zigzag_persistence::Filtered_zigzag_persistence_with_storage<>;
using face_handle = ZP::face_key;
using filtration_value = ZP::filtration_value;
using Interval_filtration = ZP::Filtration_value_interval;

void print_barcode(ZP& zp) {
std::cout << std::endl << "Current barcode:" << std::endl;
for (Interval_filtration& bar : zp.get_persistence_diagram(0, true)) {
//stream out content of bar
std::cout << bar << std::endl;
//to access the content of the bar, it can either be used as a struct:
// bar.birth
// bar.death
// bar.dim
//or as a tuple
// std::get<0>(bar) <- birth
// std::get<1>(bar) <- death
// std::get<2>(bar) <- dim
}
}

void print_indices(ZP& zp) {
std::cout << std::endl << "Current pairs:" << std::endl;
for (auto& bar : zp.get_index_persistence_diagram()) {
//stream out content of bar
std::cout << bar << std::endl;
//to access the content of the bar, it can either be used as a struct:
// bar.birth
// bar.death
// bar.dim
//or as a tuple:
// std::get<0>(bar) <- birth
// std::get<1>(bar) <- death
// std::get<2>(bar) <- dim
}
}

std::vector<std::vector<face_handle> > get_boundaries() {
return {{},
{},
{},
{0, 1},
{0, 2},
{},
{1, 2},
{},
{5, 7},
{},
{3, 4, 6},
{7, 9},
{5, 9},
{8, 11, 12},
{10}, // remove
{13}, // remove
{1, 7},
{3, 4, 6},
{2, 7},
{8, 11, 12},
{0, 7},
{4, 18, 20},
{6, 16, 18},
{3, 16, 20},
{19}, // remove
{8}, // remove
{12}, // remove
{17, 21, 22, 23},
{27}}; // remove
}

std::vector<filtration_value> get_filtration_values() {
return {0, 0, 0,
1, 1, 1,
2, 2, 2,
3, 3, 3, 3,
4,
5,
6, 6, 6,
7, 7, 7, 7, 7, 7,
8,
9, 9, 9,
10};
}

std::vector<bool> get_directions() {
return {true, true, true, true, true, true, true, true, true, true, true, true, true, true,
false, false,
true, true, true, true, true, true, true, true,
false, false, false,
true,
false};
}

std::vector<unsigned int> get_batch_sizes() {
return {14, 2, 8, 3, 1, 1};
}

int main(int argc, char* const argv[]) {
std::clog << "********** Example **********" << std::endl;

ZP zp;

std::vector<std::vector<face_handle> > simplices = get_boundaries();
std::vector<filtration_value> fils = get_filtration_values();
std::vector<bool> dirs = get_directions();

for (unsigned int i = 0; i < simplices.size(); ++i) {
if (i > 0 && dirs[i] != dirs[i - 1]) {
print_barcode(zp);
print_indices(zp);
}
if (dirs[i]) {
int dim = simplices[i].size() == 0 ? 0 : simplices[i].size() - 1;
zp.insert_face(i, simplices[i], dim, fils[i]);
} else {
auto id = simplices[i][0];
int dim = simplices[id].size() == 0 ? 0 : simplices[id].size() - 1;
zp.remove_face(id, dim, fils[i]);
}
}

print_barcode(zp);
print_indices(zp);

return 0;
}
Loading
Loading